1
0
Fork 0
metamask-extension/development/verify-locale-strings.js

286 lines
8.1 KiB
JavaScript
Executable File

#!/usr/bin/env node
// //////////////////////////////////////////////////////////////////////////////
//
// Locale verification script
//
// usage:
//
// node app/scripts/verify-locale-strings.js [<locale>] [--fix] [--quiet]
//
// This script will validate that locales have no unused messages. It will check
// the English locale against string literals found under `ui/`, and it will check
// other locales by comparing them to the English locale. It will also validate
// that non-English locales have all descriptions present in the English locale.
//
// A report will be printed to the console detailing any unused messages.
//
// The if the optional '--fix' argument is given, locales will be automatically
// updated to remove any unused messages.
//
// The optional '--quiet' argument reduces the verbosity of the output, printing
// just a single summary of results for each locale verified
//
// //////////////////////////////////////////////////////////////////////////////
const fs = require('fs');
const { promisify } = require('util');
const log = require('loglevel');
const glob = require('fast-glob');
const matchAll = require('string.prototype.matchall').getPolyfill();
const localeIndex = require('../app/_locales/index.json');
const {
compareLocalesForMissingDescriptions,
compareLocalesForMissingItems,
getLocale,
getLocalePath,
} = require('./lib/locales');
const readFile = promisify(fs.readFile);
const writeFile = promisify(fs.writeFile);
log.setDefaultLevel('info');
let fix = false;
let specifiedLocale;
for (const arg of process.argv.slice(2)) {
if (arg === '--fix') {
fix = true;
} else if (arg === '--quiet') {
log.setLevel('error');
} else {
specifiedLocale = arg;
}
}
main().catch((error) => {
log.error(error);
process.exit(1);
});
async function main() {
if (specifiedLocale) {
log.info(`Verifying selected locale "${specifiedLocale}":\n`);
const localeEntry = localeIndex.find(
(localeMeta) => localeMeta.code === specifiedLocale,
);
if (!localeEntry) {
throw new Error(`No localize entry found for ${specifiedLocale}`);
}
const failed =
specifiedLocale === 'en'
? await verifyEnglishLocale()
: await verifyLocale(specifiedLocale);
if (failed) {
process.exit(1);
} else {
console.log('No invalid entries!');
}
} else {
log.info('Verifying all locales:\n');
let failed = await verifyEnglishLocale(fix);
const localeCodes = localeIndex
.filter((localeMeta) => localeMeta.code !== 'en')
.map((localeMeta) => localeMeta.code);
for (const code of localeCodes) {
const localeFailed = await verifyLocale(code, fix);
failed = failed || localeFailed;
}
if (failed) {
process.exit(1);
} else {
console.log('No invalid entries!');
}
}
}
// eslint-disable-next-line consistent-return
async function writeLocale(code, locale) {
try {
const localeFilePath = getLocalePath(code);
return writeFile(
localeFilePath,
`${JSON.stringify(locale, null, 2)}\n`,
'utf8',
);
} catch (e) {
if (e.code === 'ENOENT') {
log.error('Locale file not found');
} else {
log.error(`Error writing your locale ("${code}") file: `, e);
}
process.exit(1);
}
}
async function verifyLocale(code) {
const englishLocale = await getLocale('en');
const targetLocale = await getLocale(code);
const extraItems = compareLocalesForMissingItems({
base: targetLocale,
subject: englishLocale,
});
if (extraItems.length) {
console.log(`**${code}**: ${extraItems.length} unused messages`);
log.info('Extra items that should not be localized:');
extraItems.forEach(function (key) {
log.info(` - [ ] ${key}`);
});
}
const missingDescriptions = compareLocalesForMissingDescriptions({
englishLocale,
targetLocale,
});
if (missingDescriptions.length) {
console.log(
`**${code}**: ${missingDescriptions.length} missing descriptions`,
);
log.info('Messages with missing descriptions:');
missingDescriptions.forEach(function (key) {
log.info(` - [ ] ${key}`);
});
}
if (extraItems.length > 0 || missingDescriptions.length > 0) {
if (fix) {
const newLocale = { ...targetLocale };
for (const item of extraItems) {
delete newLocale[item];
}
for (const message of Object.keys(englishLocale)) {
if (englishLocale[message].description && targetLocale[message]) {
targetLocale[message].description =
englishLocale[message].description;
}
}
await writeLocale(code, newLocale);
}
return true;
}
return false;
}
async function verifyEnglishLocale() {
const englishLocale = await getLocale('en');
// As time allows we'll switch to only performing the strict search.
// In the meantime we'll use glob to specify which paths can be strict searched
// and gradually phase out the key based search
const globsToStrictSearch = [
'ui/components/app/metamask-translation/*.js',
'ui/components/app/metamask-translation/*.ts',
'ui/pages/confirmation/templates/*.js',
'ui/pages/confirmation/templates/*.ts',
];
const testGlob = '**/*.test.js';
const javascriptFiles = await glob(
[
'ui/**/*.js',
'ui/**/*.ts',
'ui/**/*.tsx',
'shared/**/*.js',
'shared/**/*.ts',
'shared/**/*.tsx',
'app/scripts/constants/**/*.js',
'app/scripts/constants/**/*.ts',
'app/scripts/platforms/**/*.js',
],
{
ignore: [...globsToStrictSearch, testGlob],
},
);
const javascriptFilesToStrictSearch = await glob(globsToStrictSearch, {
ignore: [testGlob],
});
const strictSearchRegex =
/\bt\(\s*'(\w+)'\s*\)|\btranslationKey:\s*'(\w+)'/gu;
// match "t(`...`)" because constructing message keys from template strings
// prevents this script from finding the messages, and then inappropriately
// deletes them
const templateStringRegex = /\bt\(`.*`\)/gu;
const templateUsage = [];
// match the keys from the locale file
const keyRegex = /'(\w+)'|"(\w+)"/gu;
const usedMessages = new Set();
for await (const fileContents of getFileContents(javascriptFiles)) {
for (const match of matchAll.call(fileContents, keyRegex)) {
usedMessages.add(match[1] || match[2]);
}
const templateMatches = fileContents.match(templateStringRegex);
if (templateMatches) {
templateMatches.forEach((match) => templateUsage.push(match));
}
}
for await (const fileContents of getFileContents(
javascriptFilesToStrictSearch,
)) {
for (const match of matchAll.call(fileContents, strictSearchRegex)) {
usedMessages.add(match[1] || match[2] || match[3] || match[4]);
}
const templateMatches = fileContents.match(templateStringRegex);
if (templateMatches) {
templateMatches.forEach((match) => templateUsage.push(match));
}
}
// never consider these messages as unused
const messageExceptions = [
'appName',
'appNameBeta',
'appNameFlask',
'appNameMmi',
'appDescription',
];
const englishMessages = Object.keys(englishLocale);
const unusedMessages = englishMessages.filter(
(message) =>
!messageExceptions.includes(message) && !usedMessages.has(message),
);
if (unusedMessages.length) {
console.log(`**en**: ${unusedMessages.length} unused messages`);
log.info(`Messages not present in UI:`);
unusedMessages.forEach(function (key) {
log.info(` - [ ] ${key}`);
});
}
if (templateUsage.length) {
log.info(`Forbidden use of template strings in 't' function:`);
templateUsage.forEach(function (occurrence) {
log.info(` - ${occurrence}`);
});
}
if (!unusedMessages.length && !templateUsage.length) {
return false; // failed === false
}
if (unusedMessages.length > 0 && fix) {
const newLocale = { ...englishLocale };
for (const key of unusedMessages) {
delete newLocale[key];
}
await writeLocale('en', newLocale);
}
return true; // failed === true
}
async function* getFileContents(filenames) {
for (const filename of filenames) {
yield readFile(filename, 'utf8');
}
}