-
Notifications
You must be signed in to change notification settings - Fork 12k
/
Copy pathi18n-options.ts
251 lines (214 loc) · 8 KB
/
i18n-options.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
/**
* @license
* Copyright Google Inc. 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 { json, virtualFs } from '@angular-devkit/core';
import * as fs from 'fs';
import * as os from 'os';
import * as path from 'path';
import * as rimraf from 'rimraf';
import { readTsconfig } from '../angular-cli-files/utilities/read-tsconfig';
import { Schema as BrowserBuilderSchema } from '../browser/schema';
import { Schema as ServerBuilderSchema } from '../server/schema';
import { createTranslationLoader } from './load-translations';
export interface I18nOptions {
inlineLocales: Set<string>;
sourceLocale: string;
locales: Record<
string,
{ file: string; format?: string; translation?: unknown; dataPath?: string }
>;
flatOutput?: boolean;
readonly shouldInline: boolean;
}
export function createI18nOptions(
metadata: json.JsonObject,
inline?: boolean | string[],
): I18nOptions {
if (
metadata.i18n !== undefined &&
(typeof metadata.i18n !== 'object' || !metadata.i18n || Array.isArray(metadata.i18n))
) {
throw new Error('Project i18n field is malformed. Expected an object.');
}
metadata = metadata.i18n || {};
if (metadata.sourceLocale !== undefined && typeof metadata.sourceLocale !== 'string') {
throw new Error('Project i18n sourceLocale field is malformed. Expected a string.');
}
const i18n: I18nOptions = {
inlineLocales: new Set<string>(),
// en-US is the default locale added to Angular applications (https://angular.io/guide/i18n#i18n-pipes)
sourceLocale: metadata.sourceLocale || 'en-US',
locales: {},
get shouldInline() {
return this.inlineLocales.size > 0;
},
};
if (
metadata.locales !== undefined &&
(!metadata.locales || typeof metadata.locales !== 'object' || Array.isArray(metadata.locales))
) {
throw new Error('Project i18n locales field is malformed. Expected an object.');
} else if (metadata.locales) {
for (const [locale, translationFile] of Object.entries(metadata.locales)) {
if (typeof translationFile !== 'string') {
throw new Error(
`Project i18n locales field value for '${locale}' is malformed. Expected a string.`,
);
}
if (locale === i18n.sourceLocale) {
throw new Error(
`An i18n locale identifier ('${locale}') cannot both be a source locale and provide a translation.`,
);
}
i18n.locales[locale] = {
file: translationFile,
};
}
}
if (inline === true) {
i18n.inlineLocales.add(i18n.sourceLocale);
Object.keys(i18n.locales).forEach(locale => i18n.inlineLocales.add(locale));
} else if (inline) {
for (const locale of inline) {
if (!i18n.locales[locale] && i18n.sourceLocale !== locale) {
throw new Error(`Requested locale '${locale}' is not defined for the project.`);
}
i18n.inlineLocales.add(locale);
}
}
return i18n;
}
export async function configureI18nBuild<T extends BrowserBuilderSchema | ServerBuilderSchema>(
context: BuilderContext,
options: T,
): Promise<{
buildOptions: T;
i18n: I18nOptions;
}> {
if (!context.target) {
throw new Error('The builder requires a target.');
}
const buildOptions = { ...options };
if (
buildOptions.localize === true ||
(Array.isArray(buildOptions.localize) && buildOptions.localize.length > 1)
) {
throw new Error('Using the localize option for multiple locales is temporarily disabled.');
}
const tsConfig = readTsconfig(buildOptions.tsConfig, context.workspaceRoot);
const usingIvy = tsConfig.options.enableIvy !== false;
const metadata = await context.getProjectMetadata(context.target);
const i18n = createI18nOptions(metadata, buildOptions.localize);
// Until 11.0, support deprecated i18n options when not using new localize option
// i18nFormat is automatically calculated
if (buildOptions.localize === undefined && usingIvy) {
mergeDeprecatedI18nOptions(i18n, buildOptions.i18nLocale, buildOptions.i18nFile);
} else if (buildOptions.localize !== undefined && !usingIvy) {
buildOptions.localize = undefined;
context.logger.warn(`Option 'localize' is not supported with View Engine.`);
}
if (i18n.inlineLocales.size > 0) {
const projectRoot = path.join(context.workspaceRoot, (metadata.root as string) || '');
const localeDataBasePath = findLocaleDataBasePath(projectRoot);
if (!localeDataBasePath) {
throw new Error(
`Unable to find locale data within '@angular/common'. Please ensure '@angular/common' is installed.`,
);
}
// Load locales
const loader = await createTranslationLoader();
const usedFormats = new Set<string>();
for (const [locale, desc] of Object.entries(i18n.locales)) {
if (i18n.inlineLocales.has(locale) && desc.file) {
const result = loader(path.join(projectRoot, desc.file));
usedFormats.add(result.format);
if (usedFormats.size > 1 && tsConfig.options.enableI18nLegacyMessageIdFormat !== false) {
// This limitation is only for legacy message id support (defaults to true as of 9.0)
throw new Error(
'Localization currently only supports using one type of translation file format for the entire application.',
);
}
desc.format = result.format;
desc.translation = result.translation;
const localeDataPath = findLocaleDataPath(locale, localeDataBasePath);
if (!localeDataPath) {
context.logger.warn(
`Locale data for '${locale}' cannot be found. No locale data will be included for this locale.`,
);
} else {
// Temporarily disable pending FW locale data fix
// desc.dataPath = localeDataPath;
}
}
}
// Legacy message id's require the format of the translations
if (usedFormats.size > 0) {
buildOptions.i18nFormat = [...usedFormats][0];
}
// If only one locale is specified set the deprecated option to enable the webpack plugin
// transform to register the locale directly in the output bundle.
if (i18n.inlineLocales.size === 1) {
buildOptions.i18nLocale = [...i18n.inlineLocales][0];
}
}
// If inlining store the output in a temporary location to facilitate post-processing
if (i18n.shouldInline) {
const tempPath = fs.mkdtempSync(path.join(fs.realpathSync(os.tmpdir()), 'angular-cli-i18n-'));
buildOptions.outputPath = tempPath;
// Remove temporary directory used for i18n processing
process.on('exit', () => {
try {
rimraf.sync(tempPath);
} catch {}
});
}
return { buildOptions, i18n };
}
function mergeDeprecatedI18nOptions(
i18n: I18nOptions,
i18nLocale: string | undefined,
i18nFile: string | undefined,
): I18nOptions {
if (i18nFile !== undefined && i18nLocale === undefined) {
throw new Error(`Option 'i18nFile' cannot be used without the 'i18nLocale' option.`);
}
if (i18nLocale !== undefined) {
i18n.inlineLocales.clear();
i18n.inlineLocales.add(i18nLocale);
if (i18nFile !== undefined) {
i18n.locales[i18nLocale] = { file: i18nFile };
} else {
// If no file, treat the locale as the source locale
// This mimics deprecated behavior
i18n.sourceLocale = i18nLocale;
}
i18n.flatOutput = true;
}
return i18n;
}
function findLocaleDataBasePath(projectRoot: string): string | null {
try {
const commonPath = path.dirname(
require.resolve('@angular/common/package.json', { paths: [projectRoot] }),
);
const localesPath = path.join(commonPath, 'locales/global');
if (!fs.existsSync(localesPath)) {
return null;
}
return localesPath;
} catch {
return null;
}
}
function findLocaleDataPath(locale: string, basePath: string): string | null {
const localeDataPath = path.join(basePath, locale + '.js');
if (!fs.existsSync(localeDataPath)) {
return null;
}
return localeDataPath;
}