UNPKG

@lingui/cli

Version:

Lingui CLI to extract messages, compile catalogs, and manage translation workflows

86 lines (85 loc) 3.87 kB
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); }