-
Notifications
You must be signed in to change notification settings - Fork 12k
/
Copy pathi18n-inlining.ts
138 lines (123 loc) · 3.76 KB
/
i18n-inlining.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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
/**
* @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 { BuilderContext } from '@angular-devkit/architect';
import { EmittedFiles } from '@angular-devkit/build-webpack';
import * as fs from 'fs';
import * as path from 'path';
import { BundleActionExecutor } from './action-executor';
import { copyAssets } from './copy-assets';
import { I18nOptions } from './i18n-options';
import { InlineOptions } from './process-bundle';
import { Spinner } from './spinner';
function emittedFilesToInlineOptions(
emittedFiles: EmittedFiles[],
scriptsEntryPointName: string[],
emittedPath: string,
outputPath: string,
es5: boolean,
missingTranslation: 'error' | 'warning' | 'ignore' | undefined,
context: BuilderContext,
): { options: InlineOptions[]; originalFiles: string[] } {
const options: InlineOptions[] = [];
const originalFiles: string[] = [];
for (const emittedFile of emittedFiles) {
if (
emittedFile.asset ||
emittedFile.extension !== '.js' ||
(emittedFile.name && scriptsEntryPointName.includes(emittedFile.name))
) {
continue;
}
const originalPath = path.join(emittedPath, emittedFile.file);
const action: InlineOptions = {
filename: emittedFile.file,
code: fs.readFileSync(originalPath, 'utf8'),
es5,
outputPath,
missingTranslation,
setLocale: emittedFile.name === 'main' || emittedFile.name === 'vendor',
};
originalFiles.push(originalPath);
try {
const originalMapPath = originalPath + '.map';
action.map = fs.readFileSync(originalMapPath, 'utf8');
originalFiles.push(originalMapPath);
} catch (err) {
if (err.code !== 'ENOENT') {
throw err;
}
}
context.logger.debug(`i18n file queued for processing: ${action.filename}`);
options.push(action);
}
return { options, originalFiles };
}
export async function i18nInlineEmittedFiles(
context: BuilderContext,
emittedFiles: EmittedFiles[],
i18n: I18nOptions,
baseOutputPath: string,
outputPaths: string[],
scriptsEntryPointName: string[],
emittedPath: string,
es5: boolean,
missingTranslation: 'error' | 'warning' | 'ignore' | undefined,
): Promise<boolean> {
const executor = new BundleActionExecutor({ i18n });
let hasErrors = false;
const spinner = new Spinner();
spinner.start('Generating localized bundles...');
try {
const { options, originalFiles: processedFiles } = emittedFilesToInlineOptions(
emittedFiles,
scriptsEntryPointName,
emittedPath,
baseOutputPath,
es5,
missingTranslation,
context,
);
for await (const result of executor.inlineAll(options)) {
context.logger.debug(`i18n file processed: ${result.file}`);
for (const diagnostic of result.diagnostics) {
spinner.stop();
if (diagnostic.type === 'error') {
hasErrors = true;
context.logger.error(diagnostic.message);
} else {
context.logger.warn(diagnostic.message);
}
spinner.start();
}
}
// Copy any non-processed files into the output locations
await copyAssets(
[
{
glob: '**/*',
input: emittedPath,
output: '',
ignore: [...processedFiles].map((f) => path.relative(emittedPath, f)),
},
],
outputPaths,
'',
);
} catch (err) {
spinner.fail('Localized bundle generation failed: ' + err.message);
return false;
} finally {
executor.stop();
}
if (hasErrors) {
spinner.fail('Localized bundle generation failed.');
} else {
spinner.succeed('Localized bundle generation complete.');
}
return !hasErrors;
}