$array * @phpstan-assert-if-true array> $array */ public static function is_array_recursive_string(array $array): bool { foreach ($array as $value) { if (!is_string($value) && !(is_array($value) && self::is_array_recursive_string($value))) { return false; } } return true; } /** * @return array>> */ public function load(): array { $i18n = []; $dirs = new DirectoryIterator(I18N_PATH); foreach ($dirs as $dir) { if ($dir->isDot()) { continue; } $files = new DirectoryIterator($dir->getPathname()); foreach ($files as $file) { if (!$file->isFile()) { continue; } if ($file->getFilename() === 'plurals.php') { continue; } $i18n[$dir->getFilename()][$file->getFilename()] = $this->flatten($this->process($file->getPathname()), $file->getBasename('.php')); } } return $i18n; } /** * @param array>> $i18n */ public function dump(array $i18n): void { foreach ($i18n as $language => $file) { $dir = I18N_PATH . DIRECTORY_SEPARATOR . $language; if (!file_exists($dir)) { mkdir($dir, 0770, true); } foreach ($file as $name => $content) { $filename = $dir . DIRECTORY_SEPARATOR . $name; file_put_contents($filename, $this->format($content)); } } } /** * Process the content of an i18n file * @return array> */ private function process(string $filename): array { $fileContent = file_get_contents($filename); if (!is_string($fileContent)) { return []; } $content = str_replace(' todo\',', ' -> dirty\',', ' -> ignore\',', ], $content); try { $content = eval($content); } catch (ParseError $ex) { if (defined('STDERR')) { fwrite(STDERR, "Error while processing: $filename\n"); fwrite(STDERR, $ex->getMessage()); } die(1); } if (is_array($content) && self::is_array_recursive_string($content)) { return $content; } return []; } /** * Flatten an array of translation * * @param array|mixed> $translation * @return array */ private function flatten(array $translation, string $prefix = ''): array { $a = []; if ('' !== $prefix) { $prefix .= '.'; } foreach ($translation as $key => $value) { $key = (string)$key; if (is_array($value) && self::is_array_recursive_string($value)) { $a += $this->flatten($value, $prefix . $key); } elseif (is_string($value) || $value instanceof I18nValue) { $a[$prefix . $key] = new I18nValue($value); } } return $a; } /** * Unflatten an array of translation * * The first key is dropped since it represents the filename and we have * no use of it. * * @param array $translation * @return array */ private function unflatten(array $translation): array { $a = []; ksort($translation, SORT_NATURAL); foreach ($translation as $compoundKey => $value) { $keys = explode('.', $compoundKey); array_shift($keys); $current =& $a; $lastIndex = count($keys) - 1; foreach ($keys as $index => $key) { $normalisedKey = ctype_digit($key) ? (int)$key : $key; if ($index === $lastIndex) { $current[$normalisedKey] = $value->__toString(); continue; } if (!isset($current[$normalisedKey]) || !is_array($current[$normalisedKey])) { $current[$normalisedKey] = []; } $current =& $current[$normalisedKey]; } unset($current); } return $a; } /** * Format an array of translation * * It takes an array of translation and format it to be dumped in a * translation file. The array is first converted to a string then some * formatting regexes are applied to match the original content. * * @param array $translation */ private function format(array $translation): string { $translation = var_export($this->unflatten($translation), true); $patterns = [ '/ -> todo\',/', '/ -> dirty\',/', '/ -> ignore\',/', '/array \(/', '/=>\s*array/', '/(\w) {2}/', '/ {2}/', ]; $replacements = [ "',\t// TODO", // Double quoting is mandatory to have a tab instead of the \t string "',\t// DIRTY", // Double quoting is mandatory to have a tab instead of the \t string "',\t// IGNORE", // Double quoting is mandatory to have a tab instead of the \t string 'array(', '=> array', '$1 ', "\t", // Double quoting is mandatory to have a tab instead of the \t string ]; $translation = preg_replace($patterns, $replacements, $translation); return <<