@lingui/cli
Version:
Lingui CLI to extract messages, compile catalogs, and manage translation workflows
86 lines (85 loc) • 3.87 kB
JavaScript
import * as t from "@babel/types";
import { generate } from "@babel/generator";
import { compileMessageOrThrow, } from "@lingui/message-utils/compileMessage";
import pseudoLocalize from "./pseudoLocalize.js";
export function createCompiledCatalog(locale, messages, options) {
const { strict = false, namespace = "cjs", pseudoLocale, compilerBabelOptions = {}, outputPrefix = "/*eslint-disable*/", } = options;
const shouldPseudolocalize = locale === pseudoLocale;
const errors = [];
const compiledMessages = Object.keys(messages)
.sort()
.reduce((obj, key) => {
// Don't use `key` as a fallback translation in strict mode.
const translation = (messages[key] || (!strict ? key : ""));
try {
obj[key] = compile(translation, shouldPseudolocalize);
}
catch (e) {
errors.push({
id: key,
source: translation,
error: e,
});
}
return obj;
}, {});
if (namespace === "json") {
return { source: JSON.stringify({ messages: compiledMessages }), errors };
}
const ast = buildExportStatement(
//build JSON.parse(<compiledMessages>) statement
t.callExpression(t.memberExpression(t.identifier("JSON"), t.identifier("parse")), [t.stringLiteral(JSON.stringify(compiledMessages))]), namespace);
const code = generate(ast, {
minified: true,
jsescOption: {
minimal: true,
},
...compilerBabelOptions,
}).code;
return { source: `${outputPrefix}` + code, errors };
}
function buildExportStatement(expression, namespace) {
if (namespace === "ts") {
// import type { Messages } from "@lingui/core";
const importMessagesTypeDeclaration = t.importDeclaration([t.importSpecifier(t.identifier("Messages"), t.identifier("Messages"))], t.stringLiteral("@lingui/core"));
importMessagesTypeDeclaration.importKind = "type";
// Cast the expression to `Messages`
const castExpression = t.tsAsExpression(expression, t.tsTypeReference(t.identifier("Messages")));
// export const messages = ({ message: "Translation" } as Messages)
const exportDeclaration = t.exportNamedDeclaration(t.variableDeclaration("const", [
t.variableDeclarator(t.identifier("messages"), castExpression),
]));
return t.program([importMessagesTypeDeclaration, exportDeclaration]);
}
else if (namespace === "es") {
// export const messages = { message: "Translation" }
return t.exportNamedDeclaration(t.variableDeclaration("const", [
t.variableDeclarator(t.identifier("messages"), expression),
]));
}
else {
let exportExpression;
const matches = namespace.match(/^(window|global)\.([^.\s]+)$/);
if (namespace === "cjs") {
// module.exports.messages = { message: "Translation" }
exportExpression = t.memberExpression(t.identifier("module"), t.identifier("exports"));
}
else if (matches) {
// window.i18nMessages = { messages: { message: "Translation" }}
exportExpression = t.memberExpression(t.identifier(matches[1]), t.identifier(matches[2]));
}
else {
throw new Error(`Invalid namespace param: "${namespace}"`);
}
return t.expressionStatement(t.assignmentExpression("=", exportExpression, t.objectExpression([
t.objectProperty(t.identifier("messages"), expression),
])));
}
}
/**
* Compile string message into AST tree. Message format is parsed/compiled into
* JS arrays, which are handled in client.
*/
export function compile(message, shouldPseudolocalize = false) {
return compileMessageOrThrow(message, (value) => shouldPseudolocalize ? pseudoLocalize(value) : value);
}