Files
dbgate/common/translations-cli/program.js

131 lines
4.3 KiB
JavaScript

//@ts-check
const fs = require('fs');
const { program } = require('commander');
const {
resolveDirs,
resolveExtensions,
resolveFile,
ensureFileDirExists,
getTranslationChanges,
setLanguageTranslations,
getAllNonDefaultLanguages,
} = require('./helpers');
const { extractAllTranslations } = require('./extract');
const { getMissingTranslations } = require('./addMissing');
/**
* @typedef {{ extensions: string[], directories: string[], outputFile: string}} ExtractConfig
* @typedef {ExtractConfig & { verbose?: boolean }} ExtractOptions
*/
/** @type {ExtractConfig} */
const defaultConfig = {
extensions: ['.js', '.ts', '.svelte'],
directories: ['app', 'packages/web'],
outputFile: './translations/en-US.json',
};
program.name('dbgate-translations-cli').description('CLI tool for managing translation').version('1.0.0');
program
.command('extract')
.description('Extract translation keys from source files')
.option('-d, --directories <directories...>', 'directories to search', defaultConfig.directories)
.option('-e, --extensions <extensions...>', 'file extensions to process', defaultConfig.extensions)
.option('-o, --outputFile <file>', 'output file path', defaultConfig.outputFile)
.option('-v, --verbose', 'verbose mode')
.action(async (/** @type {ExtractOptions} */ options) => {
try {
const { directories, extensions, outputFile, verbose } = options;
const resolvedRirectories = resolveDirs(directories);
const resolvedExtensions = resolveExtensions(extensions);
const translations = await extractAllTranslations(resolvedRirectories, resolvedExtensions);
const resolvedOutputFile = resolveFile(outputFile);
ensureFileDirExists(resolvedOutputFile);
/** @type {Record<string, string>} */
let existingTranslations = {};
if (fs.existsSync(resolvedOutputFile)) {
existingTranslations = JSON.parse(fs.readFileSync(resolvedOutputFile, 'utf-8'));
}
const { added, removed, updated } = getTranslationChanges(existingTranslations, translations);
console.log('\nTranslation changes:');
console.log(`- Added: ${added.length} keys`);
console.log(`- Removed: ${removed.length} keys`);
console.log(`- Updated: ${updated.length} keys`);
console.log(`- Total: ${Object.keys(translations).length} keys`);
if (verbose) {
if (added.length > 0) {
console.log('\nNew keys:');
added.forEach(key => console.log(` + ${key}`));
}
if (removed.length > 0) {
console.log('\nRemoved keys:');
removed.forEach(key => console.log(` - ${key}`));
}
if (updated.length > 0) {
console.log('\nUpdated keys:');
updated.forEach(key => {
console.log(` ~ ${key}`);
console.log(` Old: ${existingTranslations[key]}`);
console.log(` New: ${translations[key]}`);
});
}
}
fs.writeFileSync(resolvedOutputFile, JSON.stringify(translations, null, 2));
} catch (error) {
console.error(error);
console.error('Error during extraction:', error.message);
process.exit(1);
}
});
const ALL_LANGUAGES = 'all';
/**
* @param {string} target
*/
function addMissingTranslations(target) {
console.log(`Adding missing keys for language: ${target}`);
const { result, stats } = getMissingTranslations(target);
console.log(`Added: ${stats.added}, Removed: ${stats.removed}, Total: ${stats.newLength}`);
setLanguageTranslations(target, result);
console.log(`New translations for ${target} were saved.`);
}
program
.command('add-missing')
.description('Add missing keys for a langauge to the translation file')
.option('-t, --target <target>', 'language to add missing translations to', ALL_LANGUAGES)
.action(options => {
try {
const { target } = options;
const languages = getAllNonDefaultLanguages();
if (target === ALL_LANGUAGES) {
console.log('Adding missing keys for all languages\n');
for (const language of languages) {
addMissingTranslations(language);
console.log();
}
} else {
addMissingTranslations(target);
}
} catch (error) {
console.error(error);
console.error('Error during add-missing:', error.message);
process.exit(1);
}
});
module.exports = { program };