isDot() || in_array($app->getBasename(), $untranslatedApps, true)) { continue; } if (!file_exists($app->getPathname() . '/l10n')) { if (!str_contains($txConfig, '[o:nextcloud:p:nextcloud:r:' . $app->getBasename() . ']')) { $errors[] = $app->getBasename() . "\n" . ' App is not translation synced via transifex and also not marked as untranslated' . "\n"; } continue; } $directories[] = $app->getPathname() . '/l10n'; } foreach ($directories as $dir) { if (!file_exists($dir)) { continue; } $directory = new \DirectoryIterator($dir); foreach ($directory as $file) { if ($file->getExtension() !== 'json') { continue; } $content = file_get_contents($file->getPathname()); $language = pathinfo($file->getFilename(), PATHINFO_FILENAME); if (!in_array($language, $rtlLanguages, true) && preg_match('/[' . implode('', $rtlCharacters) . ']/u', $content)) { $errors[] = $file->getPathname() . "\n" . ' ' . 'Contains a RTL limited character in the translations.' . "\n"; } $json = json_decode($content, true); $translations = json_encode($json['translations']); if (strpos($translations, '|') !== false) { $errors[] = $file->getPathname() . "\n" . ' ' . 'Contains a | in the translations.' . "\n"; } if (json_last_error() !== JSON_ERROR_NONE) { $errors[] = $file->getPathname() . "\n" . ' ' . json_last_error_msg() . "\n"; } else { $valid++; } if ($isDebug && $file->getFilename() === 'en_GB.json') { $sourceStrings = json_encode(array_keys($json['translations'])); if (strpos($sourceStrings, '\u2019') !== false) { $errors[] = $file->getPathname() . "\n" . ' ' . 'Contains a unicode single quote "’" in the english source string, please replace with normal single quotes.' . "\n" . ' ' . 'Please note that this only updates after a sync to transifex.' . "\n"; } } } } if (count($errors) > 0) { echo "\033[0;31m" . sprintf('ERROR: There were %d errors:', count($errors)) . "\033[0m\n\n"; echo implode("\n", $errors); exit(1); } echo "\033[0;32m" . 'OK: ' . $valid . ' files parse' . "\033[0m\n"; exit(0);