-
Notifications
You must be signed in to change notification settings - Fork 12k
/
Copy pathload-translations.ts
83 lines (74 loc) · 2.83 KB
/
load-translations.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
import type { Diagnostics } from '@angular/localize/tools';
import { createHash } from 'crypto';
import * as fs from 'fs';
import { loadEsmModule } from './load-esm';
export type TranslationLoader = (path: string) => {
translations: Record<string, import('@angular/localize').ɵParsedTranslation>;
format: string;
locale?: string;
diagnostics: Diagnostics;
integrity: string;
};
export async function createTranslationLoader(): Promise<TranslationLoader> {
const { parsers, diagnostics } = await importParsers();
return (path: string) => {
const content = fs.readFileSync(path, 'utf8');
const unusedParsers = new Map();
for (const [format, parser] of Object.entries(parsers)) {
const analysis = parser.analyze(path, content);
if (analysis.canParse) {
// Types don't overlap here so we need to use any.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
const { locale, translations } = parser.parse(path, content, analysis.hint as any);
const integrity = 'sha256-' + createHash('sha256').update(content).digest('base64');
return { format, locale, translations, diagnostics, integrity };
} else {
unusedParsers.set(parser, analysis);
}
}
const messages: string[] = [];
for (const [parser, analysis] of unusedParsers.entries()) {
messages.push(analysis.diagnostics.formatDiagnostics(`*** ${parser.constructor.name} ***`));
}
throw new Error(
`Unsupported translation file format in ${path}. The following parsers were tried:\n` +
messages.join('\n'),
);
};
}
async function importParsers() {
try {
// Load ESM `@angular/localize/tools` using the TypeScript dynamic import workaround.
// Once TypeScript provides support for keeping the dynamic import this workaround can be
// changed to a direct dynamic import.
const {
Diagnostics,
ArbTranslationParser,
SimpleJsonTranslationParser,
Xliff1TranslationParser,
Xliff2TranslationParser,
XtbTranslationParser,
} = await loadEsmModule<typeof import('@angular/localize/tools')>('@angular/localize/tools');
const diagnostics = new Diagnostics();
const parsers = {
arb: new ArbTranslationParser(),
json: new SimpleJsonTranslationParser(),
xlf: new Xliff1TranslationParser(),
xlf2: new Xliff2TranslationParser(),
// The name ('xmb') needs to match the AOT compiler option
xmb: new XtbTranslationParser(),
};
return { parsers, diagnostics };
} catch {
throw new Error(
`Unable to load translation file parsers. Please ensure '@angular/localize' is installed.`,
);
}
}