208 lines
5.5 KiB
JavaScript
Executable file
208 lines
5.5 KiB
JavaScript
Executable file
#!/usr/bin/env node
|
|
|
|
const fs = require('fs/promises');
|
|
const path = require('path');
|
|
|
|
async function isDir(dirPath) {
|
|
if (!dirPath) {
|
|
return false;
|
|
}
|
|
try {
|
|
const stat = await fs.stat(dirPath);
|
|
return stat.isDirectory();
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This is the main function which starts multiple workers
|
|
* in order to speed up the po file => app.js
|
|
* locale conversions
|
|
*/
|
|
async function main({ localeRoot, outputDir } = {}) {
|
|
if (!(await isDir(localeRoot))) {
|
|
throw new Error(`Provided localeRoot: '${localeRoot}' doesn't seem to be a folder`);
|
|
}
|
|
|
|
if (!(await isDir(outputDir))) {
|
|
throw new Error(`Provided outputDir '${outputDir}' doesn't seem to be a folder`);
|
|
}
|
|
|
|
// eslint-disable-next-line global-require
|
|
const glob = require('glob');
|
|
// eslint-disable-next-line global-require
|
|
const { Worker } = require('jest-worker');
|
|
|
|
const locales = glob.sync('*/*.po', { cwd: localeRoot });
|
|
|
|
const worker = new Worker(__filename, {
|
|
exposedMethods: ['convertPoFileForLocale'],
|
|
silent: false,
|
|
enableWorkerThreads: true,
|
|
});
|
|
worker.getStdout().pipe(process.stdout);
|
|
worker.getStderr().pipe(process.stderr);
|
|
|
|
await Promise.all(
|
|
locales.map((localeFile) => {
|
|
const locale = path.dirname(localeFile);
|
|
return worker.convertPoFileForLocale({
|
|
locale,
|
|
localeFile: path.join(localeRoot, localeFile),
|
|
resultDir: path.join(outputDir, locale),
|
|
});
|
|
}),
|
|
);
|
|
|
|
await worker.end();
|
|
|
|
console.log('Done converting all the po files');
|
|
}
|
|
|
|
/**
|
|
* This is the conversion logic for: po => JS object for jed
|
|
*/
|
|
function convertPoToJed(data, locale) {
|
|
// eslint-disable-next-line global-require
|
|
const { parse } = require('gettext-parser/lib/poparser');
|
|
const DEFAULT_CONTEXT = '';
|
|
|
|
/**
|
|
* TODO: This replacer might be unnecessary _or_ even cause bugs.
|
|
* due to potential unnecessary double escaping.
|
|
* But for now it is here to ensure that the old and new output
|
|
* are equivalent.
|
|
* @param str
|
|
* @returns {string}
|
|
*/
|
|
function escapeMsgid(str) {
|
|
return `${str}`.replace(/([\\"])/g, '\\$1');
|
|
}
|
|
|
|
/**
|
|
* TODO: This replacer might be unnecessary _or_ even cause bugs.
|
|
* due to potential unnecessary double escaping.
|
|
* But for now it is here to ensure that the old and new output
|
|
* are equivalent.
|
|
*
|
|
* NOTE: The replacements of `\n` and `\t` need to be iterated on,
|
|
* because: In the cases where we see those chars, they:
|
|
* - likely need or could be trimmed because they do nothing
|
|
* - they seem to escaped in a way that is broken anyhow
|
|
* @param str
|
|
* @returns {string}
|
|
*/
|
|
function escapeMsgstr(str) {
|
|
return `${str}`.replace(/[\t\n"\\]/g, (match) => {
|
|
if (match === '\n') {
|
|
return '\\n';
|
|
}
|
|
if (match === '\t') {
|
|
return '\\t';
|
|
}
|
|
return `\\${match}`;
|
|
});
|
|
}
|
|
|
|
const { headers = {}, translations: parsed } = parse(data);
|
|
|
|
const translations = Object.values(parsed[DEFAULT_CONTEXT] ?? {}).reduce((acc, entry) => {
|
|
const { msgid, msgstr } = entry;
|
|
|
|
/* TODO: If a msgid has no translation, we can just drop the whole key,
|
|
as jed will just fallback to the keys
|
|
We are not doing that yet, because we do want to ensure that
|
|
the results of the new and old way of generating the files matches.
|
|
if (msgstr.every((x) => x === '')) {
|
|
return acc;
|
|
}
|
|
*/
|
|
|
|
acc[escapeMsgid(msgid)] = msgstr.map(escapeMsgstr);
|
|
|
|
return acc;
|
|
}, {});
|
|
|
|
// Do not bother if the file has no actual translations
|
|
if (!Object.keys(translations).length) {
|
|
return { jed: null };
|
|
}
|
|
|
|
if (headers['Plural-Forms']) {
|
|
headers.plural_forms = headers['Plural-Forms'];
|
|
}
|
|
|
|
// Format required for jed: http://messageformat.github.io/Jed/
|
|
const jed = {
|
|
domain: 'app',
|
|
locale_data: {
|
|
app: {
|
|
...translations,
|
|
// Ensure that the header data which is attached to a message with id ""
|
|
// is not accidentally overwritten by an empty externalized string
|
|
'': {
|
|
...headers,
|
|
domain: 'app',
|
|
lang: locale,
|
|
},
|
|
},
|
|
},
|
|
};
|
|
|
|
return { jed };
|
|
}
|
|
|
|
/**
|
|
* This is the function which the workers actually execute
|
|
* 1. It reads the po
|
|
* 2. converts it with convertPoToJed
|
|
* 3. writes the file to
|
|
*/
|
|
async function convertPoFileForLocale({ locale, localeFile, resultDir }) {
|
|
const poContent = await fs.readFile(localeFile);
|
|
|
|
const { jed } = await convertPoToJed(poContent, locale);
|
|
|
|
if (jed === null) {
|
|
console.log(`${locale}: No translations. Skipping creation of app.js`);
|
|
return;
|
|
}
|
|
|
|
await fs.mkdir(resultDir, { recursive: true });
|
|
|
|
await fs.writeFile(
|
|
path.join(resultDir, 'app.js'),
|
|
`window.translations = ${JSON.stringify(jed)}`,
|
|
'utf8',
|
|
);
|
|
console.log(`Created app.js in ${resultDir}`);
|
|
}
|
|
|
|
/*
|
|
Start the main thread only if we are not part of a worker
|
|
*/
|
|
if (!process.env.JEST_WORKER_ID) {
|
|
// eslint-disable-next-line global-require
|
|
const argumentsParser = require('commander');
|
|
|
|
const args = argumentsParser
|
|
.option('-l, --locale-root <locale_root>', 'Extract messages from subfolders in this directory')
|
|
.option('-o, --output-dir <output_dir>', 'Write app.js files into subfolders in this directory')
|
|
.parse(process.argv);
|
|
|
|
main(args).catch((e) => {
|
|
console.warn(`Something went wrong: ${e.message}`);
|
|
console.warn(args.printHelp());
|
|
process.exitCode = 1;
|
|
});
|
|
}
|
|
|
|
/*
|
|
Expose the function for workers
|
|
*/
|
|
module.exports = {
|
|
main,
|
|
convertPoToJed,
|
|
convertPoFileForLocale,
|
|
};
|