mirror of
https://github.com/scratchfoundation/scratch-l10n.git
synced 2025-01-05 04:02:04 -05:00
50761ad970
* tx-pull-www: pull www translations from transifex * validate-www: check www for correct placeholders * add commands to pull and validate www to package.json * add validating www to the test step * add pulling www to the update-translations script for the cron job * changed .tx/config to match the new folder names `scratch-website.<resource-slug>`, and added missing folders. This isn’t needed for the scripts, but it’s useful if you need to pull the translation for a single language/resource
54 lines
2.2 KiB
JavaScript
54 lines
2.2 KiB
JavaScript
import assert from 'assert';
|
|
import parse from 'format-message-parse';
|
|
|
|
const flattenJson = (translations) => {
|
|
let messages = Object.keys(translations).reduce((collection, id) => {
|
|
collection[id] = translations[id].message;
|
|
return collection;
|
|
}, {});
|
|
return JSON.stringify(messages, null, 4);
|
|
};
|
|
|
|
// filter placeholders out of a message
|
|
// parse('a message with a {value} and {count, plural, one {one} other {more}}.')
|
|
// returns an array:
|
|
// [ 'a message with a ',
|
|
// [ 'value' ],
|
|
// ' and ',
|
|
// [ 'count', 'plural', 0, { one: [Array], other: [Array] } ],
|
|
// '.'
|
|
// ]
|
|
// placeholders are always an array, so filter for array elements to find the placeholders
|
|
const placeholders = message => (
|
|
// this will throw an error if the message is not valid ICU
|
|
// single quote (as in French l'année) messes up the parse and is not
|
|
// relevant for this check, so strip them out
|
|
parse(message.replace(/'/g, '')).filter(item => Array.isArray(item))
|
|
);
|
|
|
|
const validMessage = (message, source) => {
|
|
const transPlaceholders = placeholders(message);
|
|
const srcPlaceholders = placeholders(source);
|
|
// different number of placeholders
|
|
if (transPlaceholders.length !== srcPlaceholders.length) {
|
|
return false;
|
|
}
|
|
// TODO: Add checking to make sure placeholders in source have not been translated
|
|
return true;
|
|
};
|
|
|
|
const validateTranslations = (translation, source) => {
|
|
const locale = translation.locale;
|
|
const translations = translation.translations;
|
|
const transKeys = Object.keys(translations);
|
|
const sourceKeys = Object.keys(source);
|
|
assert.strictEqual(transKeys.length, sourceKeys.length, `locale ${locale} has a different number of message keys`);
|
|
transKeys.map(item => assert(sourceKeys.includes(item), `locale ${locale} has key ${item} not in the source`));
|
|
sourceKeys.map(item => assert(transKeys.includes(item), `locale ${locale} is missing key ${item}`));
|
|
sourceKeys.map(item => assert(
|
|
validMessage(translations[item], source[item]),
|
|
`locale ${locale}: "${translations[item]}" is not a valid translation for "${source[item]}"`)
|
|
);
|
|
};
|
|
|
|
export {flattenJson, validateTranslations, validMessage};
|