@vocab/core 1.7.5 → 1.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/README.md +11 -0
- package/dist/icu-handler.cjs +3 -3
- package/dist/icu-handler.cjs.map +1 -1
- package/dist/icu-handler.d.cts +1 -1
- package/dist/icu-handler.d.mts +1 -1
- package/dist/icu-handler.mjs +1 -2
- package/dist/icu-handler.mjs.map +1 -1
- package/dist/index.cjs +35 -21
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +1 -1
- package/dist/index.d.mts +1 -1
- package/dist/index.mjs +8 -14
- package/dist/index.mjs.map +1 -1
- package/dist/runtime.cjs +6 -6
- package/dist/runtime.cjs.map +1 -1
- package/dist/runtime.d.cts +1 -1
- package/dist/runtime.d.mts +1 -1
- package/dist/runtime.mjs +1 -2
- package/dist/runtime.mjs.map +1 -1
- package/dist/translation-file.cjs +2 -2
- package/dist/translation-file.d.cts +1 -1
- package/dist/translation-file.d.mts +1 -1
- package/dist/translation-file.mjs +1 -1
- package/dist/{types-C00NNtxU.d.cts → types-Bybu5Wvr.d.cts} +1 -1
- package/dist/{types-oikDld6a.d.mts → types-DW00hfr8.d.mts} +1 -1
- package/package.json +10 -10
- package/dist/rolldown-runtime-CHUWjh9U.mjs +0 -7
- package/dist/rolldown-runtime-y5NzCiFd.cjs +0 -34
- package/icu-handler/package.json +0 -5
- package/runtime/package.json +0 -5
- package/translation-file/package.json +0 -5
package/dist/README.md
CHANGED
|
@@ -732,6 +732,17 @@ This flag accepts an array of glob patterns to ignore.
|
|
|
732
732
|
vocab push --branch my-branch --ignore "**/dist/**" "**/another_ignored_directory/**"
|
|
733
733
|
```
|
|
734
734
|
|
|
735
|
+
#### Auto-Translation
|
|
736
|
+
|
|
737
|
+
By default, Phrase may not apply the project's automatic translation behaviour for new keys uploaded via API.
|
|
738
|
+
|
|
739
|
+
The `--auto-translate` flag instructs Phrase to automatically translate any missing keys using machine translation.. See [Phrase auto-translate API Documentation] for more information.
|
|
740
|
+
|
|
741
|
+
```sh
|
|
742
|
+
vocab push --branch my-branch --auto-translate
|
|
743
|
+
```
|
|
744
|
+
|
|
745
|
+
[Phrase auto-translate API Documentation]: https://developers.phrase.com/en/api/strings/uploads/upload-a-new-file#body-autotranslate
|
|
735
746
|
[phrase]: https://developers.phrase.com/api/
|
|
736
747
|
|
|
737
748
|
#### [Tags]
|
package/dist/icu-handler.cjs
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
2
|
+
require("./index.cjs");
|
|
2
3
|
let intl_messageformat = require("intl-messageformat");
|
|
3
|
-
|
|
4
4
|
//#region src/icu-handler.ts
|
|
5
5
|
const moduleCache = /* @__PURE__ */ new WeakMap();
|
|
6
6
|
const getParsedICUMessages = (m, locale) => {
|
|
@@ -17,7 +17,7 @@ const getParsedICUMessages = (m, locale) => {
|
|
|
17
17
|
});
|
|
18
18
|
return parsedICUMessages;
|
|
19
19
|
};
|
|
20
|
-
|
|
21
20
|
//#endregion
|
|
22
21
|
exports.getParsedICUMessages = getParsedICUMessages;
|
|
22
|
+
|
|
23
23
|
//# sourceMappingURL=icu-handler.cjs.map
|
package/dist/icu-handler.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"icu-handler.cjs","names":["
|
|
1
|
+
{"version":3,"file":"icu-handler.cjs","names":["IntlMessageFormat"],"sources":["../src/icu-handler.ts"],"sourcesContent":["import type { ParsedICUMessages, TranslationMessagesByKey } from './types';\nimport { IntlMessageFormat } from 'intl-messageformat';\n\ntype ICUMessagesByLocale = Record<string, ParsedICUMessages<any>>;\n\nconst moduleCache = new WeakMap<\n TranslationMessagesByKey,\n ICUMessagesByLocale\n>();\n\nexport const getParsedICUMessages = (\n m: TranslationMessagesByKey,\n locale: string,\n): ParsedICUMessages<any> => {\n const moduleCachedResult = moduleCache.get(m);\n\n if (moduleCachedResult && moduleCachedResult[locale]) {\n return moduleCachedResult[locale];\n }\n\n const parsedICUMessages: ParsedICUMessages<any> = {};\n\n for (const translation of Object.keys(m)) {\n const intlMessageFormat = new IntlMessageFormat(m[translation], locale);\n parsedICUMessages[translation] = {\n format: (params: any) => intlMessageFormat.format(params),\n };\n }\n\n moduleCache.set(m, { ...moduleCachedResult, [locale]: parsedICUMessages });\n\n return parsedICUMessages;\n};\n"],"mappings":";;;;AAKA,MAAM,8BAAc,IAAI,SAGrB;AAEH,MAAa,wBACX,GACA,WAC2B;CAC3B,MAAM,qBAAqB,YAAY,IAAI,EAAE;AAE7C,KAAI,sBAAsB,mBAAmB,QAC3C,QAAO,mBAAmB;CAG5B,MAAM,oBAA4C,EAAE;AAEpD,MAAK,MAAM,eAAe,OAAO,KAAK,EAAE,EAAE;EACxC,MAAM,oBAAoB,IAAIA,mBAAAA,kBAAkB,EAAE,cAAc,OAAO;AACvE,oBAAkB,eAAe,EAC/B,SAAS,WAAgB,kBAAkB,OAAO,OAAO,EAC1D;;AAGH,aAAY,IAAI,GAAG;EAAE,GAAG;GAAqB,SAAS;EAAmB,CAAC;AAE1E,QAAO"}
|
package/dist/icu-handler.d.cts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { l as ParsedICUMessages, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { l as ParsedICUMessages, y as TranslationMessagesByKey } from "./types-Bybu5Wvr.cjs";
|
|
2
2
|
|
|
3
3
|
//#region src/icu-handler.d.ts
|
|
4
4
|
declare const getParsedICUMessages: (m: TranslationMessagesByKey, locale: string) => ParsedICUMessages<any>;
|
package/dist/icu-handler.d.mts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { l as ParsedICUMessages, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { l as ParsedICUMessages, y as TranslationMessagesByKey } from "./types-DW00hfr8.mjs";
|
|
2
2
|
|
|
3
3
|
//#region src/icu-handler.d.ts
|
|
4
4
|
declare const getParsedICUMessages: (m: TranslationMessagesByKey, locale: string) => ParsedICUMessages<any>;
|
package/dist/icu-handler.mjs
CHANGED
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import { IntlMessageFormat } from "intl-messageformat";
|
|
2
|
-
|
|
3
2
|
//#region src/icu-handler.ts
|
|
4
3
|
const moduleCache = /* @__PURE__ */ new WeakMap();
|
|
5
4
|
const getParsedICUMessages = (m, locale) => {
|
|
@@ -16,7 +15,7 @@ const getParsedICUMessages = (m, locale) => {
|
|
|
16
15
|
});
|
|
17
16
|
return parsedICUMessages;
|
|
18
17
|
};
|
|
19
|
-
|
|
20
18
|
//#endregion
|
|
21
19
|
export { getParsedICUMessages };
|
|
20
|
+
|
|
22
21
|
//# sourceMappingURL=icu-handler.mjs.map
|
package/dist/icu-handler.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"icu-handler.mjs","names":[
|
|
1
|
+
{"version":3,"file":"icu-handler.mjs","names":[],"sources":["../src/icu-handler.ts"],"sourcesContent":["import type { ParsedICUMessages, TranslationMessagesByKey } from './types';\nimport { IntlMessageFormat } from 'intl-messageformat';\n\ntype ICUMessagesByLocale = Record<string, ParsedICUMessages<any>>;\n\nconst moduleCache = new WeakMap<\n TranslationMessagesByKey,\n ICUMessagesByLocale\n>();\n\nexport const getParsedICUMessages = (\n m: TranslationMessagesByKey,\n locale: string,\n): ParsedICUMessages<any> => {\n const moduleCachedResult = moduleCache.get(m);\n\n if (moduleCachedResult && moduleCachedResult[locale]) {\n return moduleCachedResult[locale];\n }\n\n const parsedICUMessages: ParsedICUMessages<any> = {};\n\n for (const translation of Object.keys(m)) {\n const intlMessageFormat = new IntlMessageFormat(m[translation], locale);\n parsedICUMessages[translation] = {\n format: (params: any) => intlMessageFormat.format(params),\n };\n }\n\n moduleCache.set(m, { ...moduleCachedResult, [locale]: parsedICUMessages });\n\n return parsedICUMessages;\n};\n"],"mappings":";;AAKA,MAAM,8BAAc,IAAI,SAGrB;AAEH,MAAa,wBACX,GACA,WAC2B;CAC3B,MAAM,qBAAqB,YAAY,IAAI,EAAE;AAE7C,KAAI,sBAAsB,mBAAmB,QAC3C,QAAO,mBAAmB;CAG5B,MAAM,oBAA4C,EAAE;AAEpD,MAAK,MAAM,eAAe,OAAO,KAAK,EAAE,EAAE;EACxC,MAAM,oBAAoB,IAAI,kBAAkB,EAAE,cAAc,OAAO;AACvE,oBAAkB,eAAe,EAC/B,SAAS,WAAgB,kBAAkB,OAAO,OAAO,EAC1D;;AAGH,aAAY,IAAI,GAAG;EAAE,GAAG;GAAqB,SAAS;EAAmB,CAAC;AAE1E,QAAO"}
|
package/dist/index.cjs
CHANGED
|
@@ -1,27 +1,47 @@
|
|
|
1
|
-
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
2
|
+
//#region \0rolldown/runtime.js
|
|
3
|
+
var __create = Object.create;
|
|
4
|
+
var __defProp = Object.defineProperty;
|
|
5
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
6
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
7
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
8
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
9
|
+
var __copyProps = (to, from, except, desc) => {
|
|
10
|
+
if (from && typeof from === "object" || typeof from === "function") for (var keys = __getOwnPropNames(from), i = 0, n = keys.length, key; i < n; i++) {
|
|
11
|
+
key = keys[i];
|
|
12
|
+
if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, {
|
|
13
|
+
get: ((k) => from[k]).bind(null, key),
|
|
14
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
return to;
|
|
18
|
+
};
|
|
19
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", {
|
|
20
|
+
value: mod,
|
|
21
|
+
enumerable: true
|
|
22
|
+
}) : target, mod));
|
|
23
|
+
//#endregion
|
|
2
24
|
let fs = require("fs");
|
|
3
25
|
let path = require("path");
|
|
4
|
-
path =
|
|
26
|
+
path = __toESM(path);
|
|
5
27
|
let picomatch = require("picomatch");
|
|
6
|
-
picomatch =
|
|
28
|
+
picomatch = __toESM(picomatch);
|
|
7
29
|
let _formatjs_icu_messageformat_parser = require("@formatjs/icu-messageformat-parser");
|
|
8
30
|
let prettier = require("prettier");
|
|
9
|
-
prettier =
|
|
31
|
+
prettier = __toESM(prettier);
|
|
10
32
|
let picocolors = require("picocolors");
|
|
11
|
-
picocolors =
|
|
33
|
+
picocolors = __toESM(picocolors);
|
|
12
34
|
let debug = require("debug");
|
|
13
|
-
debug =
|
|
35
|
+
debug = __toESM(debug);
|
|
14
36
|
let tinyglobby = require("tinyglobby");
|
|
15
37
|
let intl_messageformat = require("intl-messageformat");
|
|
16
38
|
let _formatjs_icu_messageformat_parser_printer_js = require("@formatjs/icu-messageformat-parser/printer.js");
|
|
17
39
|
let find_up = require("find-up");
|
|
18
|
-
find_up =
|
|
40
|
+
find_up = __toESM(find_up);
|
|
19
41
|
let fastest_validator = require("fastest-validator");
|
|
20
|
-
fastest_validator =
|
|
21
|
-
|
|
42
|
+
fastest_validator = __toESM(fastest_validator);
|
|
22
43
|
//#region src/logger.ts
|
|
23
44
|
const trace = (0, debug.default)(`vocab:core`);
|
|
24
|
-
|
|
25
45
|
//#endregion
|
|
26
46
|
//#region src/utils.ts
|
|
27
47
|
const defaultTranslationDirSuffix = ".vocab";
|
|
@@ -29,7 +49,7 @@ const devTranslationFileName = "translations.json";
|
|
|
29
49
|
const globAnyPathWithOptionalPrefix = "**/?(*)";
|
|
30
50
|
const compiledVocabFileFilter = /\.vocab[\\/]index\.(?:ts|js|cjs|mjs)$/;
|
|
31
51
|
function isDevLanguageFile(filePath) {
|
|
32
|
-
return filePath.endsWith(
|
|
52
|
+
return filePath.endsWith(`/translations.json`) || filePath === "translations.json";
|
|
33
53
|
}
|
|
34
54
|
function isAltLanguageFile(filePath) {
|
|
35
55
|
return filePath.endsWith(".translations.json");
|
|
@@ -86,7 +106,6 @@ function mapValues(obj, func) {
|
|
|
86
106
|
function getTranslationMessages(translations) {
|
|
87
107
|
return mapValues(translations, (v) => v.message);
|
|
88
108
|
}
|
|
89
|
-
|
|
90
109
|
//#endregion
|
|
91
110
|
//#region src/generate-language.ts
|
|
92
111
|
function generateLanguageFromTranslations({ baseTranslations, generator }) {
|
|
@@ -122,7 +141,6 @@ function transformMessageFormatElement(transformElement) {
|
|
|
122
141
|
return transformedMessageFormatElement;
|
|
123
142
|
};
|
|
124
143
|
}
|
|
125
|
-
|
|
126
144
|
//#endregion
|
|
127
145
|
//#region src/load-translations.ts
|
|
128
146
|
function getUniqueKey(key, namespace) {
|
|
@@ -310,7 +328,6 @@ async function loadAllTranslations({ fallbacks, includeNodeModules, withTags },
|
|
|
310
328
|
}
|
|
311
329
|
return loadedTranslations;
|
|
312
330
|
}
|
|
313
|
-
|
|
314
331
|
//#endregion
|
|
315
332
|
//#region src/compile.ts
|
|
316
333
|
function extractHasTags(ast) {
|
|
@@ -360,7 +377,7 @@ function serialiseObjectToType(v) {
|
|
|
360
377
|
}
|
|
361
378
|
const serializeTypeImports = (imports, moduleName) => {
|
|
362
379
|
if (imports.size === 0) return "";
|
|
363
|
-
return `import type { ${Array.from(imports).join(", ")} } from '${moduleName}';`;
|
|
380
|
+
return `import type { ${Array.from(imports).sort().join(", ")} } from '${moduleName}';`;
|
|
364
381
|
};
|
|
365
382
|
function serialiseTranslationRuntime(value, imports, loadedTranslation) {
|
|
366
383
|
trace("Serialising translations:", loadedTranslation);
|
|
@@ -380,9 +397,8 @@ function serialiseTranslationRuntime(value, imports, loadedTranslation) {
|
|
|
380
397
|
// This file is automatically generated by Vocab.
|
|
381
398
|
// To make changes update translation.json files directly.
|
|
382
399
|
|
|
383
|
-
${serializeTypeImports(imports, "@vocab/core")}
|
|
400
|
+
${serializeTypeImports(new Set([...imports, "TranslationFile"]), "@vocab/core")}
|
|
384
401
|
import { createLanguage, createTranslationFile } from '@vocab/core/runtime';
|
|
385
|
-
import type { TranslationFile } from '@vocab/core';
|
|
386
402
|
|
|
387
403
|
const translations: TranslationFile<
|
|
388
404
|
${languagesUnionAsString},
|
|
@@ -495,7 +511,6 @@ async function writeIfChanged(filepath, contents) {
|
|
|
495
511
|
} catch {}
|
|
496
512
|
if (hasChanged) await fs.promises.writeFile(filepath, contents, { encoding: "utf-8" });
|
|
497
513
|
}
|
|
498
|
-
|
|
499
514
|
//#endregion
|
|
500
515
|
//#region src/validate/index.ts
|
|
501
516
|
function findMissingKeys(loadedTranslation, devLanguageName, altLanguages) {
|
|
@@ -533,7 +548,6 @@ async function validate(config) {
|
|
|
533
548
|
}
|
|
534
549
|
return valid;
|
|
535
550
|
}
|
|
536
|
-
|
|
537
551
|
//#endregion
|
|
538
552
|
//#region src/ValidationError.ts
|
|
539
553
|
var ValidationError = class extends Error {
|
|
@@ -545,7 +559,6 @@ var ValidationError = class extends Error {
|
|
|
545
559
|
this.rawMessage = message;
|
|
546
560
|
}
|
|
547
561
|
};
|
|
548
|
-
|
|
549
562
|
//#endregion
|
|
550
563
|
//#region src/config.ts
|
|
551
564
|
const boldCyan = (s) => picocolors.default.bold(picocolors.default.cyan(s));
|
|
@@ -657,8 +670,8 @@ function resolveConfigSync(customConfigFilePath) {
|
|
|
657
670
|
trace("No configuration file found");
|
|
658
671
|
return null;
|
|
659
672
|
}
|
|
660
|
-
|
|
661
673
|
//#endregion
|
|
674
|
+
exports.__toESM = __toESM;
|
|
662
675
|
exports.compile = compile;
|
|
663
676
|
exports.compiledVocabFileFilter = compiledVocabFileFilter;
|
|
664
677
|
exports.getAltLanguageFilePath = getAltLanguageFilePath;
|
|
@@ -672,4 +685,5 @@ exports.resolveConfigSync = resolveConfigSync;
|
|
|
672
685
|
exports.validate = validate;
|
|
673
686
|
exports.validateConfig = validateConfig;
|
|
674
687
|
exports.watch = watch;
|
|
688
|
+
|
|
675
689
|
//# sourceMappingURL=index.cjs.map
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.cjs","names":["newObj: any","generatedTranslations: TranslationsByKey<string>","IntlMessageFormat","TYPE","transformedOptions: Record<string, PluralOrSelectOption>","newLanguage: TranslationsByKey","fallbackLanguageOrder: string[]","pc","validKeys: TranslationsByKey","languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n >","namespace: string","loadedTranslations: LoadedTranslation[]","translationsType: any","params: ICUParams","pm","result: Record<LanguageName, string[]>","pc","pc","Validator","foundLanguages: string[]","foundGeneratedLanguages: string[]","findUp"],"sources":["../src/logger.ts","../src/utils.ts","../src/generate-language.ts","../src/load-translations.ts","../src/compile.ts","../src/validate/index.ts","../src/ValidationError.ts","../src/config.ts"],"sourcesContent":["import pc from 'picocolors';\nimport debug from 'debug';\n\nexport const trace = debug(`vocab:core`);\n\nexport const log = (...params: unknown[]) => {\n // eslint-disable-next-line no-console\n console.log(pc.yellow('Vocab'), ...params);\n};\n","import path from 'path';\n\nimport type {\n LanguageName,\n LanguageTarget,\n TranslationsByKey,\n TranslationMessagesByKey,\n UserConfig,\n} from './types';\nimport { trace } from './logger';\n\nexport const defaultTranslationDirSuffix = '.vocab';\nexport const devTranslationFileName = 'translations.json';\n\nexport type Fallback = 'none' | 'valid' | 'all';\n\nconst globAnyPathWithOptionalPrefix = '**/?(*)';\n\nexport const compiledVocabFileFilter = /\\.vocab[\\\\/]index\\.(?:ts|js|cjs|mjs)$/;\n\nexport function isDevLanguageFile(filePath: string) {\n return (\n filePath.endsWith(`/${devTranslationFileName}`) ||\n filePath === devTranslationFileName\n );\n}\nexport function isAltLanguageFile(filePath: string) {\n return filePath.endsWith('.translations.json');\n}\nexport function isTranslationDirectory(\n filePath: string,\n {\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n }: {\n translationsDirectorySuffix?: string;\n },\n) {\n return filePath.endsWith(translationsDirectorySuffix);\n}\n\nexport function getDevTranslationFileGlob({\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n}: {\n translationsDirectorySuffix?: string;\n}) {\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${devTranslationFileName}`;\n\n trace('getDevTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltTranslationFileGlob(config: UserConfig) {\n const altLanguages = getAltLanguages(config);\n const langMatch =\n altLanguages.length === 1 ? altLanguages[0] : `{${altLanguages.join(',')}}`;\n\n const { translationsDirectorySuffix = defaultTranslationDirSuffix } = config;\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${langMatch}.${devTranslationFileName}`;\n\n trace('getAltTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltLanguages({\n devLanguage,\n languages,\n}: {\n devLanguage: LanguageName;\n languages: LanguageTarget[];\n}) {\n return languages.map((v) => v.name).filter((lang) => lang !== devLanguage);\n}\n\nexport function getDevLanguageFileFromTsFile(tsFilePath: string) {\n const directory = path.dirname(tsFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n\n trace(`Returning dev language path ${result} for path ${tsFilePath}`);\n return result;\n}\n\nexport function getDevLanguageFileFromAltLanguageFile(\n altLanguageFilePath: string,\n) {\n const directory = path.dirname(altLanguageFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n trace(\n `Returning dev language path ${result} for path ${altLanguageFilePath}`,\n );\n return result;\n}\n\nexport function getTSFileFromDevLanguageFile(devLanguageFilePath: string) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(path.join(directory, 'index.ts'));\n\n trace(`Returning TS path ${result} for path ${devLanguageFilePath}`);\n return result;\n}\n\nexport function getAltLanguageFilePath(\n devLanguageFilePath: string,\n language: string,\n) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(\n path.join(directory, `${language}.translations.json`),\n );\n trace(\n `Returning alt language path ${result} for path ${devLanguageFilePath}`,\n );\n return path.normalize(result);\n}\n\nexport function mapValues<Key extends string, OriginalValue, ReturnValue>(\n obj: Record<Key, OriginalValue>,\n func: (val: OriginalValue) => ReturnValue,\n): TranslationMessagesByKey<Key> {\n const newObj: any = {};\n const keys = Object.keys(obj) as Key[];\n for (const key of keys) {\n newObj[key] = func(obj[key]);\n }\n return newObj;\n}\n\nexport function getTranslationMessages<Key extends string>(\n translations: TranslationsByKey<Key>,\n): TranslationMessagesByKey<Key> {\n return mapValues(translations, (v) => v.message);\n}\n","import { IntlMessageFormat } from 'intl-messageformat';\nimport {\n type MessageFormatElement,\n type PluralOrSelectOption,\n TYPE,\n} from '@formatjs/icu-messageformat-parser';\nimport { printAST } from '@formatjs/icu-messageformat-parser/printer.js';\nimport type { MessageGenerator, TranslationsByKey } from './types';\n\nexport function generateLanguageFromTranslations({\n baseTranslations,\n generator,\n}: {\n baseTranslations: TranslationsByKey<string>;\n generator: MessageGenerator;\n}): TranslationsByKey<string> {\n if (!generator.transformElement && !generator.transformMessage) {\n return baseTranslations;\n }\n\n const translationKeys = Object.keys(baseTranslations);\n const generatedTranslations: TranslationsByKey<string> = {};\n\n for (const translationKey of translationKeys) {\n const translation = baseTranslations[translationKey];\n let transformedMessage = translation.message;\n\n if (generator.transformElement) {\n const messageAst = new IntlMessageFormat(translation.message).getAst();\n const transformedAst = messageAst.map(\n transformMessageFormatElement(generator.transformElement),\n );\n transformedMessage = printAST(transformedAst);\n }\n\n if (generator.transformMessage) {\n transformedMessage = generator.transformMessage(transformedMessage);\n }\n\n generatedTranslations[translationKey] = {\n message: transformedMessage,\n };\n }\n\n return generatedTranslations;\n}\n\nfunction transformMessageFormatElement(\n transformElement: (message: string) => string,\n): (messageFormatElement: MessageFormatElement) => MessageFormatElement {\n return (messageFormatElement: MessageFormatElement) => {\n const transformedMessageFormatElement = { ...messageFormatElement };\n\n switch (transformedMessageFormatElement.type) {\n case TYPE.literal:\n const transformedValue = transformElement(\n transformedMessageFormatElement.value,\n );\n transformedMessageFormatElement.value = transformedValue;\n break;\n\n case TYPE.select:\n case TYPE.plural:\n const transformedOptions: Record<string, PluralOrSelectOption> = {\n ...transformedMessageFormatElement.options,\n };\n\n for (const key of Object.keys(transformedOptions)) {\n transformedOptions[key].value = transformedOptions[key].value.map(\n transformMessageFormatElement(transformElement),\n );\n }\n\n break;\n\n case TYPE.tag:\n const transformedChildren =\n transformedMessageFormatElement.children.map(\n transformMessageFormatElement(transformElement),\n );\n transformedMessageFormatElement.children = transformedChildren;\n break;\n\n default:\n break;\n }\n\n return transformedMessageFormatElement;\n };\n}\n","import path from 'path';\n\nimport { glob } from 'tinyglobby';\nimport type {\n TranslationsByKey,\n UserConfig,\n LoadedTranslation,\n LanguageTarget,\n LanguageName,\n TranslationFileMetadata,\n TranslationFileContents,\n} from './types';\nimport pc from 'picocolors';\n\nimport { trace } from './logger';\nimport {\n defaultTranslationDirSuffix,\n type Fallback,\n getAltLanguageFilePath,\n getAltLanguages,\n getDevTranslationFileGlob,\n} from './utils';\nimport { generateLanguageFromTranslations } from './generate-language';\n\nexport function getUniqueKey(key: string, namespace: string) {\n return `${key}.${namespace}`;\n}\n\nexport function mergeWithDevLanguageTranslation({\n translation,\n devTranslation,\n}: {\n translation: TranslationsByKey;\n devTranslation: TranslationsByKey;\n}) {\n // Only use keys from the dev translation\n const keys = Object.keys(devTranslation);\n const newLanguage: TranslationsByKey = {};\n\n for (const key of keys) {\n if (translation[key]) {\n newLanguage[key] = {\n message: translation[key].message,\n description: devTranslation[key].description,\n };\n }\n }\n\n return newLanguage;\n}\n\nfunction getLanguageFallbacks({ languages }: { languages: LanguageTarget[] }) {\n const languageFallbackMap = new Map<LanguageName, LanguageName>();\n\n for (const lang of languages) {\n if (lang.extends) {\n languageFallbackMap.set(lang.name, lang.extends);\n }\n }\n\n return languageFallbackMap;\n}\n\nexport function getLanguageHierarchy({\n languages,\n}: {\n languages: LanguageTarget[];\n}) {\n const hierarchyMap = new Map<LanguageName, LanguageName[]>();\n const fallbacks = getLanguageFallbacks({ languages });\n\n for (const lang of languages) {\n const langHierarchy = [];\n let currLang = lang.extends;\n\n while (currLang) {\n langHierarchy.push(currLang);\n\n currLang = fallbacks.get(currLang);\n }\n\n hierarchyMap.set(lang.name, langHierarchy);\n }\n\n return hierarchyMap;\n}\n\nexport function getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n}: {\n languages: LanguageTarget[];\n languageName: string;\n devLanguage: string;\n fallbacks: Fallback;\n}) {\n const languageHierarchy = getLanguageHierarchy({ languages }).get(\n languageName,\n );\n\n if (!languageHierarchy) {\n throw new Error(`Missing language hierarchy for ${languageName}`);\n }\n\n const fallbackLanguageOrder: string[] = [languageName];\n\n if (fallbacks !== 'none') {\n fallbackLanguageOrder.unshift(...languageHierarchy.reverse());\n\n if (fallbacks === 'all' && fallbackLanguageOrder[0] !== devLanguage) {\n fallbackLanguageOrder.unshift(devLanguage);\n }\n }\n\n return fallbackLanguageOrder;\n}\n\nfunction getNamespaceByFilePath(\n relativePath: string,\n { translationsDirectorySuffix = defaultTranslationDirSuffix }: UserConfig,\n) {\n let namespace = path\n .dirname(relativePath)\n .replace(/^src\\//, '')\n .replace(/\\//g, '_');\n\n if (namespace.endsWith(translationsDirectorySuffix)) {\n namespace = namespace.slice(0, -translationsDirectorySuffix.length);\n }\n\n return namespace;\n}\n\nfunction printValidationError(...params: unknown[]) {\n // eslint-disable-next-line no-console\n console.error(pc.red('Error loading translation:'), ...params);\n}\n\nfunction getTranslationsFromFile(\n translationFileContents: unknown,\n {\n isAltLanguage,\n filePath,\n withTags,\n }: { isAltLanguage: boolean; filePath: string; withTags?: boolean },\n): {\n $namespace: unknown;\n keys: TranslationsByKey;\n metadata: TranslationFileMetadata;\n} {\n if (!translationFileContents || typeof translationFileContents !== 'object') {\n throw new Error(\n `Unable to read translation file ${filePath}. Translations must be an object.`,\n );\n }\n\n const { $namespace, _meta, ...keys } =\n translationFileContents as TranslationFileContents;\n\n if (isAltLanguage && $namespace) {\n printValidationError(\n `Found $namespace in alt language file in ${filePath}. $namespace is only used in the dev language and will be ignored.`,\n );\n }\n\n if (!isAltLanguage && $namespace && typeof $namespace !== 'string') {\n printValidationError(\n `Found non-string $namespace in language file in ${filePath}. $namespace must be a string.`,\n );\n }\n\n if (isAltLanguage && _meta?.tags) {\n printValidationError(\n `Found _meta.tags in alt language file in ${filePath}. _meta.tags is only used in the dev language and will be ignored.`,\n );\n }\n\n // Never return tags if we're fetching translations for an alt language\n const includeTags = !isAltLanguage && withTags;\n const validKeys: TranslationsByKey = {};\n\n for (const [translationKey, { tags, ...translation }] of Object.entries(\n keys,\n )) {\n if (typeof translation === 'string') {\n printValidationError(\n `Found string for a translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation) {\n printValidationError(\n `Found empty translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation.message || typeof translation.message !== 'string') {\n printValidationError(\n `No message found for translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n validKeys[translationKey] = {\n ...translation,\n tags: includeTags ? tags : undefined,\n };\n }\n\n const metadata = { tags: includeTags ? _meta?.tags : undefined };\n\n return { $namespace, keys: validKeys, metadata };\n}\n\nexport function loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation,\n fallbacks,\n }: {\n filePath: string;\n languageName: string;\n devTranslation: TranslationsByKey;\n fallbacks: Fallback;\n },\n { devLanguage, languages }: UserConfig,\n): TranslationsByKey {\n const altLanguageTranslation = {};\n\n const fallbackLanguageOrder = getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n });\n\n trace(\n `Loading alt language file with precedence: ${fallbackLanguageOrder\n .slice()\n .reverse()\n .join(' -> ')}`,\n );\n\n for (const fallbackLanguage of fallbackLanguageOrder) {\n if (fallbackLanguage !== devLanguage) {\n try {\n const altFilePath = getAltLanguageFilePath(filePath, fallbackLanguage);\n delete require.cache[altFilePath];\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationFile = require(altFilePath);\n const { keys: fallbackLanguageTranslation } = getTranslationsFromFile(\n translationFile,\n {\n filePath: altFilePath,\n isAltLanguage: true,\n },\n );\n Object.assign(\n altLanguageTranslation,\n mergeWithDevLanguageTranslation({\n translation: fallbackLanguageTranslation,\n devTranslation,\n }),\n );\n } catch {\n trace(`Missing alt language file ${getAltLanguageFilePath(\n filePath,\n fallbackLanguage,\n )}\n `);\n }\n } else {\n Object.assign(altLanguageTranslation, devTranslation);\n }\n }\n\n return altLanguageTranslation;\n}\n\nfunction stripTagsFromTranslations(translations: TranslationsByKey) {\n return Object.fromEntries(\n Object.entries(translations).map(([key, { tags, ...rest }]) => [key, rest]),\n );\n}\n\nexport function loadTranslation(\n {\n filePath,\n fallbacks,\n withTags,\n }: {\n filePath: string;\n fallbacks: Fallback;\n withTags?: boolean;\n },\n userConfig: UserConfig,\n): LoadedTranslation {\n trace(\n `Loading translation file in \"${fallbacks}\" fallback mode: \"${filePath}\"`,\n );\n\n const languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n > = {};\n\n delete require.cache[filePath];\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationContent = require(filePath);\n const relativePath = path.relative(\n userConfig.projectRoot || process.cwd(),\n filePath,\n );\n const {\n $namespace,\n keys: devTranslation,\n metadata,\n } = getTranslationsFromFile(translationContent, {\n filePath,\n isAltLanguage: false,\n withTags,\n });\n const namespace: string =\n typeof $namespace === 'string'\n ? $namespace\n : getNamespaceByFilePath(relativePath, userConfig);\n\n trace(`Found file ${filePath}. Using namespace ${namespace}`);\n\n languageSet[userConfig.devLanguage] = devTranslation;\n\n const devTranslationNoTags = withTags\n ? stripTagsFromTranslations(devTranslation)\n : devTranslation;\n const altLanguages = getAltLanguages(userConfig);\n for (const languageName of altLanguages) {\n languageSet[languageName] = loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation: devTranslationNoTags,\n fallbacks,\n },\n userConfig,\n );\n }\n\n for (const generatedLanguage of userConfig.generatedLanguages || []) {\n const { name: generatedLanguageName, generator } = generatedLanguage;\n const baseLanguage = generatedLanguage.extends || userConfig.devLanguage;\n const baseTranslations = languageSet[baseLanguage];\n\n languageSet[generatedLanguageName] = generateLanguageFromTranslations({\n baseTranslations,\n generator,\n });\n }\n\n return {\n filePath,\n keys: Object.keys(devTranslation),\n namespace,\n relativePath,\n languages: languageSet,\n metadata,\n };\n}\n\nexport async function loadAllTranslations(\n {\n fallbacks,\n includeNodeModules,\n withTags,\n }: { fallbacks: Fallback; includeNodeModules: boolean; withTags?: boolean },\n config: UserConfig,\n): Promise<LoadedTranslation[]> {\n const { projectRoot, ignore = [] } = config;\n\n const translationFiles = await glob(getDevTranslationFileGlob(config), {\n ignore: includeNodeModules ? ignore : [...ignore, '**/node_modules/**'],\n dot: true,\n absolute: true,\n cwd: projectRoot,\n expandDirectories: false,\n });\n\n trace(`Found ${translationFiles.length} translation files`);\n\n const loadedTranslations: LoadedTranslation[] = [];\n const keys = new Set<string>();\n\n for (const translationFile of translationFiles) {\n const loadedTranslation = loadTranslation(\n { filePath: translationFile, fallbacks, withTags },\n config,\n );\n\n loadedTranslations.push(loadedTranslation);\n\n for (const key of loadedTranslation.keys) {\n const uniqueKey = getUniqueKey(key, loadedTranslation.namespace);\n if (keys.has(uniqueKey)) {\n trace(`Duplicate keys found`);\n throw new Error(\n `Duplicate keys found. Key with namespace ${loadedTranslation.namespace} and key ${key} was found multiple times.`,\n );\n }\n keys.add(uniqueKey);\n\n const globalKey =\n loadedTranslation.languages[config.devLanguage][key].globalKey;\n if (globalKey) {\n if (keys.has(globalKey)) {\n throw new Error(\n `Duplicate keys found. Key with global key ${globalKey} and key ${key} was found multiple times`,\n );\n }\n keys.add(globalKey);\n }\n }\n }\n\n return loadedTranslations;\n}\n","import { promises as fs, existsSync } from 'fs';\nimport path from 'path';\nimport pm from 'picomatch';\n\nimport type { LoadedTranslation, UserConfig } from './types';\nimport {\n isArgumentElement,\n isDateElement,\n isNumberElement,\n isPluralElement,\n isSelectElement,\n isTagElement,\n isTimeElement,\n type MessageFormatElement,\n parse,\n} from '@formatjs/icu-messageformat-parser';\nimport prettier from 'prettier';\n\nimport {\n getTranslationMessages,\n getDevTranslationFileGlob,\n getTSFileFromDevLanguageFile,\n getDevLanguageFileFromAltLanguageFile,\n getAltTranslationFileGlob,\n isDevLanguageFile,\n isAltLanguageFile,\n devTranslationFileName,\n isTranslationDirectory,\n} from './utils';\nimport { trace } from './logger';\nimport { loadAllTranslations, loadTranslation } from './load-translations';\n\ntype ICUParams = Record<string, string>;\n\ninterface TranslationTypeInfo {\n params: ICUParams;\n message: string;\n hasTags: boolean;\n}\n\nfunction extractHasTags(ast: MessageFormatElement[]): boolean {\n return ast.some((element) => {\n if (isSelectElement(element) || isPluralElement(element)) {\n const children = Object.values(element.options).map((o) => o.value);\n return children.some((child) => extractHasTags(child));\n }\n\n return isTagElement(element);\n });\n}\n\nfunction extractParamTypes(\n ast: MessageFormatElement[],\n currentParams: ICUParams,\n): [params: ICUParams, vocabTypesImports: Set<string>] {\n let params = { ...currentParams };\n let vocabTypesImports = new Set<string>();\n\n for (const element of ast) {\n if (isArgumentElement(element)) {\n if (!(element.value in params)) {\n // Preserve existing types for parameters that we've already parsed\n // This applies to self-referential parameters, for example `{foo, plural, 1 {{foo} thing} other {{foo} things}}`\n params[element.value] = 'string';\n }\n } else if (isNumberElement(element)) {\n params[element.value] = 'number';\n } else if (isPluralElement(element)) {\n params[element.value] = 'number';\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n } else if (isDateElement(element) || isTimeElement(element)) {\n params[element.value] = 'Date | number';\n } else if (isTagElement(element)) {\n params[element.value] = 'FormatXMLElementFn<T>';\n vocabTypesImports.add('FormatXMLElementFn');\n\n const [newParams, subImports] = extractParamTypes(\n element.children,\n params,\n );\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n } else if (isSelectElement(element)) {\n const options = Object.keys(element.options);\n\n // `other` will always be an option as the parser enforces this by default\n const nonOtherOptions = options.filter((o) => o !== 'other');\n const nonOtherOptionsUnion = nonOtherOptions\n .map((o) => `'${o}'`)\n .join(' | ');\n\n params[element.value] = `StringWithSuggestions<${nonOtherOptionsUnion}>`;\n vocabTypesImports.add('StringWithSuggestions');\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n }\n }\n\n return [params, vocabTypesImports];\n}\n\nfunction serialiseObjectToType(v: any) {\n let result = '';\n\n for (const [key, value] of Object.entries(v)) {\n result += `${JSON.stringify(key)}: ${\n value && typeof value === 'object' ? serialiseObjectToType(value) : value\n },`;\n }\n\n return `{ ${result} }`;\n}\n\nconst serializeTypeImports = (imports: Set<string>, moduleName: string) => {\n if (imports.size === 0) {\n return '';\n }\n\n const importNames = Array.from(imports);\n return `import type { ${importNames.join(', ')} } from '${moduleName}';`;\n};\n\nfunction serialiseTranslationRuntime(\n value: Map<string, TranslationTypeInfo>,\n imports: Set<string>,\n loadedTranslation: LoadedTranslation,\n) {\n trace('Serialising translations:', loadedTranslation);\n const translationsType: any = {};\n\n for (const [key, { params, message, hasTags }] of value.entries()) {\n let translationFunctionString = `() => ${message}`;\n\n if (Object.keys(params).length > 0) {\n const formatGeneric = hasTags ? '<T = string>' : '';\n const formatReturn =\n hasTags && imports.has('FormatXMLElementFn')\n ? 'ReturnType<FormatXMLElementFn<T>>'\n : 'string';\n translationFunctionString = `${formatGeneric}(values: ${serialiseObjectToType(\n params,\n )}) => ${formatReturn}`;\n }\n\n translationsType[key] = translationFunctionString;\n }\n\n const languagesUnionAsString = Object.keys(loadedTranslation.languages)\n .map((l) => `'${l}'`)\n .join(' | ');\n const languageEntries = Object.entries(loadedTranslation.languages)\n .map(\n ([languageName, translations]) =>\n `${JSON.stringify(languageName)}: createLanguage(${JSON.stringify(\n getTranslationMessages(translations),\n )})`,\n )\n .join(',');\n\n return /* ts */ `\n // This file is automatically generated by Vocab.\n // To make changes update translation.json files directly.\n\n ${serializeTypeImports(imports, '@vocab/core')}\n import { createLanguage, createTranslationFile } from '@vocab/core/runtime';\n import type { TranslationFile } from '@vocab/core';\n\n const translations: TranslationFile<\n ${languagesUnionAsString},\n ${serialiseObjectToType(translationsType)}\n > = createTranslationFile\n ({\n ${languageEntries}\n });\n\n export default translations;\n `;\n}\n\nexport async function generateRuntime(loadedTranslation: LoadedTranslation) {\n const { languages: loadedLanguages, filePath } = loadedTranslation;\n\n trace('Generating types for', filePath);\n const translationTypes = new Map<string, TranslationTypeInfo>();\n\n let imports = new Set<string>();\n\n for (const key of loadedTranslation.keys) {\n let params: ICUParams = {};\n const messages = new Set();\n let hasTags = false;\n\n for (const translatedLanguage of Object.values(loadedLanguages)) {\n if (translatedLanguage[key]) {\n const ast = parse(translatedLanguage[key].message);\n\n hasTags = hasTags || extractHasTags(ast);\n\n const [parsedParams, vocabTypesImports] = extractParamTypes(\n ast,\n params,\n );\n\n imports = new Set([...imports, ...vocabTypesImports]);\n params = parsedParams;\n\n messages.add(JSON.stringify(translatedLanguage[key].message));\n }\n }\n\n translationTypes.set(key, {\n params,\n hasTags,\n message: Array.from(messages).join(' | '),\n });\n }\n\n const prettierConfig = await prettier.resolveConfig(filePath);\n const serializedTranslationType = serialiseTranslationRuntime(\n translationTypes,\n imports,\n loadedTranslation,\n );\n const declaration = await prettier.format(serializedTranslationType, {\n ...prettierConfig,\n parser: 'typescript',\n });\n const outputFilePath = getTSFileFromDevLanguageFile(filePath);\n trace(`Writing translation types to ${outputFilePath}`);\n await writeIfChanged(outputFilePath, declaration);\n}\n\nexport async function watch(config: UserConfig) {\n const cwd = config.projectRoot || process.cwd();\n\n const ignorePatterns = [\n ...(config.ignore || []),\n '**/node_modules/**',\n '**/.git/**',\n ];\n\n const ignoredFunction = (pathToCheck: string) =>\n pm.isMatch(pathToCheck, ignorePatterns);\n\n const chokidar = await import('chokidar');\n const watcher = chokidar.watch(cwd, {\n ignored: ignoredFunction,\n ignoreInitial: true,\n });\n\n const devFileGlob = getDevTranslationFileGlob(config);\n const altFileGlob = getAltTranslationFileGlob(config);\n\n const onTranslationChange = async (relativePath: string) => {\n const matchesTranslationFiles = pm.isMatch(relativePath, [\n devFileGlob,\n altFileGlob,\n ]);\n\n if (!matchesTranslationFiles) {\n return;\n }\n\n trace(`Detected change for file ${relativePath}`);\n\n let targetFile;\n\n if (isDevLanguageFile(relativePath)) {\n targetFile = path.resolve(cwd, relativePath);\n } else if (isAltLanguageFile(relativePath)) {\n targetFile = getDevLanguageFileFromAltLanguageFile(\n path.resolve(cwd, relativePath),\n );\n }\n\n if (targetFile) {\n watcher.add(targetFile);\n try {\n const loadedTranslation = loadTranslation(\n { filePath: targetFile, fallbacks: 'all' },\n config,\n );\n\n await generateRuntime(loadedTranslation);\n } catch (e) {\n // eslint-disable-next-line no-console\n console.log('Failed to generate types for', relativePath);\n // eslint-disable-next-line no-console\n console.error(e);\n }\n }\n };\n\n const onNewDirectory = async (relativePath: string) => {\n trace('Detected new directory', relativePath);\n\n if (!isTranslationDirectory(relativePath, config)) {\n trace('Ignoring non-translation directory:', relativePath);\n return;\n }\n\n const newFilePath = path.join(relativePath, devTranslationFileName);\n if (!existsSync(newFilePath)) {\n await fs.writeFile(newFilePath, JSON.stringify({}, null, 2));\n trace('Created new empty translation file:', newFilePath);\n } else {\n trace(\n `New directory already contains translation file. Skipping creation. Existing file ${newFilePath}`,\n );\n }\n };\n\n watcher.on('addDir', onNewDirectory);\n watcher.on('add', onTranslationChange).on('change', onTranslationChange);\n\n return () => watcher.close();\n}\n\nexport async function compile(\n { watch: shouldWatch = false } = {},\n config: UserConfig,\n) {\n const translations = await loadAllTranslations(\n { fallbacks: 'all', includeNodeModules: false },\n config,\n );\n\n for (const loadedTranslation of translations) {\n await generateRuntime(loadedTranslation);\n }\n\n if (shouldWatch) {\n trace('Listening for changes to files...');\n return watch(config);\n }\n}\n\nasync function writeIfChanged(filepath: string, contents: string) {\n let hasChanged = true;\n\n try {\n const existingContents = await fs.readFile(filepath, { encoding: 'utf-8' });\n\n hasChanged = existingContents !== contents;\n } catch {\n // ignore error, likely a file doesn't exist error so we want to write anyway\n }\n\n if (hasChanged) {\n await fs.writeFile(filepath, contents, { encoding: 'utf-8' });\n }\n}\n","/* eslint-disable no-console */\nimport type { UserConfig, LoadedTranslation, LanguageName } from '../types';\nimport pc from 'picocolors';\n\nimport { loadAllTranslations } from '../load-translations';\nimport { getAltLanguages } from '../utils';\n\nexport function findMissingKeys(\n loadedTranslation: LoadedTranslation,\n devLanguageName: LanguageName,\n altLanguages: LanguageName[],\n) {\n const devLanguage = loadedTranslation.languages[devLanguageName];\n\n if (!devLanguage) {\n throw new Error(\n `Failed to load dev language: ${loadedTranslation.filePath}`,\n );\n }\n\n const result: Record<LanguageName, string[]> = {};\n let valid = true;\n\n const requiredKeys = Object.keys(devLanguage);\n\n if (requiredKeys.length > 0) {\n for (const altLanguageName of altLanguages) {\n const altLanguage = loadedTranslation.languages[altLanguageName] ?? {};\n\n for (const key of requiredKeys) {\n if (typeof altLanguage[key]?.message !== 'string') {\n if (!result[altLanguageName]) {\n result[altLanguageName] = [];\n }\n\n result[altLanguageName].push(key);\n valid = false;\n }\n }\n }\n }\n return [valid, result] as const;\n}\n\nexport async function validate(config: UserConfig) {\n const allTranslations = await loadAllTranslations(\n { fallbacks: 'valid', includeNodeModules: true },\n config,\n );\n\n let valid = true;\n\n for (const loadedTranslation of allTranslations) {\n const [translationValid, result] = findMissingKeys(\n loadedTranslation,\n config.devLanguage,\n getAltLanguages(config),\n );\n\n if (!translationValid) {\n valid = false;\n console.log(\n pc.red(\n `Incomplete translations: \"${pc.bold(\n loadedTranslation.relativePath,\n )}\"`,\n ),\n );\n\n for (const lang of Object.keys(result)) {\n const missingKeys = result[lang];\n\n console.log(\n pc.yellow(lang),\n '->',\n missingKeys.map((v) => `\"${v}\"`).join(', '),\n );\n }\n }\n }\n\n return valid;\n}\n","export class ValidationError extends Error {\n code: string;\n rawMessage: string;\n constructor(code: string, message: string) {\n super(`Invalid vocab.config.js: ${code} - ${message}`);\n this.code = code;\n this.rawMessage = message;\n }\n}\n","import type { UserConfig } from './types';\nimport path from 'path';\n\nimport pc from 'picocolors';\nimport findUp from 'find-up';\nimport Validator from 'fastest-validator';\nimport { ValidationError } from './ValidationError';\nimport { trace } from './logger';\n\nconst boldCyan = (s: string) => pc.bold(pc.cyan(s));\n\nconst validator = new Validator();\nconst schema = {\n $$strict: true,\n devLanguage: {\n type: 'string',\n },\n languages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n },\n },\n },\n generatedLanguages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n generator: {\n type: 'object',\n props: {\n transformElement: { type: 'function', optional: true },\n transformMessage: { type: 'function', optional: true },\n },\n },\n },\n },\n optional: true,\n },\n translationsDirectorySuffix: { type: 'string', optional: true },\n projectRoot: { type: 'string', optional: true },\n ignore: {\n type: 'array',\n items: 'string',\n optional: true,\n },\n};\nconst checkConfigFile = validator.compile(schema);\n\nconst splitMap = (message: string, callback: (value: string) => string) =>\n message\n .split(' ,')\n .map((v) => callback(v))\n .join(' ,');\n\nexport function validateConfig(c: UserConfig) {\n trace('Validating configuration file');\n // Note: checkConfigFile mutates the config file by applying defaults\n const isValid = checkConfigFile(c);\n if (isValid !== true) {\n throw new ValidationError(\n 'InvalidStructure',\n (Array.isArray(isValid) ? isValid : [])\n .map((v) => {\n if (v.type === 'objectStrict') {\n return `Invalid key(s) ${splitMap(\n v.actual,\n (m) => `\"${pc.cyan(m)}\"`,\n )}. Expected one of ${splitMap(v.expected, pc.green)}`;\n }\n if (v.field) {\n return v.message?.replace(v.field, pc.cyan(v.field));\n }\n return v.message;\n })\n .join(' \\n'),\n );\n }\n\n const languageStrings = c.languages.map((v) => v.name);\n\n // Dev Language should exist in languages\n if (!languageStrings.includes(c.devLanguage)) {\n throw new ValidationError(\n 'InvalidDevLanguage',\n `The dev language \"${boldCyan(\n c.devLanguage,\n )}\" was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n\n const foundLanguages: string[] = [];\n for (const lang of c.languages) {\n // Languages must only exist once\n if (foundLanguages.includes(lang.name)) {\n throw new ValidationError(\n 'DuplicateLanguage',\n `The language \"${boldCyan(lang.name)}\" was defined multiple times.`,\n );\n }\n foundLanguages.push(lang.name);\n\n // Any extends must be in languages\n if (lang.extends && !languageStrings.includes(lang.extends)) {\n throw new ValidationError(\n 'InvalidExtends',\n `The language \"${boldCyan(lang.name)}\"'s extends of ${boldCyan(\n lang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n const foundGeneratedLanguages: string[] = [];\n for (const generatedLang of c.generatedLanguages || []) {\n // Generated languages must only exist once\n if (foundGeneratedLanguages.includes(generatedLang.name)) {\n throw new ValidationError(\n 'DuplicateGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" was defined multiple times.`,\n );\n }\n foundGeneratedLanguages.push(generatedLang.name);\n\n // Generated language names must not conflict with language names\n if (languageStrings.includes(generatedLang.name)) {\n throw new ValidationError(\n 'InvalidGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" is already defined as a language.`,\n );\n }\n\n // Any extends must be in languages\n if (\n generatedLang.extends &&\n !languageStrings.includes(generatedLang.extends)\n ) {\n throw new ValidationError(\n 'InvalidExtends',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\"'s extends of ${boldCyan(\n generatedLang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n trace('Configuration file is valid');\n\n return true;\n}\n\nfunction createConfig(configFilePath: string) {\n const cwd = path.dirname(configFilePath);\n\n return {\n projectRoot: cwd,\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n ...(require(configFilePath) as UserConfig),\n };\n}\n\nexport async function resolveConfig(\n customConfigFilePath?: string,\n): Promise<UserConfig | null> {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : await findUp(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n return null;\n}\n\nexport function resolveConfigSync(\n customConfigFilePath?: string,\n): UserConfig | null {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : findUp.sync(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n\n return null;\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.cjs","names":["IntlMessageFormat","TYPE","pc","pm","pc","pc","Validator","findUp"],"sources":["../src/logger.ts","../src/utils.ts","../src/generate-language.ts","../src/load-translations.ts","../src/compile.ts","../src/validate/index.ts","../src/ValidationError.ts","../src/config.ts"],"sourcesContent":["import pc from 'picocolors';\nimport debug from 'debug';\n\nexport const trace = debug(`vocab:core`);\n\nexport const log = (...params: unknown[]) => {\n // eslint-disable-next-line no-console\n console.log(pc.yellow('Vocab'), ...params);\n};\n","import path from 'path';\n\nimport type {\n LanguageName,\n LanguageTarget,\n TranslationsByKey,\n TranslationMessagesByKey,\n UserConfig,\n} from './types';\nimport { trace } from './logger';\n\nexport const defaultTranslationDirSuffix = '.vocab';\nexport const devTranslationFileName = 'translations.json';\n\nexport type Fallback = 'none' | 'valid' | 'all';\n\nconst globAnyPathWithOptionalPrefix = '**/?(*)';\n\nexport const compiledVocabFileFilter = /\\.vocab[\\\\/]index\\.(?:ts|js|cjs|mjs)$/;\n\nexport function isDevLanguageFile(filePath: string) {\n return (\n filePath.endsWith(`/${devTranslationFileName}`) ||\n filePath === devTranslationFileName\n );\n}\nexport function isAltLanguageFile(filePath: string) {\n return filePath.endsWith('.translations.json');\n}\nexport function isTranslationDirectory(\n filePath: string,\n {\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n }: {\n translationsDirectorySuffix?: string;\n },\n) {\n return filePath.endsWith(translationsDirectorySuffix);\n}\n\nexport function getDevTranslationFileGlob({\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n}: {\n translationsDirectorySuffix?: string;\n}) {\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${devTranslationFileName}`;\n\n trace('getDevTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltTranslationFileGlob(config: UserConfig) {\n const altLanguages = getAltLanguages(config);\n const langMatch =\n altLanguages.length === 1 ? altLanguages[0] : `{${altLanguages.join(',')}}`;\n\n const { translationsDirectorySuffix = defaultTranslationDirSuffix } = config;\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${langMatch}.${devTranslationFileName}`;\n\n trace('getAltTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltLanguages({\n devLanguage,\n languages,\n}: {\n devLanguage: LanguageName;\n languages: LanguageTarget[];\n}) {\n return languages.map((v) => v.name).filter((lang) => lang !== devLanguage);\n}\n\nexport function getDevLanguageFileFromTsFile(tsFilePath: string) {\n const directory = path.dirname(tsFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n\n trace(`Returning dev language path ${result} for path ${tsFilePath}`);\n return result;\n}\n\nexport function getDevLanguageFileFromAltLanguageFile(\n altLanguageFilePath: string,\n) {\n const directory = path.dirname(altLanguageFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n trace(\n `Returning dev language path ${result} for path ${altLanguageFilePath}`,\n );\n return result;\n}\n\nexport function getTSFileFromDevLanguageFile(devLanguageFilePath: string) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(path.join(directory, 'index.ts'));\n\n trace(`Returning TS path ${result} for path ${devLanguageFilePath}`);\n return result;\n}\n\nexport function getAltLanguageFilePath(\n devLanguageFilePath: string,\n language: string,\n) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(\n path.join(directory, `${language}.translations.json`),\n );\n trace(\n `Returning alt language path ${result} for path ${devLanguageFilePath}`,\n );\n return path.normalize(result);\n}\n\nexport function mapValues<Key extends string, OriginalValue, ReturnValue>(\n obj: Record<Key, OriginalValue>,\n func: (val: OriginalValue) => ReturnValue,\n): TranslationMessagesByKey<Key> {\n const newObj: any = {};\n const keys = Object.keys(obj) as Key[];\n for (const key of keys) {\n newObj[key] = func(obj[key]);\n }\n return newObj;\n}\n\nexport function getTranslationMessages<Key extends string>(\n translations: TranslationsByKey<Key>,\n): TranslationMessagesByKey<Key> {\n return mapValues(translations, (v) => v.message);\n}\n","import { IntlMessageFormat } from 'intl-messageformat';\nimport {\n type MessageFormatElement,\n type PluralOrSelectOption,\n TYPE,\n} from '@formatjs/icu-messageformat-parser';\nimport { printAST } from '@formatjs/icu-messageformat-parser/printer.js';\nimport type { MessageGenerator, TranslationsByKey } from './types';\n\nexport function generateLanguageFromTranslations({\n baseTranslations,\n generator,\n}: {\n baseTranslations: TranslationsByKey<string>;\n generator: MessageGenerator;\n}): TranslationsByKey<string> {\n if (!generator.transformElement && !generator.transformMessage) {\n return baseTranslations;\n }\n\n const translationKeys = Object.keys(baseTranslations);\n const generatedTranslations: TranslationsByKey<string> = {};\n\n for (const translationKey of translationKeys) {\n const translation = baseTranslations[translationKey];\n let transformedMessage = translation.message;\n\n if (generator.transformElement) {\n const messageAst = new IntlMessageFormat(translation.message).getAst();\n const transformedAst = messageAst.map(\n transformMessageFormatElement(generator.transformElement),\n );\n transformedMessage = printAST(transformedAst);\n }\n\n if (generator.transformMessage) {\n transformedMessage = generator.transformMessage(transformedMessage);\n }\n\n generatedTranslations[translationKey] = {\n message: transformedMessage,\n };\n }\n\n return generatedTranslations;\n}\n\nfunction transformMessageFormatElement(\n transformElement: (message: string) => string,\n): (messageFormatElement: MessageFormatElement) => MessageFormatElement {\n return (messageFormatElement: MessageFormatElement) => {\n const transformedMessageFormatElement = { ...messageFormatElement };\n\n switch (transformedMessageFormatElement.type) {\n case TYPE.literal:\n const transformedValue = transformElement(\n transformedMessageFormatElement.value,\n );\n transformedMessageFormatElement.value = transformedValue;\n break;\n\n case TYPE.select:\n case TYPE.plural:\n const transformedOptions: Record<string, PluralOrSelectOption> = {\n ...transformedMessageFormatElement.options,\n };\n\n for (const key of Object.keys(transformedOptions)) {\n transformedOptions[key].value = transformedOptions[key].value.map(\n transformMessageFormatElement(transformElement),\n );\n }\n\n break;\n\n case TYPE.tag:\n const transformedChildren =\n transformedMessageFormatElement.children.map(\n transformMessageFormatElement(transformElement),\n );\n transformedMessageFormatElement.children = transformedChildren;\n break;\n\n default:\n break;\n }\n\n return transformedMessageFormatElement;\n };\n}\n","import path from 'path';\n\nimport { glob } from 'tinyglobby';\nimport type {\n TranslationsByKey,\n UserConfig,\n LoadedTranslation,\n LanguageTarget,\n LanguageName,\n TranslationFileMetadata,\n TranslationFileContents,\n} from './types';\nimport pc from 'picocolors';\n\nimport { trace } from './logger';\nimport {\n defaultTranslationDirSuffix,\n type Fallback,\n getAltLanguageFilePath,\n getAltLanguages,\n getDevTranslationFileGlob,\n} from './utils';\nimport { generateLanguageFromTranslations } from './generate-language';\n\nexport function getUniqueKey(key: string, namespace: string) {\n return `${key}.${namespace}`;\n}\n\nexport function mergeWithDevLanguageTranslation({\n translation,\n devTranslation,\n}: {\n translation: TranslationsByKey;\n devTranslation: TranslationsByKey;\n}) {\n // Only use keys from the dev translation\n const keys = Object.keys(devTranslation);\n const newLanguage: TranslationsByKey = {};\n\n for (const key of keys) {\n if (translation[key]) {\n newLanguage[key] = {\n message: translation[key].message,\n description: devTranslation[key].description,\n };\n }\n }\n\n return newLanguage;\n}\n\nfunction getLanguageFallbacks({ languages }: { languages: LanguageTarget[] }) {\n const languageFallbackMap = new Map<LanguageName, LanguageName>();\n\n for (const lang of languages) {\n if (lang.extends) {\n languageFallbackMap.set(lang.name, lang.extends);\n }\n }\n\n return languageFallbackMap;\n}\n\nexport function getLanguageHierarchy({\n languages,\n}: {\n languages: LanguageTarget[];\n}) {\n const hierarchyMap = new Map<LanguageName, LanguageName[]>();\n const fallbacks = getLanguageFallbacks({ languages });\n\n for (const lang of languages) {\n const langHierarchy = [];\n let currLang = lang.extends;\n\n while (currLang) {\n langHierarchy.push(currLang);\n\n currLang = fallbacks.get(currLang);\n }\n\n hierarchyMap.set(lang.name, langHierarchy);\n }\n\n return hierarchyMap;\n}\n\nexport function getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n}: {\n languages: LanguageTarget[];\n languageName: string;\n devLanguage: string;\n fallbacks: Fallback;\n}) {\n const languageHierarchy = getLanguageHierarchy({ languages }).get(\n languageName,\n );\n\n if (!languageHierarchy) {\n throw new Error(`Missing language hierarchy for ${languageName}`);\n }\n\n const fallbackLanguageOrder: string[] = [languageName];\n\n if (fallbacks !== 'none') {\n fallbackLanguageOrder.unshift(...languageHierarchy.reverse());\n\n if (fallbacks === 'all' && fallbackLanguageOrder[0] !== devLanguage) {\n fallbackLanguageOrder.unshift(devLanguage);\n }\n }\n\n return fallbackLanguageOrder;\n}\n\nfunction getNamespaceByFilePath(\n relativePath: string,\n { translationsDirectorySuffix = defaultTranslationDirSuffix }: UserConfig,\n) {\n let namespace = path\n .dirname(relativePath)\n .replace(/^src\\//, '')\n .replace(/\\//g, '_');\n\n if (namespace.endsWith(translationsDirectorySuffix)) {\n namespace = namespace.slice(0, -translationsDirectorySuffix.length);\n }\n\n return namespace;\n}\n\nfunction printValidationError(...params: unknown[]) {\n // eslint-disable-next-line no-console\n console.error(pc.red('Error loading translation:'), ...params);\n}\n\nfunction getTranslationsFromFile(\n translationFileContents: unknown,\n {\n isAltLanguage,\n filePath,\n withTags,\n }: { isAltLanguage: boolean; filePath: string; withTags?: boolean },\n): {\n $namespace: unknown;\n keys: TranslationsByKey;\n metadata: TranslationFileMetadata;\n} {\n if (!translationFileContents || typeof translationFileContents !== 'object') {\n throw new Error(\n `Unable to read translation file ${filePath}. Translations must be an object.`,\n );\n }\n\n const { $namespace, _meta, ...keys } =\n translationFileContents as TranslationFileContents;\n\n if (isAltLanguage && $namespace) {\n printValidationError(\n `Found $namespace in alt language file in ${filePath}. $namespace is only used in the dev language and will be ignored.`,\n );\n }\n\n if (!isAltLanguage && $namespace && typeof $namespace !== 'string') {\n printValidationError(\n `Found non-string $namespace in language file in ${filePath}. $namespace must be a string.`,\n );\n }\n\n if (isAltLanguage && _meta?.tags) {\n printValidationError(\n `Found _meta.tags in alt language file in ${filePath}. _meta.tags is only used in the dev language and will be ignored.`,\n );\n }\n\n // Never return tags if we're fetching translations for an alt language\n const includeTags = !isAltLanguage && withTags;\n const validKeys: TranslationsByKey = {};\n\n for (const [translationKey, { tags, ...translation }] of Object.entries(\n keys,\n )) {\n if (typeof translation === 'string') {\n printValidationError(\n `Found string for a translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation) {\n printValidationError(\n `Found empty translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation.message || typeof translation.message !== 'string') {\n printValidationError(\n `No message found for translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n validKeys[translationKey] = {\n ...translation,\n tags: includeTags ? tags : undefined,\n };\n }\n\n const metadata = { tags: includeTags ? _meta?.tags : undefined };\n\n return { $namespace, keys: validKeys, metadata };\n}\n\nexport function loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation,\n fallbacks,\n }: {\n filePath: string;\n languageName: string;\n devTranslation: TranslationsByKey;\n fallbacks: Fallback;\n },\n { devLanguage, languages }: UserConfig,\n): TranslationsByKey {\n const altLanguageTranslation = {};\n\n const fallbackLanguageOrder = getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n });\n\n trace(\n `Loading alt language file with precedence: ${fallbackLanguageOrder\n .slice()\n .reverse()\n .join(' -> ')}`,\n );\n\n for (const fallbackLanguage of fallbackLanguageOrder) {\n if (fallbackLanguage !== devLanguage) {\n try {\n const altFilePath = getAltLanguageFilePath(filePath, fallbackLanguage);\n delete require.cache[altFilePath];\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationFile = require(altFilePath);\n const { keys: fallbackLanguageTranslation } = getTranslationsFromFile(\n translationFile,\n {\n filePath: altFilePath,\n isAltLanguage: true,\n },\n );\n Object.assign(\n altLanguageTranslation,\n mergeWithDevLanguageTranslation({\n translation: fallbackLanguageTranslation,\n devTranslation,\n }),\n );\n } catch {\n trace(`Missing alt language file ${getAltLanguageFilePath(\n filePath,\n fallbackLanguage,\n )}\n `);\n }\n } else {\n Object.assign(altLanguageTranslation, devTranslation);\n }\n }\n\n return altLanguageTranslation;\n}\n\nfunction stripTagsFromTranslations(translations: TranslationsByKey) {\n return Object.fromEntries(\n Object.entries(translations).map(([key, { tags, ...rest }]) => [key, rest]),\n );\n}\n\nexport function loadTranslation(\n {\n filePath,\n fallbacks,\n withTags,\n }: {\n filePath: string;\n fallbacks: Fallback;\n withTags?: boolean;\n },\n userConfig: UserConfig,\n): LoadedTranslation {\n trace(\n `Loading translation file in \"${fallbacks}\" fallback mode: \"${filePath}\"`,\n );\n\n const languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n > = {};\n\n delete require.cache[filePath];\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationContent = require(filePath);\n const relativePath = path.relative(\n userConfig.projectRoot || process.cwd(),\n filePath,\n );\n const {\n $namespace,\n keys: devTranslation,\n metadata,\n } = getTranslationsFromFile(translationContent, {\n filePath,\n isAltLanguage: false,\n withTags,\n });\n const namespace: string =\n typeof $namespace === 'string'\n ? $namespace\n : getNamespaceByFilePath(relativePath, userConfig);\n\n trace(`Found file ${filePath}. Using namespace ${namespace}`);\n\n languageSet[userConfig.devLanguage] = devTranslation;\n\n const devTranslationNoTags = withTags\n ? stripTagsFromTranslations(devTranslation)\n : devTranslation;\n const altLanguages = getAltLanguages(userConfig);\n for (const languageName of altLanguages) {\n languageSet[languageName] = loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation: devTranslationNoTags,\n fallbacks,\n },\n userConfig,\n );\n }\n\n for (const generatedLanguage of userConfig.generatedLanguages || []) {\n const { name: generatedLanguageName, generator } = generatedLanguage;\n const baseLanguage = generatedLanguage.extends || userConfig.devLanguage;\n const baseTranslations = languageSet[baseLanguage];\n\n languageSet[generatedLanguageName] = generateLanguageFromTranslations({\n baseTranslations,\n generator,\n });\n }\n\n return {\n filePath,\n keys: Object.keys(devTranslation),\n namespace,\n relativePath,\n languages: languageSet,\n metadata,\n };\n}\n\nexport async function loadAllTranslations(\n {\n fallbacks,\n includeNodeModules,\n withTags,\n }: { fallbacks: Fallback; includeNodeModules: boolean; withTags?: boolean },\n config: UserConfig,\n): Promise<LoadedTranslation[]> {\n const { projectRoot, ignore = [] } = config;\n\n const translationFiles = await glob(getDevTranslationFileGlob(config), {\n ignore: includeNodeModules ? ignore : [...ignore, '**/node_modules/**'],\n dot: true,\n absolute: true,\n cwd: projectRoot,\n expandDirectories: false,\n });\n\n trace(`Found ${translationFiles.length} translation files`);\n\n const loadedTranslations: LoadedTranslation[] = [];\n const keys = new Set<string>();\n\n for (const translationFile of translationFiles) {\n const loadedTranslation = loadTranslation(\n { filePath: translationFile, fallbacks, withTags },\n config,\n );\n\n loadedTranslations.push(loadedTranslation);\n\n for (const key of loadedTranslation.keys) {\n const uniqueKey = getUniqueKey(key, loadedTranslation.namespace);\n if (keys.has(uniqueKey)) {\n trace(`Duplicate keys found`);\n throw new Error(\n `Duplicate keys found. Key with namespace ${loadedTranslation.namespace} and key ${key} was found multiple times.`,\n );\n }\n keys.add(uniqueKey);\n\n const globalKey =\n loadedTranslation.languages[config.devLanguage][key].globalKey;\n if (globalKey) {\n if (keys.has(globalKey)) {\n throw new Error(\n `Duplicate keys found. Key with global key ${globalKey} and key ${key} was found multiple times`,\n );\n }\n keys.add(globalKey);\n }\n }\n }\n\n return loadedTranslations;\n}\n","import { promises as fs, existsSync } from 'fs';\nimport path from 'path';\nimport pm from 'picomatch';\n\nimport type { LoadedTranslation, UserConfig } from './types';\nimport {\n isArgumentElement,\n isDateElement,\n isNumberElement,\n isPluralElement,\n isSelectElement,\n isTagElement,\n isTimeElement,\n type MessageFormatElement,\n parse,\n} from '@formatjs/icu-messageformat-parser';\nimport prettier from 'prettier';\n\nimport {\n getTranslationMessages,\n getDevTranslationFileGlob,\n getTSFileFromDevLanguageFile,\n getDevLanguageFileFromAltLanguageFile,\n getAltTranslationFileGlob,\n isDevLanguageFile,\n isAltLanguageFile,\n devTranslationFileName,\n isTranslationDirectory,\n} from './utils';\nimport { trace } from './logger';\nimport { loadAllTranslations, loadTranslation } from './load-translations';\n\ntype ICUParams = Record<string, string>;\n\ninterface TranslationTypeInfo {\n params: ICUParams;\n message: string;\n hasTags: boolean;\n}\n\nfunction extractHasTags(ast: MessageFormatElement[]): boolean {\n return ast.some((element) => {\n if (isSelectElement(element) || isPluralElement(element)) {\n const children = Object.values(element.options).map((o) => o.value);\n return children.some((child) => extractHasTags(child));\n }\n\n return isTagElement(element);\n });\n}\n\nfunction extractParamTypes(\n ast: MessageFormatElement[],\n currentParams: ICUParams,\n): [params: ICUParams, vocabTypesImports: Set<string>] {\n let params = { ...currentParams };\n let vocabTypesImports = new Set<string>();\n\n for (const element of ast) {\n if (isArgumentElement(element)) {\n if (!(element.value in params)) {\n // Preserve existing types for parameters that we've already parsed\n // This applies to self-referential parameters, for example `{foo, plural, 1 {{foo} thing} other {{foo} things}}`\n params[element.value] = 'string';\n }\n } else if (isNumberElement(element)) {\n params[element.value] = 'number';\n } else if (isPluralElement(element)) {\n params[element.value] = 'number';\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n } else if (isDateElement(element) || isTimeElement(element)) {\n params[element.value] = 'Date | number';\n } else if (isTagElement(element)) {\n params[element.value] = 'FormatXMLElementFn<T>';\n vocabTypesImports.add('FormatXMLElementFn');\n\n const [newParams, subImports] = extractParamTypes(\n element.children,\n params,\n );\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n } else if (isSelectElement(element)) {\n const options = Object.keys(element.options);\n\n // `other` will always be an option as the parser enforces this by default\n const nonOtherOptions = options.filter((o) => o !== 'other');\n const nonOtherOptionsUnion = nonOtherOptions\n .map((o) => `'${o}'`)\n .join(' | ');\n\n params[element.value] = `StringWithSuggestions<${nonOtherOptionsUnion}>`;\n vocabTypesImports.add('StringWithSuggestions');\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n }\n }\n\n return [params, vocabTypesImports];\n}\n\nfunction serialiseObjectToType(v: any) {\n let result = '';\n\n for (const [key, value] of Object.entries(v)) {\n result += `${JSON.stringify(key)}: ${\n value && typeof value === 'object' ? serialiseObjectToType(value) : value\n },`;\n }\n\n return `{ ${result} }`;\n}\n\nconst serializeTypeImports = (imports: Set<string>, moduleName: string) => {\n if (imports.size === 0) {\n return '';\n }\n\n const importNames = Array.from(imports).sort();\n return `import type { ${importNames.join(', ')} } from '${moduleName}';`;\n};\n\nfunction serialiseTranslationRuntime(\n value: Map<string, TranslationTypeInfo>,\n imports: Set<string>,\n loadedTranslation: LoadedTranslation,\n) {\n trace('Serialising translations:', loadedTranslation);\n const translationsType: any = {};\n\n for (const [key, { params, message, hasTags }] of value.entries()) {\n let translationFunctionString = `() => ${message}`;\n\n if (Object.keys(params).length > 0) {\n const formatGeneric = hasTags ? '<T = string>' : '';\n const formatReturn =\n hasTags && imports.has('FormatXMLElementFn')\n ? 'ReturnType<FormatXMLElementFn<T>>'\n : 'string';\n translationFunctionString = `${formatGeneric}(values: ${serialiseObjectToType(\n params,\n )}) => ${formatReturn}`;\n }\n\n translationsType[key] = translationFunctionString;\n }\n\n const languagesUnionAsString = Object.keys(loadedTranslation.languages)\n .map((l) => `'${l}'`)\n .join(' | ');\n const languageEntries = Object.entries(loadedTranslation.languages)\n .map(\n ([languageName, translations]) =>\n `${JSON.stringify(languageName)}: createLanguage(${JSON.stringify(\n getTranslationMessages(translations),\n )})`,\n )\n .join(',');\n\n // TranslationFile is now always imported\n const typeImports = new Set([...imports, 'TranslationFile']);\n\n return /* ts */ `\n // This file is automatically generated by Vocab.\n // To make changes update translation.json files directly.\n\n ${serializeTypeImports(typeImports, '@vocab/core')}\n import { createLanguage, createTranslationFile } from '@vocab/core/runtime';\n\n const translations: TranslationFile<\n ${languagesUnionAsString},\n ${serialiseObjectToType(translationsType)}\n > = createTranslationFile\n ({\n ${languageEntries}\n });\n\n export default translations;\n `;\n}\n\nexport async function generateRuntime(loadedTranslation: LoadedTranslation) {\n const { languages: loadedLanguages, filePath } = loadedTranslation;\n\n trace('Generating types for', filePath);\n const translationTypes = new Map<string, TranslationTypeInfo>();\n\n let imports = new Set<string>();\n\n for (const key of loadedTranslation.keys) {\n let params: ICUParams = {};\n const messages = new Set();\n let hasTags = false;\n\n for (const translatedLanguage of Object.values(loadedLanguages)) {\n if (translatedLanguage[key]) {\n const ast = parse(translatedLanguage[key].message);\n\n hasTags = hasTags || extractHasTags(ast);\n\n const [parsedParams, vocabTypesImports] = extractParamTypes(\n ast,\n params,\n );\n\n imports = new Set([...imports, ...vocabTypesImports]);\n params = parsedParams;\n\n messages.add(JSON.stringify(translatedLanguage[key].message));\n }\n }\n\n translationTypes.set(key, {\n params,\n hasTags,\n message: Array.from(messages).join(' | '),\n });\n }\n\n const prettierConfig = await prettier.resolveConfig(filePath);\n const serializedTranslationType = serialiseTranslationRuntime(\n translationTypes,\n imports,\n loadedTranslation,\n );\n const declaration = await prettier.format(serializedTranslationType, {\n ...prettierConfig,\n parser: 'typescript',\n });\n const outputFilePath = getTSFileFromDevLanguageFile(filePath);\n trace(`Writing translation types to ${outputFilePath}`);\n await writeIfChanged(outputFilePath, declaration);\n}\n\nexport async function watch(config: UserConfig) {\n const cwd = config.projectRoot || process.cwd();\n\n const ignorePatterns = [\n ...(config.ignore || []),\n '**/node_modules/**',\n '**/.git/**',\n ];\n\n const ignoredFunction = (pathToCheck: string) =>\n pm.isMatch(pathToCheck, ignorePatterns);\n\n const chokidar = await import('chokidar');\n const watcher = chokidar.watch(cwd, {\n ignored: ignoredFunction,\n ignoreInitial: true,\n });\n\n const devFileGlob = getDevTranslationFileGlob(config);\n const altFileGlob = getAltTranslationFileGlob(config);\n\n const onTranslationChange = async (relativePath: string) => {\n const matchesTranslationFiles = pm.isMatch(relativePath, [\n devFileGlob,\n altFileGlob,\n ]);\n\n if (!matchesTranslationFiles) {\n return;\n }\n\n trace(`Detected change for file ${relativePath}`);\n\n let targetFile;\n\n if (isDevLanguageFile(relativePath)) {\n targetFile = path.resolve(cwd, relativePath);\n } else if (isAltLanguageFile(relativePath)) {\n targetFile = getDevLanguageFileFromAltLanguageFile(\n path.resolve(cwd, relativePath),\n );\n }\n\n if (targetFile) {\n watcher.add(targetFile);\n try {\n const loadedTranslation = loadTranslation(\n { filePath: targetFile, fallbacks: 'all' },\n config,\n );\n\n await generateRuntime(loadedTranslation);\n } catch (e) {\n // eslint-disable-next-line no-console\n console.log('Failed to generate types for', relativePath);\n // eslint-disable-next-line no-console\n console.error(e);\n }\n }\n };\n\n const onNewDirectory = async (relativePath: string) => {\n trace('Detected new directory', relativePath);\n\n if (!isTranslationDirectory(relativePath, config)) {\n trace('Ignoring non-translation directory:', relativePath);\n return;\n }\n\n const newFilePath = path.join(relativePath, devTranslationFileName);\n if (!existsSync(newFilePath)) {\n await fs.writeFile(newFilePath, JSON.stringify({}, null, 2));\n trace('Created new empty translation file:', newFilePath);\n } else {\n trace(\n `New directory already contains translation file. Skipping creation. Existing file ${newFilePath}`,\n );\n }\n };\n\n watcher.on('addDir', onNewDirectory);\n watcher.on('add', onTranslationChange).on('change', onTranslationChange);\n\n return () => watcher.close();\n}\n\nexport async function compile(\n { watch: shouldWatch = false } = {},\n config: UserConfig,\n) {\n const translations = await loadAllTranslations(\n { fallbacks: 'all', includeNodeModules: false },\n config,\n );\n\n for (const loadedTranslation of translations) {\n await generateRuntime(loadedTranslation);\n }\n\n if (shouldWatch) {\n trace('Listening for changes to files...');\n return watch(config);\n }\n}\n\nasync function writeIfChanged(filepath: string, contents: string) {\n let hasChanged = true;\n\n try {\n const existingContents = await fs.readFile(filepath, { encoding: 'utf-8' });\n\n hasChanged = existingContents !== contents;\n } catch {\n // ignore error, likely a file doesn't exist error so we want to write anyway\n }\n\n if (hasChanged) {\n await fs.writeFile(filepath, contents, { encoding: 'utf-8' });\n }\n}\n","/* eslint-disable no-console */\nimport type { UserConfig, LoadedTranslation, LanguageName } from '../types';\nimport pc from 'picocolors';\n\nimport { loadAllTranslations } from '../load-translations';\nimport { getAltLanguages } from '../utils';\n\nexport function findMissingKeys(\n loadedTranslation: LoadedTranslation,\n devLanguageName: LanguageName,\n altLanguages: LanguageName[],\n) {\n const devLanguage = loadedTranslation.languages[devLanguageName];\n\n if (!devLanguage) {\n throw new Error(\n `Failed to load dev language: ${loadedTranslation.filePath}`,\n );\n }\n\n const result: Record<LanguageName, string[]> = {};\n let valid = true;\n\n const requiredKeys = Object.keys(devLanguage);\n\n if (requiredKeys.length > 0) {\n for (const altLanguageName of altLanguages) {\n const altLanguage = loadedTranslation.languages[altLanguageName] ?? {};\n\n for (const key of requiredKeys) {\n if (typeof altLanguage[key]?.message !== 'string') {\n if (!result[altLanguageName]) {\n result[altLanguageName] = [];\n }\n\n result[altLanguageName].push(key);\n valid = false;\n }\n }\n }\n }\n return [valid, result] as const;\n}\n\nexport async function validate(config: UserConfig) {\n const allTranslations = await loadAllTranslations(\n { fallbacks: 'valid', includeNodeModules: true },\n config,\n );\n\n let valid = true;\n\n for (const loadedTranslation of allTranslations) {\n const [translationValid, result] = findMissingKeys(\n loadedTranslation,\n config.devLanguage,\n getAltLanguages(config),\n );\n\n if (!translationValid) {\n valid = false;\n console.log(\n pc.red(\n `Incomplete translations: \"${pc.bold(\n loadedTranslation.relativePath,\n )}\"`,\n ),\n );\n\n for (const lang of Object.keys(result)) {\n const missingKeys = result[lang];\n\n console.log(\n pc.yellow(lang),\n '->',\n missingKeys.map((v) => `\"${v}\"`).join(', '),\n );\n }\n }\n }\n\n return valid;\n}\n","export class ValidationError extends Error {\n code: string;\n rawMessage: string;\n constructor(code: string, message: string) {\n super(`Invalid vocab.config.js: ${code} - ${message}`);\n this.code = code;\n this.rawMessage = message;\n }\n}\n","import type { UserConfig } from './types';\nimport path from 'path';\n\nimport pc from 'picocolors';\nimport findUp from 'find-up';\nimport Validator from 'fastest-validator';\nimport { ValidationError } from './ValidationError';\nimport { trace } from './logger';\n\nconst boldCyan = (s: string) => pc.bold(pc.cyan(s));\n\nconst validator = new Validator();\nconst schema = {\n $$strict: true,\n devLanguage: {\n type: 'string',\n },\n languages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n },\n },\n },\n generatedLanguages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n generator: {\n type: 'object',\n props: {\n transformElement: { type: 'function', optional: true },\n transformMessage: { type: 'function', optional: true },\n },\n },\n },\n },\n optional: true,\n },\n translationsDirectorySuffix: { type: 'string', optional: true },\n projectRoot: { type: 'string', optional: true },\n ignore: {\n type: 'array',\n items: 'string',\n optional: true,\n },\n};\nconst checkConfigFile = validator.compile(schema);\n\nconst splitMap = (message: string, callback: (value: string) => string) =>\n message\n .split(' ,')\n .map((v) => callback(v))\n .join(' ,');\n\nexport function validateConfig(c: UserConfig) {\n trace('Validating configuration file');\n // Note: checkConfigFile mutates the config file by applying defaults\n const isValid = checkConfigFile(c);\n if (isValid !== true) {\n throw new ValidationError(\n 'InvalidStructure',\n (Array.isArray(isValid) ? isValid : [])\n .map((v) => {\n if (v.type === 'objectStrict') {\n return `Invalid key(s) ${splitMap(\n v.actual,\n (m) => `\"${pc.cyan(m)}\"`,\n )}. Expected one of ${splitMap(v.expected, pc.green)}`;\n }\n if (v.field) {\n return v.message?.replace(v.field, pc.cyan(v.field));\n }\n return v.message;\n })\n .join(' \\n'),\n );\n }\n\n const languageStrings = c.languages.map((v) => v.name);\n\n // Dev Language should exist in languages\n if (!languageStrings.includes(c.devLanguage)) {\n throw new ValidationError(\n 'InvalidDevLanguage',\n `The dev language \"${boldCyan(\n c.devLanguage,\n )}\" was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n\n const foundLanguages: string[] = [];\n for (const lang of c.languages) {\n // Languages must only exist once\n if (foundLanguages.includes(lang.name)) {\n throw new ValidationError(\n 'DuplicateLanguage',\n `The language \"${boldCyan(lang.name)}\" was defined multiple times.`,\n );\n }\n foundLanguages.push(lang.name);\n\n // Any extends must be in languages\n if (lang.extends && !languageStrings.includes(lang.extends)) {\n throw new ValidationError(\n 'InvalidExtends',\n `The language \"${boldCyan(lang.name)}\"'s extends of ${boldCyan(\n lang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n const foundGeneratedLanguages: string[] = [];\n for (const generatedLang of c.generatedLanguages || []) {\n // Generated languages must only exist once\n if (foundGeneratedLanguages.includes(generatedLang.name)) {\n throw new ValidationError(\n 'DuplicateGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" was defined multiple times.`,\n );\n }\n foundGeneratedLanguages.push(generatedLang.name);\n\n // Generated language names must not conflict with language names\n if (languageStrings.includes(generatedLang.name)) {\n throw new ValidationError(\n 'InvalidGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" is already defined as a language.`,\n );\n }\n\n // Any extends must be in languages\n if (\n generatedLang.extends &&\n !languageStrings.includes(generatedLang.extends)\n ) {\n throw new ValidationError(\n 'InvalidExtends',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\"'s extends of ${boldCyan(\n generatedLang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n trace('Configuration file is valid');\n\n return true;\n}\n\nfunction createConfig(configFilePath: string) {\n const cwd = path.dirname(configFilePath);\n\n return {\n projectRoot: cwd,\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n ...(require(configFilePath) as UserConfig),\n };\n}\n\nexport async function resolveConfig(\n customConfigFilePath?: string,\n): Promise<UserConfig | null> {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : await findUp(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n return null;\n}\n\nexport function resolveConfigSync(\n customConfigFilePath?: string,\n): UserConfig | null {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : findUp.sync(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n\n return null;\n}\n"],"mappings":"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"}
|
package/dist/index.d.cts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { C as TranslationsByLanguage, S as TranslationsByKey, _ as TranslationKeys, a as LoadedTranslation, b as TranslationModule, c as ParsedFormatFnByKey, d as Tags, f as TranslationData, g as TranslationKey, h as TranslationFileMetadata, i as LanguageTarget, l as ParsedICUMessages, m as TranslationFileContents, n as GeneratedLanguageTarget, o as MessageGenerator, p as TranslationFile, r as LanguageName, s as ParsedFormatFn, t as FormatXMLElementFn, u as StringWithSuggestions, v as TranslationMessage, w as UserConfig, x as TranslationModuleByLanguage, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { C as TranslationsByLanguage, S as TranslationsByKey, _ as TranslationKeys, a as LoadedTranslation, b as TranslationModule, c as ParsedFormatFnByKey, d as Tags, f as TranslationData, g as TranslationKey, h as TranslationFileMetadata, i as LanguageTarget, l as ParsedICUMessages, m as TranslationFileContents, n as GeneratedLanguageTarget, o as MessageGenerator, p as TranslationFile, r as LanguageName, s as ParsedFormatFn, t as FormatXMLElementFn, u as StringWithSuggestions, v as TranslationMessage, w as UserConfig, x as TranslationModuleByLanguage, y as TranslationMessagesByKey } from "./types-Bybu5Wvr.cjs";
|
|
2
2
|
|
|
3
3
|
//#region src/compile.d.ts
|
|
4
4
|
declare function watch(config: UserConfig): Promise<() => Promise<void>>;
|
package/dist/index.d.mts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { C as TranslationsByLanguage, S as TranslationsByKey, _ as TranslationKeys, a as LoadedTranslation, b as TranslationModule, c as ParsedFormatFnByKey, d as Tags, f as TranslationData, g as TranslationKey, h as TranslationFileMetadata, i as LanguageTarget, l as ParsedICUMessages, m as TranslationFileContents, n as GeneratedLanguageTarget, o as MessageGenerator, p as TranslationFile, r as LanguageName, s as ParsedFormatFn, t as FormatXMLElementFn, u as StringWithSuggestions, v as TranslationMessage, w as UserConfig, x as TranslationModuleByLanguage, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { C as TranslationsByLanguage, S as TranslationsByKey, _ as TranslationKeys, a as LoadedTranslation, b as TranslationModule, c as ParsedFormatFnByKey, d as Tags, f as TranslationData, g as TranslationKey, h as TranslationFileMetadata, i as LanguageTarget, l as ParsedICUMessages, m as TranslationFileContents, n as GeneratedLanguageTarget, o as MessageGenerator, p as TranslationFile, r as LanguageName, s as ParsedFormatFn, t as FormatXMLElementFn, u as StringWithSuggestions, v as TranslationMessage, w as UserConfig, x as TranslationModuleByLanguage, y as TranslationMessagesByKey } from "./types-DW00hfr8.mjs";
|
|
2
2
|
|
|
3
3
|
//#region src/compile.d.ts
|
|
4
4
|
declare function watch(config: UserConfig): Promise<() => Promise<void>>;
|
package/dist/index.mjs
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { createRequire } from "node:module";
|
|
2
2
|
import { existsSync, promises } from "fs";
|
|
3
3
|
import path from "path";
|
|
4
4
|
import pm from "picomatch";
|
|
@@ -11,10 +11,11 @@ import { IntlMessageFormat } from "intl-messageformat";
|
|
|
11
11
|
import { printAST } from "@formatjs/icu-messageformat-parser/printer.js";
|
|
12
12
|
import findUp from "find-up";
|
|
13
13
|
import Validator from "fastest-validator";
|
|
14
|
-
|
|
14
|
+
//#region \0rolldown/runtime.js
|
|
15
|
+
var __require = /* @__PURE__ */ createRequire(import.meta.url);
|
|
16
|
+
//#endregion
|
|
15
17
|
//#region src/logger.ts
|
|
16
18
|
const trace = debug(`vocab:core`);
|
|
17
|
-
|
|
18
19
|
//#endregion
|
|
19
20
|
//#region src/utils.ts
|
|
20
21
|
const defaultTranslationDirSuffix = ".vocab";
|
|
@@ -22,7 +23,7 @@ const devTranslationFileName = "translations.json";
|
|
|
22
23
|
const globAnyPathWithOptionalPrefix = "**/?(*)";
|
|
23
24
|
const compiledVocabFileFilter = /\.vocab[\\/]index\.(?:ts|js|cjs|mjs)$/;
|
|
24
25
|
function isDevLanguageFile(filePath) {
|
|
25
|
-
return filePath.endsWith(
|
|
26
|
+
return filePath.endsWith(`/translations.json`) || filePath === "translations.json";
|
|
26
27
|
}
|
|
27
28
|
function isAltLanguageFile(filePath) {
|
|
28
29
|
return filePath.endsWith(".translations.json");
|
|
@@ -79,7 +80,6 @@ function mapValues(obj, func) {
|
|
|
79
80
|
function getTranslationMessages(translations) {
|
|
80
81
|
return mapValues(translations, (v) => v.message);
|
|
81
82
|
}
|
|
82
|
-
|
|
83
83
|
//#endregion
|
|
84
84
|
//#region src/generate-language.ts
|
|
85
85
|
function generateLanguageFromTranslations({ baseTranslations, generator }) {
|
|
@@ -115,7 +115,6 @@ function transformMessageFormatElement(transformElement) {
|
|
|
115
115
|
return transformedMessageFormatElement;
|
|
116
116
|
};
|
|
117
117
|
}
|
|
118
|
-
|
|
119
118
|
//#endregion
|
|
120
119
|
//#region src/load-translations.ts
|
|
121
120
|
function getUniqueKey(key, namespace) {
|
|
@@ -303,7 +302,6 @@ async function loadAllTranslations({ fallbacks, includeNodeModules, withTags },
|
|
|
303
302
|
}
|
|
304
303
|
return loadedTranslations;
|
|
305
304
|
}
|
|
306
|
-
|
|
307
305
|
//#endregion
|
|
308
306
|
//#region src/compile.ts
|
|
309
307
|
function extractHasTags(ast) {
|
|
@@ -353,7 +351,7 @@ function serialiseObjectToType(v) {
|
|
|
353
351
|
}
|
|
354
352
|
const serializeTypeImports = (imports, moduleName) => {
|
|
355
353
|
if (imports.size === 0) return "";
|
|
356
|
-
return `import type { ${Array.from(imports).join(", ")} } from '${moduleName}';`;
|
|
354
|
+
return `import type { ${Array.from(imports).sort().join(", ")} } from '${moduleName}';`;
|
|
357
355
|
};
|
|
358
356
|
function serialiseTranslationRuntime(value, imports, loadedTranslation) {
|
|
359
357
|
trace("Serialising translations:", loadedTranslation);
|
|
@@ -373,9 +371,8 @@ function serialiseTranslationRuntime(value, imports, loadedTranslation) {
|
|
|
373
371
|
// This file is automatically generated by Vocab.
|
|
374
372
|
// To make changes update translation.json files directly.
|
|
375
373
|
|
|
376
|
-
${serializeTypeImports(imports, "@vocab/core")}
|
|
374
|
+
${serializeTypeImports(new Set([...imports, "TranslationFile"]), "@vocab/core")}
|
|
377
375
|
import { createLanguage, createTranslationFile } from '@vocab/core/runtime';
|
|
378
|
-
import type { TranslationFile } from '@vocab/core';
|
|
379
376
|
|
|
380
377
|
const translations: TranslationFile<
|
|
381
378
|
${languagesUnionAsString},
|
|
@@ -488,7 +485,6 @@ async function writeIfChanged(filepath, contents) {
|
|
|
488
485
|
} catch {}
|
|
489
486
|
if (hasChanged) await promises.writeFile(filepath, contents, { encoding: "utf-8" });
|
|
490
487
|
}
|
|
491
|
-
|
|
492
488
|
//#endregion
|
|
493
489
|
//#region src/validate/index.ts
|
|
494
490
|
function findMissingKeys(loadedTranslation, devLanguageName, altLanguages) {
|
|
@@ -526,7 +522,6 @@ async function validate(config) {
|
|
|
526
522
|
}
|
|
527
523
|
return valid;
|
|
528
524
|
}
|
|
529
|
-
|
|
530
525
|
//#endregion
|
|
531
526
|
//#region src/ValidationError.ts
|
|
532
527
|
var ValidationError = class extends Error {
|
|
@@ -538,7 +533,6 @@ var ValidationError = class extends Error {
|
|
|
538
533
|
this.rawMessage = message;
|
|
539
534
|
}
|
|
540
535
|
};
|
|
541
|
-
|
|
542
536
|
//#endregion
|
|
543
537
|
//#region src/config.ts
|
|
544
538
|
const boldCyan = (s) => pc.bold(pc.cyan(s));
|
|
@@ -650,7 +644,7 @@ function resolveConfigSync(customConfigFilePath) {
|
|
|
650
644
|
trace("No configuration file found");
|
|
651
645
|
return null;
|
|
652
646
|
}
|
|
653
|
-
|
|
654
647
|
//#endregion
|
|
655
648
|
export { compile, compiledVocabFileFilter, getAltLanguageFilePath, getAltLanguages, getDevLanguageFileFromTsFile, getUniqueKey, loadAllTranslations, loadTranslation, resolveConfig, resolveConfigSync, validate, validateConfig, watch };
|
|
649
|
+
|
|
656
650
|
//# sourceMappingURL=index.mjs.map
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":["newObj: any","generatedTranslations: TranslationsByKey<string>","transformedOptions: Record<string, PluralOrSelectOption>","newLanguage: TranslationsByKey","fallbackLanguageOrder: string[]","validKeys: TranslationsByKey","languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n >","namespace: string","loadedTranslations: LoadedTranslation[]","translationsType: any","params: ICUParams","fs","result: Record<LanguageName, string[]>","foundLanguages: string[]","foundGeneratedLanguages: string[]"],"sources":["../src/logger.ts","../src/utils.ts","../src/generate-language.ts","../src/load-translations.ts","../src/compile.ts","../src/validate/index.ts","../src/ValidationError.ts","../src/config.ts"],"sourcesContent":["import pc from 'picocolors';\nimport debug from 'debug';\n\nexport const trace = debug(`vocab:core`);\n\nexport const log = (...params: unknown[]) => {\n // eslint-disable-next-line no-console\n console.log(pc.yellow('Vocab'), ...params);\n};\n","import path from 'path';\n\nimport type {\n LanguageName,\n LanguageTarget,\n TranslationsByKey,\n TranslationMessagesByKey,\n UserConfig,\n} from './types';\nimport { trace } from './logger';\n\nexport const defaultTranslationDirSuffix = '.vocab';\nexport const devTranslationFileName = 'translations.json';\n\nexport type Fallback = 'none' | 'valid' | 'all';\n\nconst globAnyPathWithOptionalPrefix = '**/?(*)';\n\nexport const compiledVocabFileFilter = /\\.vocab[\\\\/]index\\.(?:ts|js|cjs|mjs)$/;\n\nexport function isDevLanguageFile(filePath: string) {\n return (\n filePath.endsWith(`/${devTranslationFileName}`) ||\n filePath === devTranslationFileName\n );\n}\nexport function isAltLanguageFile(filePath: string) {\n return filePath.endsWith('.translations.json');\n}\nexport function isTranslationDirectory(\n filePath: string,\n {\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n }: {\n translationsDirectorySuffix?: string;\n },\n) {\n return filePath.endsWith(translationsDirectorySuffix);\n}\n\nexport function getDevTranslationFileGlob({\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n}: {\n translationsDirectorySuffix?: string;\n}) {\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${devTranslationFileName}`;\n\n trace('getDevTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltTranslationFileGlob(config: UserConfig) {\n const altLanguages = getAltLanguages(config);\n const langMatch =\n altLanguages.length === 1 ? altLanguages[0] : `{${altLanguages.join(',')}}`;\n\n const { translationsDirectorySuffix = defaultTranslationDirSuffix } = config;\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${langMatch}.${devTranslationFileName}`;\n\n trace('getAltTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltLanguages({\n devLanguage,\n languages,\n}: {\n devLanguage: LanguageName;\n languages: LanguageTarget[];\n}) {\n return languages.map((v) => v.name).filter((lang) => lang !== devLanguage);\n}\n\nexport function getDevLanguageFileFromTsFile(tsFilePath: string) {\n const directory = path.dirname(tsFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n\n trace(`Returning dev language path ${result} for path ${tsFilePath}`);\n return result;\n}\n\nexport function getDevLanguageFileFromAltLanguageFile(\n altLanguageFilePath: string,\n) {\n const directory = path.dirname(altLanguageFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n trace(\n `Returning dev language path ${result} for path ${altLanguageFilePath}`,\n );\n return result;\n}\n\nexport function getTSFileFromDevLanguageFile(devLanguageFilePath: string) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(path.join(directory, 'index.ts'));\n\n trace(`Returning TS path ${result} for path ${devLanguageFilePath}`);\n return result;\n}\n\nexport function getAltLanguageFilePath(\n devLanguageFilePath: string,\n language: string,\n) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(\n path.join(directory, `${language}.translations.json`),\n );\n trace(\n `Returning alt language path ${result} for path ${devLanguageFilePath}`,\n );\n return path.normalize(result);\n}\n\nexport function mapValues<Key extends string, OriginalValue, ReturnValue>(\n obj: Record<Key, OriginalValue>,\n func: (val: OriginalValue) => ReturnValue,\n): TranslationMessagesByKey<Key> {\n const newObj: any = {};\n const keys = Object.keys(obj) as Key[];\n for (const key of keys) {\n newObj[key] = func(obj[key]);\n }\n return newObj;\n}\n\nexport function getTranslationMessages<Key extends string>(\n translations: TranslationsByKey<Key>,\n): TranslationMessagesByKey<Key> {\n return mapValues(translations, (v) => v.message);\n}\n","import { IntlMessageFormat } from 'intl-messageformat';\nimport {\n type MessageFormatElement,\n type PluralOrSelectOption,\n TYPE,\n} from '@formatjs/icu-messageformat-parser';\nimport { printAST } from '@formatjs/icu-messageformat-parser/printer.js';\nimport type { MessageGenerator, TranslationsByKey } from './types';\n\nexport function generateLanguageFromTranslations({\n baseTranslations,\n generator,\n}: {\n baseTranslations: TranslationsByKey<string>;\n generator: MessageGenerator;\n}): TranslationsByKey<string> {\n if (!generator.transformElement && !generator.transformMessage) {\n return baseTranslations;\n }\n\n const translationKeys = Object.keys(baseTranslations);\n const generatedTranslations: TranslationsByKey<string> = {};\n\n for (const translationKey of translationKeys) {\n const translation = baseTranslations[translationKey];\n let transformedMessage = translation.message;\n\n if (generator.transformElement) {\n const messageAst = new IntlMessageFormat(translation.message).getAst();\n const transformedAst = messageAst.map(\n transformMessageFormatElement(generator.transformElement),\n );\n transformedMessage = printAST(transformedAst);\n }\n\n if (generator.transformMessage) {\n transformedMessage = generator.transformMessage(transformedMessage);\n }\n\n generatedTranslations[translationKey] = {\n message: transformedMessage,\n };\n }\n\n return generatedTranslations;\n}\n\nfunction transformMessageFormatElement(\n transformElement: (message: string) => string,\n): (messageFormatElement: MessageFormatElement) => MessageFormatElement {\n return (messageFormatElement: MessageFormatElement) => {\n const transformedMessageFormatElement = { ...messageFormatElement };\n\n switch (transformedMessageFormatElement.type) {\n case TYPE.literal:\n const transformedValue = transformElement(\n transformedMessageFormatElement.value,\n );\n transformedMessageFormatElement.value = transformedValue;\n break;\n\n case TYPE.select:\n case TYPE.plural:\n const transformedOptions: Record<string, PluralOrSelectOption> = {\n ...transformedMessageFormatElement.options,\n };\n\n for (const key of Object.keys(transformedOptions)) {\n transformedOptions[key].value = transformedOptions[key].value.map(\n transformMessageFormatElement(transformElement),\n );\n }\n\n break;\n\n case TYPE.tag:\n const transformedChildren =\n transformedMessageFormatElement.children.map(\n transformMessageFormatElement(transformElement),\n );\n transformedMessageFormatElement.children = transformedChildren;\n break;\n\n default:\n break;\n }\n\n return transformedMessageFormatElement;\n };\n}\n","import path from 'path';\n\nimport { glob } from 'tinyglobby';\nimport type {\n TranslationsByKey,\n UserConfig,\n LoadedTranslation,\n LanguageTarget,\n LanguageName,\n TranslationFileMetadata,\n TranslationFileContents,\n} from './types';\nimport pc from 'picocolors';\n\nimport { trace } from './logger';\nimport {\n defaultTranslationDirSuffix,\n type Fallback,\n getAltLanguageFilePath,\n getAltLanguages,\n getDevTranslationFileGlob,\n} from './utils';\nimport { generateLanguageFromTranslations } from './generate-language';\n\nexport function getUniqueKey(key: string, namespace: string) {\n return `${key}.${namespace}`;\n}\n\nexport function mergeWithDevLanguageTranslation({\n translation,\n devTranslation,\n}: {\n translation: TranslationsByKey;\n devTranslation: TranslationsByKey;\n}) {\n // Only use keys from the dev translation\n const keys = Object.keys(devTranslation);\n const newLanguage: TranslationsByKey = {};\n\n for (const key of keys) {\n if (translation[key]) {\n newLanguage[key] = {\n message: translation[key].message,\n description: devTranslation[key].description,\n };\n }\n }\n\n return newLanguage;\n}\n\nfunction getLanguageFallbacks({ languages }: { languages: LanguageTarget[] }) {\n const languageFallbackMap = new Map<LanguageName, LanguageName>();\n\n for (const lang of languages) {\n if (lang.extends) {\n languageFallbackMap.set(lang.name, lang.extends);\n }\n }\n\n return languageFallbackMap;\n}\n\nexport function getLanguageHierarchy({\n languages,\n}: {\n languages: LanguageTarget[];\n}) {\n const hierarchyMap = new Map<LanguageName, LanguageName[]>();\n const fallbacks = getLanguageFallbacks({ languages });\n\n for (const lang of languages) {\n const langHierarchy = [];\n let currLang = lang.extends;\n\n while (currLang) {\n langHierarchy.push(currLang);\n\n currLang = fallbacks.get(currLang);\n }\n\n hierarchyMap.set(lang.name, langHierarchy);\n }\n\n return hierarchyMap;\n}\n\nexport function getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n}: {\n languages: LanguageTarget[];\n languageName: string;\n devLanguage: string;\n fallbacks: Fallback;\n}) {\n const languageHierarchy = getLanguageHierarchy({ languages }).get(\n languageName,\n );\n\n if (!languageHierarchy) {\n throw new Error(`Missing language hierarchy for ${languageName}`);\n }\n\n const fallbackLanguageOrder: string[] = [languageName];\n\n if (fallbacks !== 'none') {\n fallbackLanguageOrder.unshift(...languageHierarchy.reverse());\n\n if (fallbacks === 'all' && fallbackLanguageOrder[0] !== devLanguage) {\n fallbackLanguageOrder.unshift(devLanguage);\n }\n }\n\n return fallbackLanguageOrder;\n}\n\nfunction getNamespaceByFilePath(\n relativePath: string,\n { translationsDirectorySuffix = defaultTranslationDirSuffix }: UserConfig,\n) {\n let namespace = path\n .dirname(relativePath)\n .replace(/^src\\//, '')\n .replace(/\\//g, '_');\n\n if (namespace.endsWith(translationsDirectorySuffix)) {\n namespace = namespace.slice(0, -translationsDirectorySuffix.length);\n }\n\n return namespace;\n}\n\nfunction printValidationError(...params: unknown[]) {\n // eslint-disable-next-line no-console\n console.error(pc.red('Error loading translation:'), ...params);\n}\n\nfunction getTranslationsFromFile(\n translationFileContents: unknown,\n {\n isAltLanguage,\n filePath,\n withTags,\n }: { isAltLanguage: boolean; filePath: string; withTags?: boolean },\n): {\n $namespace: unknown;\n keys: TranslationsByKey;\n metadata: TranslationFileMetadata;\n} {\n if (!translationFileContents || typeof translationFileContents !== 'object') {\n throw new Error(\n `Unable to read translation file ${filePath}. Translations must be an object.`,\n );\n }\n\n const { $namespace, _meta, ...keys } =\n translationFileContents as TranslationFileContents;\n\n if (isAltLanguage && $namespace) {\n printValidationError(\n `Found $namespace in alt language file in ${filePath}. $namespace is only used in the dev language and will be ignored.`,\n );\n }\n\n if (!isAltLanguage && $namespace && typeof $namespace !== 'string') {\n printValidationError(\n `Found non-string $namespace in language file in ${filePath}. $namespace must be a string.`,\n );\n }\n\n if (isAltLanguage && _meta?.tags) {\n printValidationError(\n `Found _meta.tags in alt language file in ${filePath}. _meta.tags is only used in the dev language and will be ignored.`,\n );\n }\n\n // Never return tags if we're fetching translations for an alt language\n const includeTags = !isAltLanguage && withTags;\n const validKeys: TranslationsByKey = {};\n\n for (const [translationKey, { tags, ...translation }] of Object.entries(\n keys,\n )) {\n if (typeof translation === 'string') {\n printValidationError(\n `Found string for a translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation) {\n printValidationError(\n `Found empty translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation.message || typeof translation.message !== 'string') {\n printValidationError(\n `No message found for translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n validKeys[translationKey] = {\n ...translation,\n tags: includeTags ? tags : undefined,\n };\n }\n\n const metadata = { tags: includeTags ? _meta?.tags : undefined };\n\n return { $namespace, keys: validKeys, metadata };\n}\n\nexport function loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation,\n fallbacks,\n }: {\n filePath: string;\n languageName: string;\n devTranslation: TranslationsByKey;\n fallbacks: Fallback;\n },\n { devLanguage, languages }: UserConfig,\n): TranslationsByKey {\n const altLanguageTranslation = {};\n\n const fallbackLanguageOrder = getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n });\n\n trace(\n `Loading alt language file with precedence: ${fallbackLanguageOrder\n .slice()\n .reverse()\n .join(' -> ')}`,\n );\n\n for (const fallbackLanguage of fallbackLanguageOrder) {\n if (fallbackLanguage !== devLanguage) {\n try {\n const altFilePath = getAltLanguageFilePath(filePath, fallbackLanguage);\n delete require.cache[altFilePath];\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationFile = require(altFilePath);\n const { keys: fallbackLanguageTranslation } = getTranslationsFromFile(\n translationFile,\n {\n filePath: altFilePath,\n isAltLanguage: true,\n },\n );\n Object.assign(\n altLanguageTranslation,\n mergeWithDevLanguageTranslation({\n translation: fallbackLanguageTranslation,\n devTranslation,\n }),\n );\n } catch {\n trace(`Missing alt language file ${getAltLanguageFilePath(\n filePath,\n fallbackLanguage,\n )}\n `);\n }\n } else {\n Object.assign(altLanguageTranslation, devTranslation);\n }\n }\n\n return altLanguageTranslation;\n}\n\nfunction stripTagsFromTranslations(translations: TranslationsByKey) {\n return Object.fromEntries(\n Object.entries(translations).map(([key, { tags, ...rest }]) => [key, rest]),\n );\n}\n\nexport function loadTranslation(\n {\n filePath,\n fallbacks,\n withTags,\n }: {\n filePath: string;\n fallbacks: Fallback;\n withTags?: boolean;\n },\n userConfig: UserConfig,\n): LoadedTranslation {\n trace(\n `Loading translation file in \"${fallbacks}\" fallback mode: \"${filePath}\"`,\n );\n\n const languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n > = {};\n\n delete require.cache[filePath];\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationContent = require(filePath);\n const relativePath = path.relative(\n userConfig.projectRoot || process.cwd(),\n filePath,\n );\n const {\n $namespace,\n keys: devTranslation,\n metadata,\n } = getTranslationsFromFile(translationContent, {\n filePath,\n isAltLanguage: false,\n withTags,\n });\n const namespace: string =\n typeof $namespace === 'string'\n ? $namespace\n : getNamespaceByFilePath(relativePath, userConfig);\n\n trace(`Found file ${filePath}. Using namespace ${namespace}`);\n\n languageSet[userConfig.devLanguage] = devTranslation;\n\n const devTranslationNoTags = withTags\n ? stripTagsFromTranslations(devTranslation)\n : devTranslation;\n const altLanguages = getAltLanguages(userConfig);\n for (const languageName of altLanguages) {\n languageSet[languageName] = loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation: devTranslationNoTags,\n fallbacks,\n },\n userConfig,\n );\n }\n\n for (const generatedLanguage of userConfig.generatedLanguages || []) {\n const { name: generatedLanguageName, generator } = generatedLanguage;\n const baseLanguage = generatedLanguage.extends || userConfig.devLanguage;\n const baseTranslations = languageSet[baseLanguage];\n\n languageSet[generatedLanguageName] = generateLanguageFromTranslations({\n baseTranslations,\n generator,\n });\n }\n\n return {\n filePath,\n keys: Object.keys(devTranslation),\n namespace,\n relativePath,\n languages: languageSet,\n metadata,\n };\n}\n\nexport async function loadAllTranslations(\n {\n fallbacks,\n includeNodeModules,\n withTags,\n }: { fallbacks: Fallback; includeNodeModules: boolean; withTags?: boolean },\n config: UserConfig,\n): Promise<LoadedTranslation[]> {\n const { projectRoot, ignore = [] } = config;\n\n const translationFiles = await glob(getDevTranslationFileGlob(config), {\n ignore: includeNodeModules ? ignore : [...ignore, '**/node_modules/**'],\n dot: true,\n absolute: true,\n cwd: projectRoot,\n expandDirectories: false,\n });\n\n trace(`Found ${translationFiles.length} translation files`);\n\n const loadedTranslations: LoadedTranslation[] = [];\n const keys = new Set<string>();\n\n for (const translationFile of translationFiles) {\n const loadedTranslation = loadTranslation(\n { filePath: translationFile, fallbacks, withTags },\n config,\n );\n\n loadedTranslations.push(loadedTranslation);\n\n for (const key of loadedTranslation.keys) {\n const uniqueKey = getUniqueKey(key, loadedTranslation.namespace);\n if (keys.has(uniqueKey)) {\n trace(`Duplicate keys found`);\n throw new Error(\n `Duplicate keys found. Key with namespace ${loadedTranslation.namespace} and key ${key} was found multiple times.`,\n );\n }\n keys.add(uniqueKey);\n\n const globalKey =\n loadedTranslation.languages[config.devLanguage][key].globalKey;\n if (globalKey) {\n if (keys.has(globalKey)) {\n throw new Error(\n `Duplicate keys found. Key with global key ${globalKey} and key ${key} was found multiple times`,\n );\n }\n keys.add(globalKey);\n }\n }\n }\n\n return loadedTranslations;\n}\n","import { promises as fs, existsSync } from 'fs';\nimport path from 'path';\nimport pm from 'picomatch';\n\nimport type { LoadedTranslation, UserConfig } from './types';\nimport {\n isArgumentElement,\n isDateElement,\n isNumberElement,\n isPluralElement,\n isSelectElement,\n isTagElement,\n isTimeElement,\n type MessageFormatElement,\n parse,\n} from '@formatjs/icu-messageformat-parser';\nimport prettier from 'prettier';\n\nimport {\n getTranslationMessages,\n getDevTranslationFileGlob,\n getTSFileFromDevLanguageFile,\n getDevLanguageFileFromAltLanguageFile,\n getAltTranslationFileGlob,\n isDevLanguageFile,\n isAltLanguageFile,\n devTranslationFileName,\n isTranslationDirectory,\n} from './utils';\nimport { trace } from './logger';\nimport { loadAllTranslations, loadTranslation } from './load-translations';\n\ntype ICUParams = Record<string, string>;\n\ninterface TranslationTypeInfo {\n params: ICUParams;\n message: string;\n hasTags: boolean;\n}\n\nfunction extractHasTags(ast: MessageFormatElement[]): boolean {\n return ast.some((element) => {\n if (isSelectElement(element) || isPluralElement(element)) {\n const children = Object.values(element.options).map((o) => o.value);\n return children.some((child) => extractHasTags(child));\n }\n\n return isTagElement(element);\n });\n}\n\nfunction extractParamTypes(\n ast: MessageFormatElement[],\n currentParams: ICUParams,\n): [params: ICUParams, vocabTypesImports: Set<string>] {\n let params = { ...currentParams };\n let vocabTypesImports = new Set<string>();\n\n for (const element of ast) {\n if (isArgumentElement(element)) {\n if (!(element.value in params)) {\n // Preserve existing types for parameters that we've already parsed\n // This applies to self-referential parameters, for example `{foo, plural, 1 {{foo} thing} other {{foo} things}}`\n params[element.value] = 'string';\n }\n } else if (isNumberElement(element)) {\n params[element.value] = 'number';\n } else if (isPluralElement(element)) {\n params[element.value] = 'number';\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n } else if (isDateElement(element) || isTimeElement(element)) {\n params[element.value] = 'Date | number';\n } else if (isTagElement(element)) {\n params[element.value] = 'FormatXMLElementFn<T>';\n vocabTypesImports.add('FormatXMLElementFn');\n\n const [newParams, subImports] = extractParamTypes(\n element.children,\n params,\n );\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n } else if (isSelectElement(element)) {\n const options = Object.keys(element.options);\n\n // `other` will always be an option as the parser enforces this by default\n const nonOtherOptions = options.filter((o) => o !== 'other');\n const nonOtherOptionsUnion = nonOtherOptions\n .map((o) => `'${o}'`)\n .join(' | ');\n\n params[element.value] = `StringWithSuggestions<${nonOtherOptionsUnion}>`;\n vocabTypesImports.add('StringWithSuggestions');\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n }\n }\n\n return [params, vocabTypesImports];\n}\n\nfunction serialiseObjectToType(v: any) {\n let result = '';\n\n for (const [key, value] of Object.entries(v)) {\n result += `${JSON.stringify(key)}: ${\n value && typeof value === 'object' ? serialiseObjectToType(value) : value\n },`;\n }\n\n return `{ ${result} }`;\n}\n\nconst serializeTypeImports = (imports: Set<string>, moduleName: string) => {\n if (imports.size === 0) {\n return '';\n }\n\n const importNames = Array.from(imports);\n return `import type { ${importNames.join(', ')} } from '${moduleName}';`;\n};\n\nfunction serialiseTranslationRuntime(\n value: Map<string, TranslationTypeInfo>,\n imports: Set<string>,\n loadedTranslation: LoadedTranslation,\n) {\n trace('Serialising translations:', loadedTranslation);\n const translationsType: any = {};\n\n for (const [key, { params, message, hasTags }] of value.entries()) {\n let translationFunctionString = `() => ${message}`;\n\n if (Object.keys(params).length > 0) {\n const formatGeneric = hasTags ? '<T = string>' : '';\n const formatReturn =\n hasTags && imports.has('FormatXMLElementFn')\n ? 'ReturnType<FormatXMLElementFn<T>>'\n : 'string';\n translationFunctionString = `${formatGeneric}(values: ${serialiseObjectToType(\n params,\n )}) => ${formatReturn}`;\n }\n\n translationsType[key] = translationFunctionString;\n }\n\n const languagesUnionAsString = Object.keys(loadedTranslation.languages)\n .map((l) => `'${l}'`)\n .join(' | ');\n const languageEntries = Object.entries(loadedTranslation.languages)\n .map(\n ([languageName, translations]) =>\n `${JSON.stringify(languageName)}: createLanguage(${JSON.stringify(\n getTranslationMessages(translations),\n )})`,\n )\n .join(',');\n\n return /* ts */ `\n // This file is automatically generated by Vocab.\n // To make changes update translation.json files directly.\n\n ${serializeTypeImports(imports, '@vocab/core')}\n import { createLanguage, createTranslationFile } from '@vocab/core/runtime';\n import type { TranslationFile } from '@vocab/core';\n\n const translations: TranslationFile<\n ${languagesUnionAsString},\n ${serialiseObjectToType(translationsType)}\n > = createTranslationFile\n ({\n ${languageEntries}\n });\n\n export default translations;\n `;\n}\n\nexport async function generateRuntime(loadedTranslation: LoadedTranslation) {\n const { languages: loadedLanguages, filePath } = loadedTranslation;\n\n trace('Generating types for', filePath);\n const translationTypes = new Map<string, TranslationTypeInfo>();\n\n let imports = new Set<string>();\n\n for (const key of loadedTranslation.keys) {\n let params: ICUParams = {};\n const messages = new Set();\n let hasTags = false;\n\n for (const translatedLanguage of Object.values(loadedLanguages)) {\n if (translatedLanguage[key]) {\n const ast = parse(translatedLanguage[key].message);\n\n hasTags = hasTags || extractHasTags(ast);\n\n const [parsedParams, vocabTypesImports] = extractParamTypes(\n ast,\n params,\n );\n\n imports = new Set([...imports, ...vocabTypesImports]);\n params = parsedParams;\n\n messages.add(JSON.stringify(translatedLanguage[key].message));\n }\n }\n\n translationTypes.set(key, {\n params,\n hasTags,\n message: Array.from(messages).join(' | '),\n });\n }\n\n const prettierConfig = await prettier.resolveConfig(filePath);\n const serializedTranslationType = serialiseTranslationRuntime(\n translationTypes,\n imports,\n loadedTranslation,\n );\n const declaration = await prettier.format(serializedTranslationType, {\n ...prettierConfig,\n parser: 'typescript',\n });\n const outputFilePath = getTSFileFromDevLanguageFile(filePath);\n trace(`Writing translation types to ${outputFilePath}`);\n await writeIfChanged(outputFilePath, declaration);\n}\n\nexport async function watch(config: UserConfig) {\n const cwd = config.projectRoot || process.cwd();\n\n const ignorePatterns = [\n ...(config.ignore || []),\n '**/node_modules/**',\n '**/.git/**',\n ];\n\n const ignoredFunction = (pathToCheck: string) =>\n pm.isMatch(pathToCheck, ignorePatterns);\n\n const chokidar = await import('chokidar');\n const watcher = chokidar.watch(cwd, {\n ignored: ignoredFunction,\n ignoreInitial: true,\n });\n\n const devFileGlob = getDevTranslationFileGlob(config);\n const altFileGlob = getAltTranslationFileGlob(config);\n\n const onTranslationChange = async (relativePath: string) => {\n const matchesTranslationFiles = pm.isMatch(relativePath, [\n devFileGlob,\n altFileGlob,\n ]);\n\n if (!matchesTranslationFiles) {\n return;\n }\n\n trace(`Detected change for file ${relativePath}`);\n\n let targetFile;\n\n if (isDevLanguageFile(relativePath)) {\n targetFile = path.resolve(cwd, relativePath);\n } else if (isAltLanguageFile(relativePath)) {\n targetFile = getDevLanguageFileFromAltLanguageFile(\n path.resolve(cwd, relativePath),\n );\n }\n\n if (targetFile) {\n watcher.add(targetFile);\n try {\n const loadedTranslation = loadTranslation(\n { filePath: targetFile, fallbacks: 'all' },\n config,\n );\n\n await generateRuntime(loadedTranslation);\n } catch (e) {\n // eslint-disable-next-line no-console\n console.log('Failed to generate types for', relativePath);\n // eslint-disable-next-line no-console\n console.error(e);\n }\n }\n };\n\n const onNewDirectory = async (relativePath: string) => {\n trace('Detected new directory', relativePath);\n\n if (!isTranslationDirectory(relativePath, config)) {\n trace('Ignoring non-translation directory:', relativePath);\n return;\n }\n\n const newFilePath = path.join(relativePath, devTranslationFileName);\n if (!existsSync(newFilePath)) {\n await fs.writeFile(newFilePath, JSON.stringify({}, null, 2));\n trace('Created new empty translation file:', newFilePath);\n } else {\n trace(\n `New directory already contains translation file. Skipping creation. Existing file ${newFilePath}`,\n );\n }\n };\n\n watcher.on('addDir', onNewDirectory);\n watcher.on('add', onTranslationChange).on('change', onTranslationChange);\n\n return () => watcher.close();\n}\n\nexport async function compile(\n { watch: shouldWatch = false } = {},\n config: UserConfig,\n) {\n const translations = await loadAllTranslations(\n { fallbacks: 'all', includeNodeModules: false },\n config,\n );\n\n for (const loadedTranslation of translations) {\n await generateRuntime(loadedTranslation);\n }\n\n if (shouldWatch) {\n trace('Listening for changes to files...');\n return watch(config);\n }\n}\n\nasync function writeIfChanged(filepath: string, contents: string) {\n let hasChanged = true;\n\n try {\n const existingContents = await fs.readFile(filepath, { encoding: 'utf-8' });\n\n hasChanged = existingContents !== contents;\n } catch {\n // ignore error, likely a file doesn't exist error so we want to write anyway\n }\n\n if (hasChanged) {\n await fs.writeFile(filepath, contents, { encoding: 'utf-8' });\n }\n}\n","/* eslint-disable no-console */\nimport type { UserConfig, LoadedTranslation, LanguageName } from '../types';\nimport pc from 'picocolors';\n\nimport { loadAllTranslations } from '../load-translations';\nimport { getAltLanguages } from '../utils';\n\nexport function findMissingKeys(\n loadedTranslation: LoadedTranslation,\n devLanguageName: LanguageName,\n altLanguages: LanguageName[],\n) {\n const devLanguage = loadedTranslation.languages[devLanguageName];\n\n if (!devLanguage) {\n throw new Error(\n `Failed to load dev language: ${loadedTranslation.filePath}`,\n );\n }\n\n const result: Record<LanguageName, string[]> = {};\n let valid = true;\n\n const requiredKeys = Object.keys(devLanguage);\n\n if (requiredKeys.length > 0) {\n for (const altLanguageName of altLanguages) {\n const altLanguage = loadedTranslation.languages[altLanguageName] ?? {};\n\n for (const key of requiredKeys) {\n if (typeof altLanguage[key]?.message !== 'string') {\n if (!result[altLanguageName]) {\n result[altLanguageName] = [];\n }\n\n result[altLanguageName].push(key);\n valid = false;\n }\n }\n }\n }\n return [valid, result] as const;\n}\n\nexport async function validate(config: UserConfig) {\n const allTranslations = await loadAllTranslations(\n { fallbacks: 'valid', includeNodeModules: true },\n config,\n );\n\n let valid = true;\n\n for (const loadedTranslation of allTranslations) {\n const [translationValid, result] = findMissingKeys(\n loadedTranslation,\n config.devLanguage,\n getAltLanguages(config),\n );\n\n if (!translationValid) {\n valid = false;\n console.log(\n pc.red(\n `Incomplete translations: \"${pc.bold(\n loadedTranslation.relativePath,\n )}\"`,\n ),\n );\n\n for (const lang of Object.keys(result)) {\n const missingKeys = result[lang];\n\n console.log(\n pc.yellow(lang),\n '->',\n missingKeys.map((v) => `\"${v}\"`).join(', '),\n );\n }\n }\n }\n\n return valid;\n}\n","export class ValidationError extends Error {\n code: string;\n rawMessage: string;\n constructor(code: string, message: string) {\n super(`Invalid vocab.config.js: ${code} - ${message}`);\n this.code = code;\n this.rawMessage = message;\n }\n}\n","import type { UserConfig } from './types';\nimport path from 'path';\n\nimport pc from 'picocolors';\nimport findUp from 'find-up';\nimport Validator from 'fastest-validator';\nimport { ValidationError } from './ValidationError';\nimport { trace } from './logger';\n\nconst boldCyan = (s: string) => pc.bold(pc.cyan(s));\n\nconst validator = new Validator();\nconst schema = {\n $$strict: true,\n devLanguage: {\n type: 'string',\n },\n languages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n },\n },\n },\n generatedLanguages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n generator: {\n type: 'object',\n props: {\n transformElement: { type: 'function', optional: true },\n transformMessage: { type: 'function', optional: true },\n },\n },\n },\n },\n optional: true,\n },\n translationsDirectorySuffix: { type: 'string', optional: true },\n projectRoot: { type: 'string', optional: true },\n ignore: {\n type: 'array',\n items: 'string',\n optional: true,\n },\n};\nconst checkConfigFile = validator.compile(schema);\n\nconst splitMap = (message: string, callback: (value: string) => string) =>\n message\n .split(' ,')\n .map((v) => callback(v))\n .join(' ,');\n\nexport function validateConfig(c: UserConfig) {\n trace('Validating configuration file');\n // Note: checkConfigFile mutates the config file by applying defaults\n const isValid = checkConfigFile(c);\n if (isValid !== true) {\n throw new ValidationError(\n 'InvalidStructure',\n (Array.isArray(isValid) ? isValid : [])\n .map((v) => {\n if (v.type === 'objectStrict') {\n return `Invalid key(s) ${splitMap(\n v.actual,\n (m) => `\"${pc.cyan(m)}\"`,\n )}. Expected one of ${splitMap(v.expected, pc.green)}`;\n }\n if (v.field) {\n return v.message?.replace(v.field, pc.cyan(v.field));\n }\n return v.message;\n })\n .join(' \\n'),\n );\n }\n\n const languageStrings = c.languages.map((v) => v.name);\n\n // Dev Language should exist in languages\n if (!languageStrings.includes(c.devLanguage)) {\n throw new ValidationError(\n 'InvalidDevLanguage',\n `The dev language \"${boldCyan(\n c.devLanguage,\n )}\" was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n\n const foundLanguages: string[] = [];\n for (const lang of c.languages) {\n // Languages must only exist once\n if (foundLanguages.includes(lang.name)) {\n throw new ValidationError(\n 'DuplicateLanguage',\n `The language \"${boldCyan(lang.name)}\" was defined multiple times.`,\n );\n }\n foundLanguages.push(lang.name);\n\n // Any extends must be in languages\n if (lang.extends && !languageStrings.includes(lang.extends)) {\n throw new ValidationError(\n 'InvalidExtends',\n `The language \"${boldCyan(lang.name)}\"'s extends of ${boldCyan(\n lang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n const foundGeneratedLanguages: string[] = [];\n for (const generatedLang of c.generatedLanguages || []) {\n // Generated languages must only exist once\n if (foundGeneratedLanguages.includes(generatedLang.name)) {\n throw new ValidationError(\n 'DuplicateGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" was defined multiple times.`,\n );\n }\n foundGeneratedLanguages.push(generatedLang.name);\n\n // Generated language names must not conflict with language names\n if (languageStrings.includes(generatedLang.name)) {\n throw new ValidationError(\n 'InvalidGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" is already defined as a language.`,\n );\n }\n\n // Any extends must be in languages\n if (\n generatedLang.extends &&\n !languageStrings.includes(generatedLang.extends)\n ) {\n throw new ValidationError(\n 'InvalidExtends',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\"'s extends of ${boldCyan(\n generatedLang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n trace('Configuration file is valid');\n\n return true;\n}\n\nfunction createConfig(configFilePath: string) {\n const cwd = path.dirname(configFilePath);\n\n return {\n projectRoot: cwd,\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n ...(require(configFilePath) as UserConfig),\n };\n}\n\nexport async function resolveConfig(\n customConfigFilePath?: string,\n): Promise<UserConfig | null> {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : await findUp(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n return null;\n}\n\nexport function resolveConfigSync(\n customConfigFilePath?: string,\n): UserConfig | null {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : findUp.sync(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n\n return null;\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","names":["fs"],"sources":["../src/logger.ts","../src/utils.ts","../src/generate-language.ts","../src/load-translations.ts","../src/compile.ts","../src/validate/index.ts","../src/ValidationError.ts","../src/config.ts"],"sourcesContent":["import pc from 'picocolors';\nimport debug from 'debug';\n\nexport const trace = debug(`vocab:core`);\n\nexport const log = (...params: unknown[]) => {\n // eslint-disable-next-line no-console\n console.log(pc.yellow('Vocab'), ...params);\n};\n","import path from 'path';\n\nimport type {\n LanguageName,\n LanguageTarget,\n TranslationsByKey,\n TranslationMessagesByKey,\n UserConfig,\n} from './types';\nimport { trace } from './logger';\n\nexport const defaultTranslationDirSuffix = '.vocab';\nexport const devTranslationFileName = 'translations.json';\n\nexport type Fallback = 'none' | 'valid' | 'all';\n\nconst globAnyPathWithOptionalPrefix = '**/?(*)';\n\nexport const compiledVocabFileFilter = /\\.vocab[\\\\/]index\\.(?:ts|js|cjs|mjs)$/;\n\nexport function isDevLanguageFile(filePath: string) {\n return (\n filePath.endsWith(`/${devTranslationFileName}`) ||\n filePath === devTranslationFileName\n );\n}\nexport function isAltLanguageFile(filePath: string) {\n return filePath.endsWith('.translations.json');\n}\nexport function isTranslationDirectory(\n filePath: string,\n {\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n }: {\n translationsDirectorySuffix?: string;\n },\n) {\n return filePath.endsWith(translationsDirectorySuffix);\n}\n\nexport function getDevTranslationFileGlob({\n translationsDirectorySuffix = defaultTranslationDirSuffix,\n}: {\n translationsDirectorySuffix?: string;\n}) {\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${devTranslationFileName}`;\n\n trace('getDevTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltTranslationFileGlob(config: UserConfig) {\n const altLanguages = getAltLanguages(config);\n const langMatch =\n altLanguages.length === 1 ? altLanguages[0] : `{${altLanguages.join(',')}}`;\n\n const { translationsDirectorySuffix = defaultTranslationDirSuffix } = config;\n const result = `${globAnyPathWithOptionalPrefix}${translationsDirectorySuffix}/${langMatch}.${devTranslationFileName}`;\n\n trace('getAltTranslationFileGlob', result);\n\n return result;\n}\n\nexport function getAltLanguages({\n devLanguage,\n languages,\n}: {\n devLanguage: LanguageName;\n languages: LanguageTarget[];\n}) {\n return languages.map((v) => v.name).filter((lang) => lang !== devLanguage);\n}\n\nexport function getDevLanguageFileFromTsFile(tsFilePath: string) {\n const directory = path.dirname(tsFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n\n trace(`Returning dev language path ${result} for path ${tsFilePath}`);\n return result;\n}\n\nexport function getDevLanguageFileFromAltLanguageFile(\n altLanguageFilePath: string,\n) {\n const directory = path.dirname(altLanguageFilePath);\n const result = path.normalize(path.join(directory, devTranslationFileName));\n trace(\n `Returning dev language path ${result} for path ${altLanguageFilePath}`,\n );\n return result;\n}\n\nexport function getTSFileFromDevLanguageFile(devLanguageFilePath: string) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(path.join(directory, 'index.ts'));\n\n trace(`Returning TS path ${result} for path ${devLanguageFilePath}`);\n return result;\n}\n\nexport function getAltLanguageFilePath(\n devLanguageFilePath: string,\n language: string,\n) {\n const directory = path.dirname(devLanguageFilePath);\n const result = path.normalize(\n path.join(directory, `${language}.translations.json`),\n );\n trace(\n `Returning alt language path ${result} for path ${devLanguageFilePath}`,\n );\n return path.normalize(result);\n}\n\nexport function mapValues<Key extends string, OriginalValue, ReturnValue>(\n obj: Record<Key, OriginalValue>,\n func: (val: OriginalValue) => ReturnValue,\n): TranslationMessagesByKey<Key> {\n const newObj: any = {};\n const keys = Object.keys(obj) as Key[];\n for (const key of keys) {\n newObj[key] = func(obj[key]);\n }\n return newObj;\n}\n\nexport function getTranslationMessages<Key extends string>(\n translations: TranslationsByKey<Key>,\n): TranslationMessagesByKey<Key> {\n return mapValues(translations, (v) => v.message);\n}\n","import { IntlMessageFormat } from 'intl-messageformat';\nimport {\n type MessageFormatElement,\n type PluralOrSelectOption,\n TYPE,\n} from '@formatjs/icu-messageformat-parser';\nimport { printAST } from '@formatjs/icu-messageformat-parser/printer.js';\nimport type { MessageGenerator, TranslationsByKey } from './types';\n\nexport function generateLanguageFromTranslations({\n baseTranslations,\n generator,\n}: {\n baseTranslations: TranslationsByKey<string>;\n generator: MessageGenerator;\n}): TranslationsByKey<string> {\n if (!generator.transformElement && !generator.transformMessage) {\n return baseTranslations;\n }\n\n const translationKeys = Object.keys(baseTranslations);\n const generatedTranslations: TranslationsByKey<string> = {};\n\n for (const translationKey of translationKeys) {\n const translation = baseTranslations[translationKey];\n let transformedMessage = translation.message;\n\n if (generator.transformElement) {\n const messageAst = new IntlMessageFormat(translation.message).getAst();\n const transformedAst = messageAst.map(\n transformMessageFormatElement(generator.transformElement),\n );\n transformedMessage = printAST(transformedAst);\n }\n\n if (generator.transformMessage) {\n transformedMessage = generator.transformMessage(transformedMessage);\n }\n\n generatedTranslations[translationKey] = {\n message: transformedMessage,\n };\n }\n\n return generatedTranslations;\n}\n\nfunction transformMessageFormatElement(\n transformElement: (message: string) => string,\n): (messageFormatElement: MessageFormatElement) => MessageFormatElement {\n return (messageFormatElement: MessageFormatElement) => {\n const transformedMessageFormatElement = { ...messageFormatElement };\n\n switch (transformedMessageFormatElement.type) {\n case TYPE.literal:\n const transformedValue = transformElement(\n transformedMessageFormatElement.value,\n );\n transformedMessageFormatElement.value = transformedValue;\n break;\n\n case TYPE.select:\n case TYPE.plural:\n const transformedOptions: Record<string, PluralOrSelectOption> = {\n ...transformedMessageFormatElement.options,\n };\n\n for (const key of Object.keys(transformedOptions)) {\n transformedOptions[key].value = transformedOptions[key].value.map(\n transformMessageFormatElement(transformElement),\n );\n }\n\n break;\n\n case TYPE.tag:\n const transformedChildren =\n transformedMessageFormatElement.children.map(\n transformMessageFormatElement(transformElement),\n );\n transformedMessageFormatElement.children = transformedChildren;\n break;\n\n default:\n break;\n }\n\n return transformedMessageFormatElement;\n };\n}\n","import path from 'path';\n\nimport { glob } from 'tinyglobby';\nimport type {\n TranslationsByKey,\n UserConfig,\n LoadedTranslation,\n LanguageTarget,\n LanguageName,\n TranslationFileMetadata,\n TranslationFileContents,\n} from './types';\nimport pc from 'picocolors';\n\nimport { trace } from './logger';\nimport {\n defaultTranslationDirSuffix,\n type Fallback,\n getAltLanguageFilePath,\n getAltLanguages,\n getDevTranslationFileGlob,\n} from './utils';\nimport { generateLanguageFromTranslations } from './generate-language';\n\nexport function getUniqueKey(key: string, namespace: string) {\n return `${key}.${namespace}`;\n}\n\nexport function mergeWithDevLanguageTranslation({\n translation,\n devTranslation,\n}: {\n translation: TranslationsByKey;\n devTranslation: TranslationsByKey;\n}) {\n // Only use keys from the dev translation\n const keys = Object.keys(devTranslation);\n const newLanguage: TranslationsByKey = {};\n\n for (const key of keys) {\n if (translation[key]) {\n newLanguage[key] = {\n message: translation[key].message,\n description: devTranslation[key].description,\n };\n }\n }\n\n return newLanguage;\n}\n\nfunction getLanguageFallbacks({ languages }: { languages: LanguageTarget[] }) {\n const languageFallbackMap = new Map<LanguageName, LanguageName>();\n\n for (const lang of languages) {\n if (lang.extends) {\n languageFallbackMap.set(lang.name, lang.extends);\n }\n }\n\n return languageFallbackMap;\n}\n\nexport function getLanguageHierarchy({\n languages,\n}: {\n languages: LanguageTarget[];\n}) {\n const hierarchyMap = new Map<LanguageName, LanguageName[]>();\n const fallbacks = getLanguageFallbacks({ languages });\n\n for (const lang of languages) {\n const langHierarchy = [];\n let currLang = lang.extends;\n\n while (currLang) {\n langHierarchy.push(currLang);\n\n currLang = fallbacks.get(currLang);\n }\n\n hierarchyMap.set(lang.name, langHierarchy);\n }\n\n return hierarchyMap;\n}\n\nexport function getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n}: {\n languages: LanguageTarget[];\n languageName: string;\n devLanguage: string;\n fallbacks: Fallback;\n}) {\n const languageHierarchy = getLanguageHierarchy({ languages }).get(\n languageName,\n );\n\n if (!languageHierarchy) {\n throw new Error(`Missing language hierarchy for ${languageName}`);\n }\n\n const fallbackLanguageOrder: string[] = [languageName];\n\n if (fallbacks !== 'none') {\n fallbackLanguageOrder.unshift(...languageHierarchy.reverse());\n\n if (fallbacks === 'all' && fallbackLanguageOrder[0] !== devLanguage) {\n fallbackLanguageOrder.unshift(devLanguage);\n }\n }\n\n return fallbackLanguageOrder;\n}\n\nfunction getNamespaceByFilePath(\n relativePath: string,\n { translationsDirectorySuffix = defaultTranslationDirSuffix }: UserConfig,\n) {\n let namespace = path\n .dirname(relativePath)\n .replace(/^src\\//, '')\n .replace(/\\//g, '_');\n\n if (namespace.endsWith(translationsDirectorySuffix)) {\n namespace = namespace.slice(0, -translationsDirectorySuffix.length);\n }\n\n return namespace;\n}\n\nfunction printValidationError(...params: unknown[]) {\n // eslint-disable-next-line no-console\n console.error(pc.red('Error loading translation:'), ...params);\n}\n\nfunction getTranslationsFromFile(\n translationFileContents: unknown,\n {\n isAltLanguage,\n filePath,\n withTags,\n }: { isAltLanguage: boolean; filePath: string; withTags?: boolean },\n): {\n $namespace: unknown;\n keys: TranslationsByKey;\n metadata: TranslationFileMetadata;\n} {\n if (!translationFileContents || typeof translationFileContents !== 'object') {\n throw new Error(\n `Unable to read translation file ${filePath}. Translations must be an object.`,\n );\n }\n\n const { $namespace, _meta, ...keys } =\n translationFileContents as TranslationFileContents;\n\n if (isAltLanguage && $namespace) {\n printValidationError(\n `Found $namespace in alt language file in ${filePath}. $namespace is only used in the dev language and will be ignored.`,\n );\n }\n\n if (!isAltLanguage && $namespace && typeof $namespace !== 'string') {\n printValidationError(\n `Found non-string $namespace in language file in ${filePath}. $namespace must be a string.`,\n );\n }\n\n if (isAltLanguage && _meta?.tags) {\n printValidationError(\n `Found _meta.tags in alt language file in ${filePath}. _meta.tags is only used in the dev language and will be ignored.`,\n );\n }\n\n // Never return tags if we're fetching translations for an alt language\n const includeTags = !isAltLanguage && withTags;\n const validKeys: TranslationsByKey = {};\n\n for (const [translationKey, { tags, ...translation }] of Object.entries(\n keys,\n )) {\n if (typeof translation === 'string') {\n printValidationError(\n `Found string for a translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation) {\n printValidationError(\n `Found empty translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n if (!translation.message || typeof translation.message !== 'string') {\n printValidationError(\n `No message found for translation \"${translationKey}\" in ${filePath}. Translation must be an object of the format {message: string}.`,\n );\n continue;\n }\n\n validKeys[translationKey] = {\n ...translation,\n tags: includeTags ? tags : undefined,\n };\n }\n\n const metadata = { tags: includeTags ? _meta?.tags : undefined };\n\n return { $namespace, keys: validKeys, metadata };\n}\n\nexport function loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation,\n fallbacks,\n }: {\n filePath: string;\n languageName: string;\n devTranslation: TranslationsByKey;\n fallbacks: Fallback;\n },\n { devLanguage, languages }: UserConfig,\n): TranslationsByKey {\n const altLanguageTranslation = {};\n\n const fallbackLanguageOrder = getFallbackLanguageOrder({\n languages,\n languageName,\n devLanguage,\n fallbacks,\n });\n\n trace(\n `Loading alt language file with precedence: ${fallbackLanguageOrder\n .slice()\n .reverse()\n .join(' -> ')}`,\n );\n\n for (const fallbackLanguage of fallbackLanguageOrder) {\n if (fallbackLanguage !== devLanguage) {\n try {\n const altFilePath = getAltLanguageFilePath(filePath, fallbackLanguage);\n delete require.cache[altFilePath];\n\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationFile = require(altFilePath);\n const { keys: fallbackLanguageTranslation } = getTranslationsFromFile(\n translationFile,\n {\n filePath: altFilePath,\n isAltLanguage: true,\n },\n );\n Object.assign(\n altLanguageTranslation,\n mergeWithDevLanguageTranslation({\n translation: fallbackLanguageTranslation,\n devTranslation,\n }),\n );\n } catch {\n trace(`Missing alt language file ${getAltLanguageFilePath(\n filePath,\n fallbackLanguage,\n )}\n `);\n }\n } else {\n Object.assign(altLanguageTranslation, devTranslation);\n }\n }\n\n return altLanguageTranslation;\n}\n\nfunction stripTagsFromTranslations(translations: TranslationsByKey) {\n return Object.fromEntries(\n Object.entries(translations).map(([key, { tags, ...rest }]) => [key, rest]),\n );\n}\n\nexport function loadTranslation(\n {\n filePath,\n fallbacks,\n withTags,\n }: {\n filePath: string;\n fallbacks: Fallback;\n withTags?: boolean;\n },\n userConfig: UserConfig,\n): LoadedTranslation {\n trace(\n `Loading translation file in \"${fallbacks}\" fallback mode: \"${filePath}\"`,\n );\n\n const languageSet: Record<\n string,\n Record<string, { message: string; description?: string | undefined }>\n > = {};\n\n delete require.cache[filePath];\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n const translationContent = require(filePath);\n const relativePath = path.relative(\n userConfig.projectRoot || process.cwd(),\n filePath,\n );\n const {\n $namespace,\n keys: devTranslation,\n metadata,\n } = getTranslationsFromFile(translationContent, {\n filePath,\n isAltLanguage: false,\n withTags,\n });\n const namespace: string =\n typeof $namespace === 'string'\n ? $namespace\n : getNamespaceByFilePath(relativePath, userConfig);\n\n trace(`Found file ${filePath}. Using namespace ${namespace}`);\n\n languageSet[userConfig.devLanguage] = devTranslation;\n\n const devTranslationNoTags = withTags\n ? stripTagsFromTranslations(devTranslation)\n : devTranslation;\n const altLanguages = getAltLanguages(userConfig);\n for (const languageName of altLanguages) {\n languageSet[languageName] = loadAltLanguageFile(\n {\n filePath,\n languageName,\n devTranslation: devTranslationNoTags,\n fallbacks,\n },\n userConfig,\n );\n }\n\n for (const generatedLanguage of userConfig.generatedLanguages || []) {\n const { name: generatedLanguageName, generator } = generatedLanguage;\n const baseLanguage = generatedLanguage.extends || userConfig.devLanguage;\n const baseTranslations = languageSet[baseLanguage];\n\n languageSet[generatedLanguageName] = generateLanguageFromTranslations({\n baseTranslations,\n generator,\n });\n }\n\n return {\n filePath,\n keys: Object.keys(devTranslation),\n namespace,\n relativePath,\n languages: languageSet,\n metadata,\n };\n}\n\nexport async function loadAllTranslations(\n {\n fallbacks,\n includeNodeModules,\n withTags,\n }: { fallbacks: Fallback; includeNodeModules: boolean; withTags?: boolean },\n config: UserConfig,\n): Promise<LoadedTranslation[]> {\n const { projectRoot, ignore = [] } = config;\n\n const translationFiles = await glob(getDevTranslationFileGlob(config), {\n ignore: includeNodeModules ? ignore : [...ignore, '**/node_modules/**'],\n dot: true,\n absolute: true,\n cwd: projectRoot,\n expandDirectories: false,\n });\n\n trace(`Found ${translationFiles.length} translation files`);\n\n const loadedTranslations: LoadedTranslation[] = [];\n const keys = new Set<string>();\n\n for (const translationFile of translationFiles) {\n const loadedTranslation = loadTranslation(\n { filePath: translationFile, fallbacks, withTags },\n config,\n );\n\n loadedTranslations.push(loadedTranslation);\n\n for (const key of loadedTranslation.keys) {\n const uniqueKey = getUniqueKey(key, loadedTranslation.namespace);\n if (keys.has(uniqueKey)) {\n trace(`Duplicate keys found`);\n throw new Error(\n `Duplicate keys found. Key with namespace ${loadedTranslation.namespace} and key ${key} was found multiple times.`,\n );\n }\n keys.add(uniqueKey);\n\n const globalKey =\n loadedTranslation.languages[config.devLanguage][key].globalKey;\n if (globalKey) {\n if (keys.has(globalKey)) {\n throw new Error(\n `Duplicate keys found. Key with global key ${globalKey} and key ${key} was found multiple times`,\n );\n }\n keys.add(globalKey);\n }\n }\n }\n\n return loadedTranslations;\n}\n","import { promises as fs, existsSync } from 'fs';\nimport path from 'path';\nimport pm from 'picomatch';\n\nimport type { LoadedTranslation, UserConfig } from './types';\nimport {\n isArgumentElement,\n isDateElement,\n isNumberElement,\n isPluralElement,\n isSelectElement,\n isTagElement,\n isTimeElement,\n type MessageFormatElement,\n parse,\n} from '@formatjs/icu-messageformat-parser';\nimport prettier from 'prettier';\n\nimport {\n getTranslationMessages,\n getDevTranslationFileGlob,\n getTSFileFromDevLanguageFile,\n getDevLanguageFileFromAltLanguageFile,\n getAltTranslationFileGlob,\n isDevLanguageFile,\n isAltLanguageFile,\n devTranslationFileName,\n isTranslationDirectory,\n} from './utils';\nimport { trace } from './logger';\nimport { loadAllTranslations, loadTranslation } from './load-translations';\n\ntype ICUParams = Record<string, string>;\n\ninterface TranslationTypeInfo {\n params: ICUParams;\n message: string;\n hasTags: boolean;\n}\n\nfunction extractHasTags(ast: MessageFormatElement[]): boolean {\n return ast.some((element) => {\n if (isSelectElement(element) || isPluralElement(element)) {\n const children = Object.values(element.options).map((o) => o.value);\n return children.some((child) => extractHasTags(child));\n }\n\n return isTagElement(element);\n });\n}\n\nfunction extractParamTypes(\n ast: MessageFormatElement[],\n currentParams: ICUParams,\n): [params: ICUParams, vocabTypesImports: Set<string>] {\n let params = { ...currentParams };\n let vocabTypesImports = new Set<string>();\n\n for (const element of ast) {\n if (isArgumentElement(element)) {\n if (!(element.value in params)) {\n // Preserve existing types for parameters that we've already parsed\n // This applies to self-referential parameters, for example `{foo, plural, 1 {{foo} thing} other {{foo} things}}`\n params[element.value] = 'string';\n }\n } else if (isNumberElement(element)) {\n params[element.value] = 'number';\n } else if (isPluralElement(element)) {\n params[element.value] = 'number';\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n } else if (isDateElement(element) || isTimeElement(element)) {\n params[element.value] = 'Date | number';\n } else if (isTagElement(element)) {\n params[element.value] = 'FormatXMLElementFn<T>';\n vocabTypesImports.add('FormatXMLElementFn');\n\n const [newParams, subImports] = extractParamTypes(\n element.children,\n params,\n );\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n } else if (isSelectElement(element)) {\n const options = Object.keys(element.options);\n\n // `other` will always be an option as the parser enforces this by default\n const nonOtherOptions = options.filter((o) => o !== 'other');\n const nonOtherOptionsUnion = nonOtherOptions\n .map((o) => `'${o}'`)\n .join(' | ');\n\n params[element.value] = `StringWithSuggestions<${nonOtherOptionsUnion}>`;\n vocabTypesImports.add('StringWithSuggestions');\n\n const children = Object.values(element.options).map((o) => o.value);\n\n for (const child of children) {\n const [newParams, subImports] = extractParamTypes(child, params);\n\n vocabTypesImports = new Set([...vocabTypesImports, ...subImports]);\n params = newParams;\n }\n }\n }\n\n return [params, vocabTypesImports];\n}\n\nfunction serialiseObjectToType(v: any) {\n let result = '';\n\n for (const [key, value] of Object.entries(v)) {\n result += `${JSON.stringify(key)}: ${\n value && typeof value === 'object' ? serialiseObjectToType(value) : value\n },`;\n }\n\n return `{ ${result} }`;\n}\n\nconst serializeTypeImports = (imports: Set<string>, moduleName: string) => {\n if (imports.size === 0) {\n return '';\n }\n\n const importNames = Array.from(imports).sort();\n return `import type { ${importNames.join(', ')} } from '${moduleName}';`;\n};\n\nfunction serialiseTranslationRuntime(\n value: Map<string, TranslationTypeInfo>,\n imports: Set<string>,\n loadedTranslation: LoadedTranslation,\n) {\n trace('Serialising translations:', loadedTranslation);\n const translationsType: any = {};\n\n for (const [key, { params, message, hasTags }] of value.entries()) {\n let translationFunctionString = `() => ${message}`;\n\n if (Object.keys(params).length > 0) {\n const formatGeneric = hasTags ? '<T = string>' : '';\n const formatReturn =\n hasTags && imports.has('FormatXMLElementFn')\n ? 'ReturnType<FormatXMLElementFn<T>>'\n : 'string';\n translationFunctionString = `${formatGeneric}(values: ${serialiseObjectToType(\n params,\n )}) => ${formatReturn}`;\n }\n\n translationsType[key] = translationFunctionString;\n }\n\n const languagesUnionAsString = Object.keys(loadedTranslation.languages)\n .map((l) => `'${l}'`)\n .join(' | ');\n const languageEntries = Object.entries(loadedTranslation.languages)\n .map(\n ([languageName, translations]) =>\n `${JSON.stringify(languageName)}: createLanguage(${JSON.stringify(\n getTranslationMessages(translations),\n )})`,\n )\n .join(',');\n\n // TranslationFile is now always imported\n const typeImports = new Set([...imports, 'TranslationFile']);\n\n return /* ts */ `\n // This file is automatically generated by Vocab.\n // To make changes update translation.json files directly.\n\n ${serializeTypeImports(typeImports, '@vocab/core')}\n import { createLanguage, createTranslationFile } from '@vocab/core/runtime';\n\n const translations: TranslationFile<\n ${languagesUnionAsString},\n ${serialiseObjectToType(translationsType)}\n > = createTranslationFile\n ({\n ${languageEntries}\n });\n\n export default translations;\n `;\n}\n\nexport async function generateRuntime(loadedTranslation: LoadedTranslation) {\n const { languages: loadedLanguages, filePath } = loadedTranslation;\n\n trace('Generating types for', filePath);\n const translationTypes = new Map<string, TranslationTypeInfo>();\n\n let imports = new Set<string>();\n\n for (const key of loadedTranslation.keys) {\n let params: ICUParams = {};\n const messages = new Set();\n let hasTags = false;\n\n for (const translatedLanguage of Object.values(loadedLanguages)) {\n if (translatedLanguage[key]) {\n const ast = parse(translatedLanguage[key].message);\n\n hasTags = hasTags || extractHasTags(ast);\n\n const [parsedParams, vocabTypesImports] = extractParamTypes(\n ast,\n params,\n );\n\n imports = new Set([...imports, ...vocabTypesImports]);\n params = parsedParams;\n\n messages.add(JSON.stringify(translatedLanguage[key].message));\n }\n }\n\n translationTypes.set(key, {\n params,\n hasTags,\n message: Array.from(messages).join(' | '),\n });\n }\n\n const prettierConfig = await prettier.resolveConfig(filePath);\n const serializedTranslationType = serialiseTranslationRuntime(\n translationTypes,\n imports,\n loadedTranslation,\n );\n const declaration = await prettier.format(serializedTranslationType, {\n ...prettierConfig,\n parser: 'typescript',\n });\n const outputFilePath = getTSFileFromDevLanguageFile(filePath);\n trace(`Writing translation types to ${outputFilePath}`);\n await writeIfChanged(outputFilePath, declaration);\n}\n\nexport async function watch(config: UserConfig) {\n const cwd = config.projectRoot || process.cwd();\n\n const ignorePatterns = [\n ...(config.ignore || []),\n '**/node_modules/**',\n '**/.git/**',\n ];\n\n const ignoredFunction = (pathToCheck: string) =>\n pm.isMatch(pathToCheck, ignorePatterns);\n\n const chokidar = await import('chokidar');\n const watcher = chokidar.watch(cwd, {\n ignored: ignoredFunction,\n ignoreInitial: true,\n });\n\n const devFileGlob = getDevTranslationFileGlob(config);\n const altFileGlob = getAltTranslationFileGlob(config);\n\n const onTranslationChange = async (relativePath: string) => {\n const matchesTranslationFiles = pm.isMatch(relativePath, [\n devFileGlob,\n altFileGlob,\n ]);\n\n if (!matchesTranslationFiles) {\n return;\n }\n\n trace(`Detected change for file ${relativePath}`);\n\n let targetFile;\n\n if (isDevLanguageFile(relativePath)) {\n targetFile = path.resolve(cwd, relativePath);\n } else if (isAltLanguageFile(relativePath)) {\n targetFile = getDevLanguageFileFromAltLanguageFile(\n path.resolve(cwd, relativePath),\n );\n }\n\n if (targetFile) {\n watcher.add(targetFile);\n try {\n const loadedTranslation = loadTranslation(\n { filePath: targetFile, fallbacks: 'all' },\n config,\n );\n\n await generateRuntime(loadedTranslation);\n } catch (e) {\n // eslint-disable-next-line no-console\n console.log('Failed to generate types for', relativePath);\n // eslint-disable-next-line no-console\n console.error(e);\n }\n }\n };\n\n const onNewDirectory = async (relativePath: string) => {\n trace('Detected new directory', relativePath);\n\n if (!isTranslationDirectory(relativePath, config)) {\n trace('Ignoring non-translation directory:', relativePath);\n return;\n }\n\n const newFilePath = path.join(relativePath, devTranslationFileName);\n if (!existsSync(newFilePath)) {\n await fs.writeFile(newFilePath, JSON.stringify({}, null, 2));\n trace('Created new empty translation file:', newFilePath);\n } else {\n trace(\n `New directory already contains translation file. Skipping creation. Existing file ${newFilePath}`,\n );\n }\n };\n\n watcher.on('addDir', onNewDirectory);\n watcher.on('add', onTranslationChange).on('change', onTranslationChange);\n\n return () => watcher.close();\n}\n\nexport async function compile(\n { watch: shouldWatch = false } = {},\n config: UserConfig,\n) {\n const translations = await loadAllTranslations(\n { fallbacks: 'all', includeNodeModules: false },\n config,\n );\n\n for (const loadedTranslation of translations) {\n await generateRuntime(loadedTranslation);\n }\n\n if (shouldWatch) {\n trace('Listening for changes to files...');\n return watch(config);\n }\n}\n\nasync function writeIfChanged(filepath: string, contents: string) {\n let hasChanged = true;\n\n try {\n const existingContents = await fs.readFile(filepath, { encoding: 'utf-8' });\n\n hasChanged = existingContents !== contents;\n } catch {\n // ignore error, likely a file doesn't exist error so we want to write anyway\n }\n\n if (hasChanged) {\n await fs.writeFile(filepath, contents, { encoding: 'utf-8' });\n }\n}\n","/* eslint-disable no-console */\nimport type { UserConfig, LoadedTranslation, LanguageName } from '../types';\nimport pc from 'picocolors';\n\nimport { loadAllTranslations } from '../load-translations';\nimport { getAltLanguages } from '../utils';\n\nexport function findMissingKeys(\n loadedTranslation: LoadedTranslation,\n devLanguageName: LanguageName,\n altLanguages: LanguageName[],\n) {\n const devLanguage = loadedTranslation.languages[devLanguageName];\n\n if (!devLanguage) {\n throw new Error(\n `Failed to load dev language: ${loadedTranslation.filePath}`,\n );\n }\n\n const result: Record<LanguageName, string[]> = {};\n let valid = true;\n\n const requiredKeys = Object.keys(devLanguage);\n\n if (requiredKeys.length > 0) {\n for (const altLanguageName of altLanguages) {\n const altLanguage = loadedTranslation.languages[altLanguageName] ?? {};\n\n for (const key of requiredKeys) {\n if (typeof altLanguage[key]?.message !== 'string') {\n if (!result[altLanguageName]) {\n result[altLanguageName] = [];\n }\n\n result[altLanguageName].push(key);\n valid = false;\n }\n }\n }\n }\n return [valid, result] as const;\n}\n\nexport async function validate(config: UserConfig) {\n const allTranslations = await loadAllTranslations(\n { fallbacks: 'valid', includeNodeModules: true },\n config,\n );\n\n let valid = true;\n\n for (const loadedTranslation of allTranslations) {\n const [translationValid, result] = findMissingKeys(\n loadedTranslation,\n config.devLanguage,\n getAltLanguages(config),\n );\n\n if (!translationValid) {\n valid = false;\n console.log(\n pc.red(\n `Incomplete translations: \"${pc.bold(\n loadedTranslation.relativePath,\n )}\"`,\n ),\n );\n\n for (const lang of Object.keys(result)) {\n const missingKeys = result[lang];\n\n console.log(\n pc.yellow(lang),\n '->',\n missingKeys.map((v) => `\"${v}\"`).join(', '),\n );\n }\n }\n }\n\n return valid;\n}\n","export class ValidationError extends Error {\n code: string;\n rawMessage: string;\n constructor(code: string, message: string) {\n super(`Invalid vocab.config.js: ${code} - ${message}`);\n this.code = code;\n this.rawMessage = message;\n }\n}\n","import type { UserConfig } from './types';\nimport path from 'path';\n\nimport pc from 'picocolors';\nimport findUp from 'find-up';\nimport Validator from 'fastest-validator';\nimport { ValidationError } from './ValidationError';\nimport { trace } from './logger';\n\nconst boldCyan = (s: string) => pc.bold(pc.cyan(s));\n\nconst validator = new Validator();\nconst schema = {\n $$strict: true,\n devLanguage: {\n type: 'string',\n },\n languages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n },\n },\n },\n generatedLanguages: {\n type: 'array',\n items: {\n type: 'object',\n props: {\n name: { type: 'string' },\n extends: { type: 'string', optional: true },\n generator: {\n type: 'object',\n props: {\n transformElement: { type: 'function', optional: true },\n transformMessage: { type: 'function', optional: true },\n },\n },\n },\n },\n optional: true,\n },\n translationsDirectorySuffix: { type: 'string', optional: true },\n projectRoot: { type: 'string', optional: true },\n ignore: {\n type: 'array',\n items: 'string',\n optional: true,\n },\n};\nconst checkConfigFile = validator.compile(schema);\n\nconst splitMap = (message: string, callback: (value: string) => string) =>\n message\n .split(' ,')\n .map((v) => callback(v))\n .join(' ,');\n\nexport function validateConfig(c: UserConfig) {\n trace('Validating configuration file');\n // Note: checkConfigFile mutates the config file by applying defaults\n const isValid = checkConfigFile(c);\n if (isValid !== true) {\n throw new ValidationError(\n 'InvalidStructure',\n (Array.isArray(isValid) ? isValid : [])\n .map((v) => {\n if (v.type === 'objectStrict') {\n return `Invalid key(s) ${splitMap(\n v.actual,\n (m) => `\"${pc.cyan(m)}\"`,\n )}. Expected one of ${splitMap(v.expected, pc.green)}`;\n }\n if (v.field) {\n return v.message?.replace(v.field, pc.cyan(v.field));\n }\n return v.message;\n })\n .join(' \\n'),\n );\n }\n\n const languageStrings = c.languages.map((v) => v.name);\n\n // Dev Language should exist in languages\n if (!languageStrings.includes(c.devLanguage)) {\n throw new ValidationError(\n 'InvalidDevLanguage',\n `The dev language \"${boldCyan(\n c.devLanguage,\n )}\" was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n\n const foundLanguages: string[] = [];\n for (const lang of c.languages) {\n // Languages must only exist once\n if (foundLanguages.includes(lang.name)) {\n throw new ValidationError(\n 'DuplicateLanguage',\n `The language \"${boldCyan(lang.name)}\" was defined multiple times.`,\n );\n }\n foundLanguages.push(lang.name);\n\n // Any extends must be in languages\n if (lang.extends && !languageStrings.includes(lang.extends)) {\n throw new ValidationError(\n 'InvalidExtends',\n `The language \"${boldCyan(lang.name)}\"'s extends of ${boldCyan(\n lang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n const foundGeneratedLanguages: string[] = [];\n for (const generatedLang of c.generatedLanguages || []) {\n // Generated languages must only exist once\n if (foundGeneratedLanguages.includes(generatedLang.name)) {\n throw new ValidationError(\n 'DuplicateGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" was defined multiple times.`,\n );\n }\n foundGeneratedLanguages.push(generatedLang.name);\n\n // Generated language names must not conflict with language names\n if (languageStrings.includes(generatedLang.name)) {\n throw new ValidationError(\n 'InvalidGeneratedLanguage',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\" is already defined as a language.`,\n );\n }\n\n // Any extends must be in languages\n if (\n generatedLang.extends &&\n !languageStrings.includes(generatedLang.extends)\n ) {\n throw new ValidationError(\n 'InvalidExtends',\n `The generated language \"${boldCyan(\n generatedLang.name,\n )}\"'s extends of ${boldCyan(\n generatedLang.extends,\n )} was not found in languages ${languageStrings.join(', ')}.`,\n );\n }\n }\n\n trace('Configuration file is valid');\n\n return true;\n}\n\nfunction createConfig(configFilePath: string) {\n const cwd = path.dirname(configFilePath);\n\n return {\n projectRoot: cwd,\n // eslint-disable-next-line @typescript-eslint/no-require-imports\n ...(require(configFilePath) as UserConfig),\n };\n}\n\nexport async function resolveConfig(\n customConfigFilePath?: string,\n): Promise<UserConfig | null> {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : await findUp(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n return null;\n}\n\nexport function resolveConfigSync(\n customConfigFilePath?: string,\n): UserConfig | null {\n const configFilePath = customConfigFilePath\n ? path.resolve(customConfigFilePath)\n : findUp.sync(['vocab.config.js', 'vocab.config.cjs']);\n\n if (configFilePath) {\n trace(`Resolved configuration file to ${configFilePath}`);\n return createConfig(configFilePath);\n }\n trace('No configuration file found');\n\n return null;\n}\n"],"mappings":"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"}
|
package/dist/runtime.cjs
CHANGED
|
@@ -1,13 +1,13 @@
|
|
|
1
|
-
|
|
2
|
-
const
|
|
3
|
-
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
2
|
+
const require_icu_handler = require("./icu-handler.cjs");
|
|
3
|
+
const require_translation_file = require("./translation-file.cjs");
|
|
4
4
|
//#region src/runtime.ts
|
|
5
|
-
const createLanguage = (module
|
|
6
|
-
getValue: (locale) => require_icu_handler.getParsedICUMessages(module
|
|
5
|
+
const createLanguage = (module) => ({
|
|
6
|
+
getValue: (locale) => require_icu_handler.getParsedICUMessages(module, locale),
|
|
7
7
|
load: () => Promise.resolve()
|
|
8
8
|
});
|
|
9
|
-
|
|
10
9
|
//#endregion
|
|
11
10
|
exports.createLanguage = createLanguage;
|
|
12
11
|
exports.createTranslationFile = require_translation_file.createTranslationFile;
|
|
12
|
+
|
|
13
13
|
//# sourceMappingURL=runtime.cjs.map
|
package/dist/runtime.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"runtime.cjs","names":["getParsedICUMessages"
|
|
1
|
+
{"version":3,"file":"runtime.cjs","names":["getParsedICUMessages"],"sources":["../src/runtime.ts"],"sourcesContent":["import type { TranslationModule, TranslationMessagesByKey } from './types';\n\nimport { getParsedICUMessages } from './icu-handler';\n\nexport { createTranslationFile } from './translation-file';\n\nexport const createLanguage = (\n module: TranslationMessagesByKey,\n): TranslationModule<any> => ({\n getValue: (locale) => getParsedICUMessages(module, locale),\n load: () => Promise.resolve(),\n});\n"],"mappings":";;;;AAMA,MAAa,kBACX,YAC4B;CAC5B,WAAW,WAAWA,oBAAAA,qBAAqB,QAAQ,OAAO;CAC1D,YAAY,QAAQ,SAAS;CAC9B"}
|
package/dist/runtime.d.cts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { b as TranslationModule, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { b as TranslationModule, y as TranslationMessagesByKey } from "./types-Bybu5Wvr.cjs";
|
|
2
2
|
import { createTranslationFile } from "./translation-file.cjs";
|
|
3
3
|
|
|
4
4
|
//#region src/runtime.d.ts
|
package/dist/runtime.d.mts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { b as TranslationModule, y as TranslationMessagesByKey } from "./types-
|
|
1
|
+
import { b as TranslationModule, y as TranslationMessagesByKey } from "./types-DW00hfr8.mjs";
|
|
2
2
|
import { createTranslationFile } from "./translation-file.mjs";
|
|
3
3
|
|
|
4
4
|
//#region src/runtime.d.ts
|
package/dist/runtime.mjs
CHANGED
|
@@ -1,12 +1,11 @@
|
|
|
1
1
|
import { getParsedICUMessages } from "./icu-handler.mjs";
|
|
2
2
|
import { createTranslationFile } from "./translation-file.mjs";
|
|
3
|
-
|
|
4
3
|
//#region src/runtime.ts
|
|
5
4
|
const createLanguage = (module) => ({
|
|
6
5
|
getValue: (locale) => getParsedICUMessages(module, locale),
|
|
7
6
|
load: () => Promise.resolve()
|
|
8
7
|
});
|
|
9
|
-
|
|
10
8
|
//#endregion
|
|
11
9
|
export { createLanguage, createTranslationFile };
|
|
10
|
+
|
|
12
11
|
//# sourceMappingURL=runtime.mjs.map
|
package/dist/runtime.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"runtime.mjs","names":[],"sources":["../src/runtime.ts"],"sourcesContent":["import type { TranslationModule, TranslationMessagesByKey } from './types';\n\nimport { getParsedICUMessages } from './icu-handler';\n\nexport { createTranslationFile } from './translation-file';\n\nexport const createLanguage = (\n module: TranslationMessagesByKey,\n): TranslationModule<any> => ({\n getValue: (locale) => getParsedICUMessages(module, locale),\n load: () => Promise.resolve(),\n});\n"],"mappings":"
|
|
1
|
+
{"version":3,"file":"runtime.mjs","names":[],"sources":["../src/runtime.ts"],"sourcesContent":["import type { TranslationModule, TranslationMessagesByKey } from './types';\n\nimport { getParsedICUMessages } from './icu-handler';\n\nexport { createTranslationFile } from './translation-file';\n\nexport const createLanguage = (\n module: TranslationMessagesByKey,\n): TranslationModule<any> => ({\n getValue: (locale) => getParsedICUMessages(module, locale),\n load: () => Promise.resolve(),\n});\n"],"mappings":";;;AAMA,MAAa,kBACX,YAC4B;CAC5B,WAAW,WAAW,qBAAqB,QAAQ,OAAO;CAC1D,YAAY,QAAQ,SAAS;CAC9B"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
|
|
2
2
|
//#region src/translation-file.ts
|
|
3
3
|
function createTranslationFile(translationsByLanguage) {
|
|
4
4
|
function getByLanguage(language) {
|
|
@@ -23,7 +23,7 @@ function createTranslationFile(translationsByLanguage) {
|
|
|
23
23
|
}
|
|
24
24
|
};
|
|
25
25
|
}
|
|
26
|
-
|
|
27
26
|
//#endregion
|
|
28
27
|
exports.createTranslationFile = createTranslationFile;
|
|
28
|
+
|
|
29
29
|
//# sourceMappingURL=translation-file.cjs.map
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { c as ParsedFormatFnByKey, p as TranslationFile, r as LanguageName, x as TranslationModuleByLanguage } from "./types-
|
|
1
|
+
import { c as ParsedFormatFnByKey, p as TranslationFile, r as LanguageName, x as TranslationModuleByLanguage } from "./types-Bybu5Wvr.cjs";
|
|
2
2
|
|
|
3
3
|
//#region src/translation-file.d.ts
|
|
4
4
|
declare function createTranslationFile<Language extends LanguageName, FormatFnByKey extends ParsedFormatFnByKey>(translationsByLanguage: TranslationModuleByLanguage<Language, FormatFnByKey>): TranslationFile<Language, FormatFnByKey>;
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { c as ParsedFormatFnByKey, p as TranslationFile, r as LanguageName, x as TranslationModuleByLanguage } from "./types-
|
|
1
|
+
import { c as ParsedFormatFnByKey, p as TranslationFile, r as LanguageName, x as TranslationModuleByLanguage } from "./types-DW00hfr8.mjs";
|
|
2
2
|
|
|
3
3
|
//#region src/translation-file.d.ts
|
|
4
4
|
declare function createTranslationFile<Language extends LanguageName, FormatFnByKey extends ParsedFormatFnByKey>(translationsByLanguage: TranslationModuleByLanguage<Language, FormatFnByKey>): TranslationFile<Language, FormatFnByKey>;
|
|
@@ -143,4 +143,4 @@ type LoadedTranslation<Key extends TranslationKey = string> = {
|
|
|
143
143
|
};
|
|
144
144
|
//#endregion
|
|
145
145
|
export { TranslationsByLanguage as C, TranslationsByKey as S, TranslationKeys as _, LoadedTranslation as a, TranslationModule as b, ParsedFormatFnByKey as c, Tags as d, TranslationData as f, TranslationKey as g, TranslationFileMetadata as h, LanguageTarget as i, ParsedICUMessages as l, TranslationFileContents as m, GeneratedLanguageTarget as n, MessageGenerator as o, TranslationFile as p, LanguageName as r, ParsedFormatFn as s, FormatXMLElementFn as t, StringWithSuggestions as u, TranslationMessage as v, UserConfig as w, TranslationModuleByLanguage as x, TranslationMessagesByKey as y };
|
|
146
|
-
//# sourceMappingURL=types-
|
|
146
|
+
//# sourceMappingURL=types-Bybu5Wvr.d.cts.map
|
|
@@ -143,4 +143,4 @@ type LoadedTranslation<Key extends TranslationKey = string> = {
|
|
|
143
143
|
};
|
|
144
144
|
//#endregion
|
|
145
145
|
export { TranslationsByLanguage as C, TranslationsByKey as S, TranslationKeys as _, LoadedTranslation as a, TranslationModule as b, ParsedFormatFnByKey as c, Tags as d, TranslationData as f, TranslationKey as g, TranslationFileMetadata as h, LanguageTarget as i, ParsedICUMessages as l, TranslationFileContents as m, GeneratedLanguageTarget as n, MessageGenerator as o, TranslationFile as p, LanguageName as r, ParsedFormatFn as s, FormatXMLElementFn as t, StringWithSuggestions as u, TranslationMessage as v, UserConfig as w, TranslationModuleByLanguage as x, TranslationMessagesByKey as y };
|
|
146
|
-
//# sourceMappingURL=types-
|
|
146
|
+
//# sourceMappingURL=types-DW00hfr8.d.mts.map
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@vocab/core",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.8.0",
|
|
4
4
|
"repository": {
|
|
5
5
|
"type": "git",
|
|
6
6
|
"url": "https://github.com/seek-oss/vocab.git",
|
|
@@ -11,20 +11,20 @@
|
|
|
11
11
|
"types": "./dist/index.d.cts",
|
|
12
12
|
"exports": {
|
|
13
13
|
".": {
|
|
14
|
-
"
|
|
15
|
-
"
|
|
14
|
+
"import": "./dist/index.mjs",
|
|
15
|
+
"require": "./dist/index.cjs"
|
|
16
16
|
},
|
|
17
17
|
"./icu-handler": {
|
|
18
|
-
"
|
|
19
|
-
"
|
|
18
|
+
"import": "./dist/icu-handler.mjs",
|
|
19
|
+
"require": "./dist/icu-handler.cjs"
|
|
20
20
|
},
|
|
21
21
|
"./runtime": {
|
|
22
|
-
"
|
|
23
|
-
"
|
|
22
|
+
"import": "./dist/runtime.mjs",
|
|
23
|
+
"require": "./dist/runtime.cjs"
|
|
24
24
|
},
|
|
25
25
|
"./translation-file": {
|
|
26
|
-
"
|
|
27
|
-
"
|
|
26
|
+
"import": "./dist/translation-file.mjs",
|
|
27
|
+
"require": "./dist/translation-file.cjs"
|
|
28
28
|
},
|
|
29
29
|
"./package.json": "./package.json"
|
|
30
30
|
},
|
|
@@ -44,7 +44,7 @@
|
|
|
44
44
|
"find-up": "^5.0.0",
|
|
45
45
|
"intl-messageformat": "^10.0.0",
|
|
46
46
|
"picocolors": "^1.0.0",
|
|
47
|
-
"picomatch": "^4.0.
|
|
47
|
+
"picomatch": "^4.0.4",
|
|
48
48
|
"prettier": "^3.5.3",
|
|
49
49
|
"tinyglobby": "^0.2.15"
|
|
50
50
|
},
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
//#region rolldown:runtime
|
|
2
|
-
var __create = Object.create;
|
|
3
|
-
var __defProp = Object.defineProperty;
|
|
4
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
-
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
-
var __copyProps = (to, from, except, desc) => {
|
|
9
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
10
|
-
for (var keys = __getOwnPropNames(from), i = 0, n = keys.length, key; i < n; i++) {
|
|
11
|
-
key = keys[i];
|
|
12
|
-
if (!__hasOwnProp.call(to, key) && key !== except) {
|
|
13
|
-
__defProp(to, key, {
|
|
14
|
-
get: ((k) => from[k]).bind(null, key),
|
|
15
|
-
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
16
|
-
});
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
}
|
|
20
|
-
return to;
|
|
21
|
-
};
|
|
22
|
-
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", {
|
|
23
|
-
value: mod,
|
|
24
|
-
enumerable: true
|
|
25
|
-
}) : target, mod));
|
|
26
|
-
|
|
27
|
-
//#endregion
|
|
28
|
-
|
|
29
|
-
Object.defineProperty(exports, '__toESM', {
|
|
30
|
-
enumerable: true,
|
|
31
|
-
get: function () {
|
|
32
|
-
return __toESM;
|
|
33
|
-
}
|
|
34
|
-
});
|
package/icu-handler/package.json
DELETED
package/runtime/package.json
DELETED