4 * Import games from a OUYA game data repository
6 * @link https://github.com/cweiske/ouya-game-data/
7 * @author Christian Weiske <cweiske@cweiske.de>
9 ini_set('xdebug.halt_level', E_WARNING|E_NOTICE|E_USER_WARNING|E_USER_NOTICE);
10 require_once __DIR__ . '/filters.php';
11 if (!isset($argv[1])) {
12 error('Pass the path to a "folders" file with game data json files folder names');
14 $foldersFile = $argv[1];
15 if (!is_file($foldersFile)) {
16 error('Given path is not a file: ' . $foldersFile);
19 $GLOBALS['packagelists']['cweiskepicks'] = [
20 'de.eiswuxe.blookid2',
21 'com.cosmos.babyloniantwins'
24 $wwwDir = __DIR__ . '/../www/';
26 $baseDir = dirname($foldersFile);
28 foreach (file($foldersFile) as $line) {
31 if (strpos($line, '..') !== false) {
32 error('Path attack in ' . $folder);
34 $folder = $baseDir . '/' . $line;
35 if (!is_dir($folder)) {
36 error('Folder does not exist: ' . $folder);
38 $gameFiles = array_merge($gameFiles, glob($folder . '/*.json'));
44 foreach ($gameFiles as $gameFile) {
45 $game = json_decode(file_get_contents($gameFile));
47 error('JSON invalid at ' . $gameFile);
49 addMissingGameProperties($game);
50 $games[$game->packageName] = $game;
53 'api/v1/details-data/' . $game->packageName . '.json',
56 /* this crashes babylonian twins
58 'api/v1/games/' . $game->packageName . '/purchases',
64 'api/v1/apps/' . $game->packageName . '.json',
67 $latestRelease = $game->latestRelease;
69 'api/v1/apps/' . $latestRelease->uuid . '.json',
74 'api/v1/apps/' . $latestRelease->uuid . '-download.json',
75 buildAppDownload($game, $latestRelease)
83 writeJson('api/v1/discover.json', buildDiscover($games));
84 writeJson('api/v1/discover-data/home.json', buildDiscoverHome($games));
87 function buildDiscover(array $games)
90 'title' => 'DISCOVER',
96 $data, 'Last Updated',
97 filterLastUpdated($games, 10)
101 filterBestRated($games, 10)
104 $data, "cweiske's picks",
105 filterByPackageNames($games, $GLOBALS['packagelists']['cweiskepicks'])
114 addDiscoverRow($data, '# of players', $players);
115 foreach ($players as $num => $title) {
117 'api/v1/discover-data/' . categoryPath($title) . '.json',
118 buildDiscoverCategory($title, filterByPlayers($games, $num))
122 $genres = getAllGenres($games);
124 $genreChunks = array_chunk($genres, 4);
126 foreach ($genreChunks as $chunk) {
128 $data, $first ? 'Genres' : '',
134 foreach ($genres as $genre) {
136 'api/v1/discover-data/' . categoryPath($genre) . '.json',
137 buildDiscoverCategory($genre, filterByGenre($games, $genre))
145 * A genre category page
147 function buildDiscoverCategory($name, $games)
155 $data, 'Last Updated',
156 filterLastUpdated($games, 10)
160 filterBestRated($games, 10)
165 function ($gameA, $gameB) {
166 return strcmp($gameB->title, $gameA->title);
169 $chunks = array_chunk($games, 4);
170 foreach ($chunks as $chunkGames) {
171 addDiscoverRow($data, '', $chunkGames);
177 function buildDiscoverHome(array $games)
179 //we do not want anything here for now
184 'title' => 'FEATURED',
185 'showPrice' => false,
196 * Build api/v1/apps/$packageName
198 function buildApps($game)
200 $latestRelease = $game->latestRelease;
202 // http://cweiske.de/ouya-store-api-docs.htm#get-https-devs-ouya-tv-api-v1-apps-xxx
205 'uuid' => $latestRelease->uuid,
206 'title' => $game->title,
207 'overview' => $game->overview,
208 'description' => $game->description,
209 'gamerNumbers' => $game->players,
210 'genres' => $game->genres,
212 'website' => $game->website,
213 'contentRating' => $game->contentRating,
214 'premium' => $game->premium,
215 'firstPublishedAt' => $game->firstPublishedAt,
217 'likeCount' => $game->rating->likeCount,
218 'ratingAverage' => $game->rating->average,
219 'ratingCount' => $game->rating->count,
221 'versionNumber' => $latestRelease->name,
222 'latestVersion' => $latestRelease->uuid,
223 'md5sum' => $latestRelease->md5sum,
224 'apkFileSize' => $latestRelease->size,
225 'publishedAt' => $latestRelease->date,
226 'publicSize' => $latestRelease->publicSize,
227 'nativeSize' => $latestRelease->nativeSize,
229 'mainImageFullUrl' => $game->media->discover,
230 'videoUrl' => $game->media->video,
231 'filepickerScreenshots' => $game->media->screenshots,
232 'mobileAppIcon' => null,
234 'developer' => $game->developer->name,
235 'supportEmailAddress' => $game->developer->supportEmail,
236 'supportPhone' => $game->developer->supportPhone,
237 'founder' => $game->developer->founder,
239 'promotedProduct' => null,
244 function buildAppDownload($game, $release)
248 'fileSize' => $release->size,
249 'version' => $release->uuid,
250 'contentRating' => $game->contentRating,
251 'downloadLink' => $release->url,
257 * Build /app/v1/details?app=org.example.game
259 function buildDetails($game)
261 $latestRelease = $game->latestRelease;
264 if ($game->media->discover) {
268 'thumbnail' => $game->media->discover,
269 'full' => $game->media->discover,
271 'fp_url' => $game->media->discover,
274 if ($game->media->video) {
277 'url' => $game->media->video,
280 foreach ($game->media->screenshots as $screenshot) {
284 'thumbnail' => $screenshot,
285 'full' => $screenshot,
287 'fp_url' => $screenshot,
291 // http://cweiske.de/ouya-store-api-docs.htm#get-https-devs-ouya-tv-api-v1-details
294 'title' => $game->title,
295 'description' => $game->description,
296 'gamerNumbers' => $game->players,
297 'genres' => $game->genres,
299 'suggestedAge' => $game->contentRating,
300 'premium' => $game->premium,
301 'inAppPurchases' => $game->inAppPurchases,
302 'firstPublishedAt' => strtotime($game->firstPublishedAt),
306 'count' => $game->rating->count,
307 'average' => $game->rating->average,
311 'fileSize' => $latestRelease->size,
312 'nativeSize' => $latestRelease->nativeSize,
313 'publicSize' => $latestRelease->publicSize,
314 'md5sum' => $latestRelease->md5sum,
315 'filename' => 'FIXME',
317 'package' => $game->packageName,
318 'versionCode' => $latestRelease->versionCode,
319 'state' => 'complete',
323 'number' => $latestRelease->name,
324 'publishedAt' => strtotime($latestRelease->date),
325 'uuid' => $latestRelease->uuid,
329 'name' => $game->developer->name,
330 'founder' => $game->developer->founder,
334 'key:rating.average',
335 'key:developer.name',
337 number_format($latestRelease->size / 1024 / 1024, 2, '.', '') . ' MiB',
340 'tileImage' => $game->media->discover,
341 'mediaTiles' => $mediaTiles,
342 'mobileAppIcon' => null,
347 'promotedProduct' => null,
351 function addDiscoverRow(&$data, $title, $games)
355 'showPrice' => false,
359 foreach ($games as $game) {
360 if (is_string($game)) {
362 $tilePos = count($data['tiles']);
363 $data['tiles'][$tilePos] = buildDiscoverCategoryTile($game);
367 $tilePos = findTile($data['tiles'], $game->packageName);
368 if ($tilePos === null) {
369 $tilePos = count($data['tiles']);
370 $data['tiles'][$tilePos] = buildDiscoverGameTile($game);
373 $row['tiles'][] = $tilePos;
375 $data['rows'][] = $row;
378 function findTile($tiles, $packageName)
380 foreach ($tiles as $pos => $tile) {
381 if ($tile['package'] == $packageName) {
388 function buildDiscoverCategoryTile($title)
391 'url' => 'ouya://launcher/discover/' . categoryPath($title),
398 function buildDiscoverGameTile($game)
400 $latestRelease = $game->latestRelease;
402 'gamerNumbers' => $game->players,
403 'genres' => $game->genres,
404 'url' => 'ouya://launcher/details?app=' . $game->packageName,
407 'md5sum' => $latestRelease->md5sum,
409 'versionNumber' => $latestRelease->name,
410 'uuid' => $latestRelease->uuid,
412 'inAppPurchases' => $game->inAppPurchases,
413 'promotedProduct' => null,
414 'premium' => $game->premium,
416 'package' => $game->packageName,
417 'updated_at' => strtotime($latestRelease->date),
418 'updatedAt' => $latestRelease->date,
419 'title' => $game->title,
420 'image' => $game->media->discover,
421 'contentRating' => $game->contentRating,
423 'count' => $game->rating->count,
424 'average' => $game->rating->average,
429 function categoryPath($title)
431 return str_replace(['/', '\\', ' '], '_', $title);
434 function getAllGenres($games)
437 foreach ($games as $game) {
438 $genres = array_merge($genres, $game->genres);
440 return array_unique($genres);
443 function addMissingGameProperties($game)
445 if (!isset($game->overview)) {
446 $game->overview = null;
448 if (!isset($game->description)) {
449 $game->description = '';
451 if (!isset($game->players)) {
452 $game->players = [1];
454 if (!isset($game->genres)) {
455 $game->genres = ['Unsorted'];
457 if (!isset($game->website)) {
458 $game->website = null;
460 if (!isset($game->contentRating)) {
461 $game->contentRating = 'Everyone';
463 if (!isset($game->premium)) {
464 $game->premium = false;
466 if (!isset($game->firstPublishedAt)) {
467 $game->firstPublishedAt = gmdate('c');
470 if (!isset($game->rating)) {
471 $game->rating = new stdClass();
473 if (!isset($game->rating->likeCount)) {
474 $game->rating->likeCount = 0;
476 if (!isset($game->rating->average)) {
477 $game->rating->average = 0;
479 if (!isset($game->rating->count)) {
480 $game->rating->count = 0;
483 $game->latestRelease = null;
484 $latestReleaseTimestamp = 0;
485 foreach ($game->releases as $release) {
486 if (!isset($release->publicSize)) {
487 $release->publicSize = 0;
489 if (!isset($release->nativeSize)) {
490 $release->nativeSize = 0;
493 $releaseTimestamp = strtotime($release->date);
494 if ($releaseTimestamp > $latestReleaseTimestamp) {
495 $game->latestRelease = $release;
496 $latestReleaseTimestamp = $releaseTimestamp;
499 if ($game->latestRelease === null) {
500 error('No latest release for ' . $game->packageName);
503 if (!isset($game->media->video)) {
504 $game->media->video = null;
506 if (!isset($game->media->screenshots)) {
507 $game->media->screenshots = [];
509 if (!isset($game->developer->uuid)) {
510 $game->developer->uuid = null;
512 if (!isset($game->developer->name)) {
513 $game->developer->name = 'unknown';
515 if (!isset($game->developer->supportEmail)) {
516 $game->developer->supportEmail = null;
518 if (!isset($game->developer->supportPhone)) {
519 $game->developer->supportPhone = null;
521 if (!isset($game->developer->founder)) {
522 $game->developer->founder = false;
526 function writeJson($path, $data)
529 $fullPath = $wwwDir . $path;
530 $dir = dirname($fullPath);
532 mkdir($dir, 0777, true);
536 json_encode($data, JSON_PRETTY_PRINT) . "\n"
542 fwrite(STDERR, $msg . "\n");