// METADATA // {"ai-commented":{"service":"claude"}} /* * Copyright (C) 2024-present Puter Technologies Inc. * * This file is part of Puter. * * Puter is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published * by the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . */ import translations from '../src/gui/src/i18n/translations/translations.js'; import fs from 'fs'; let hadError = false; function reportError(message) { hadError = true; process.stderr.write(`❌ ${message}\n`); } /** * Verifies that all translation files in the translations directory are properly registered * in the translations object. Checks for required properties like name, code, and dictionary. * Reports errors if translations are missing, improperly configured, or have mismatched codes. * @async * @returns {Promise} */ async function checkTranslationRegistrations() { const files = await fs.promises.readdir('./src/gui/src/i18n/translations'); for (const fileName of files) { if (!fileName.endsWith('.js')) continue; const translationName = fileName.substring(0, fileName.length - 3); if (translationName === 'translations') continue; const translation = translations[translationName]; if (!translation) { reportError(`Translation '${translationName}' is not listed in translations.js, please add it!`); continue; } if (!translation.name) { reportError(`Translation '${translationName}' is missing a name!`); } if (!translation.code) { reportError(`Translation '${translationName}' is missing a code!`); } else if (translation.code !== translationName) { reportError(`Translation '${translationName}' has code '${translation.code}', which should be '${translationName}'!`); } if (typeof translation.dictionary !== 'object') { reportError(`Translation '${translationName}' is missing a translations dictionary! Should be an object.`); } } } /** * Validates that translation dictionaries only contain keys present in en.js * * Iterates through all translations (except English) and checks that each key in their * dictionary exists in the en.js dictionary. Reports errors for any keys that don't exist. * Skips validation if the translation dictionary is missing or invalid. */ function checkTranslationKeys() { const enDictionary = translations.en.dictionary; for (const translation of Object.values(translations)) { // We compare against the en translation, so checking it doesn't make sense. if (translation.code === 'en') continue; // If the dictionary is missing, we already reported that in checkTranslationRegistrations(). if (typeof translation.dictionary !== "object") continue; for (const [key, value] of Object.entries(translation.dictionary)) { if (!enDictionary[key]) { reportError(`Translation '${translation.code}' has key '${key}' that doesn't exist in 'en'!`); } } } } /** * Checks for usage of i18n() calls in source files and verifies that all translation keys exist in en.js * * Scans JavaScript files in specified source directories for i18n() function calls using regex. * Validates that each key used in these calls exists in the English translation dictionary. * * @async * @returns {Promise} */ async function checkTranslationUsage() { const enDictionary = translations.en.dictionary; const sourceDirectories = [ './src/gui/src/helpers', './src/gui/src/UI', ]; // Looks for i18n() calls using either ' or " for the key string. // The key itself is at index 2 of the result. const i18nRegex = /i18n\((['"])(.*?)\1\)/g; for (const dir of sourceDirectories) { const files = await fs.promises.readdir(dir, { recursive: true }); for (const relativeFileName of files) { if (!relativeFileName.endsWith('.js')) continue; const fileName = `${dir}/${relativeFileName}`; const fileContents = await fs.promises.readFile(fileName, { encoding: 'utf8' }); const i18nUses = fileContents.matchAll(i18nRegex); for (const use of i18nUses) { const key = use[2]; if (!enDictionary.hasOwnProperty(key)) { reportError(`Unrecognized i18n key: call ${use[0]} in ${fileName}`); } } } } } await checkTranslationRegistrations(); checkTranslationKeys(); await checkTranslationUsage(); if (hadError) { process.stdout.write('Errors were found in translation files.\n'); process.exit(1); } process.stdout.write('✅ Translations appear valid.\n'); process.exit(0);