@angular-devkit/build-angular 16.2.1 → 17.0.0-next.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.
Files changed (41) hide show
  1. package/README.md +14 -10
  2. package/builders.json +10 -0
  3. package/package.json +30 -28
  4. package/src/builders/dev-server/vite-server.d.ts +1 -1
  5. package/src/builders/dev-server/vite-server.js +11 -8
  6. package/src/builders/extract-i18n/application-extraction.d.ts +17 -0
  7. package/src/builders/extract-i18n/application-extraction.js +137 -0
  8. package/src/builders/extract-i18n/builder.d.ts +17 -0
  9. package/src/builders/extract-i18n/builder.js +152 -0
  10. package/src/builders/extract-i18n/index.d.ts +4 -14
  11. package/src/builders/extract-i18n/index.js +4 -247
  12. package/src/builders/extract-i18n/options.d.ts +29 -0
  13. package/src/builders/extract-i18n/options.js +82 -0
  14. package/src/builders/extract-i18n/webpack-extraction.d.ts +21 -0
  15. package/src/builders/extract-i18n/webpack-extraction.js +100 -0
  16. package/src/builders/prerender/index.d.ts +20 -0
  17. package/src/builders/prerender/index.js +180 -0
  18. package/src/builders/prerender/render-worker.d.ts +30 -0
  19. package/src/builders/prerender/render-worker.js +126 -0
  20. package/src/builders/prerender/schema.d.ts +22 -0
  21. package/src/builders/prerender/schema.js +5 -0
  22. package/src/builders/prerender/schema.json +39 -0
  23. package/src/builders/prerender/utils.d.ts +22 -0
  24. package/src/builders/prerender/utils.js +79 -0
  25. package/src/builders/ssr-dev-server/index.d.ts +23 -0
  26. package/src/builders/ssr-dev-server/index.js +309 -0
  27. package/src/builders/ssr-dev-server/schema.d.ts +64 -0
  28. package/src/builders/ssr-dev-server/schema.js +5 -0
  29. package/src/builders/ssr-dev-server/schema.json +75 -0
  30. package/src/builders/ssr-dev-server/utils.d.ts +15 -0
  31. package/src/builders/ssr-dev-server/utils.js +75 -0
  32. package/src/tools/babel/webpack-loader.js +2 -2
  33. package/src/tools/esbuild/angular/compilation/angular-compilation.d.ts +7 -2
  34. package/src/tools/esbuild/angular/compilation/angular-compilation.js +21 -1
  35. package/src/tools/esbuild/angular/compiler-plugin.js +8 -13
  36. package/src/tools/esbuild/application-code-bundle.js +3 -5
  37. package/src/tools/esbuild/utils.js +5 -1
  38. package/src/tools/sass/rebasing-importer.js +2 -2
  39. package/src/utils/index-file/inline-critical-css.js +13 -26
  40. package/src/tools/esbuild/external-packages-plugin.d.ts +0 -17
  41. package/src/tools/esbuild/external-packages-plugin.js +0 -49
@@ -0,0 +1,137 @@
1
+ "use strict";
2
+ /**
3
+ * @license
4
+ * Copyright Google LLC All Rights Reserved.
5
+ *
6
+ * Use of this source code is governed by an MIT-style license that can be
7
+ * found in the LICENSE file at https://angular.io/license
8
+ */
9
+ var __importDefault = (this && this.__importDefault) || function (mod) {
10
+ return (mod && mod.__esModule) ? mod : { "default": mod };
11
+ };
12
+ Object.defineProperty(exports, "__esModule", { value: true });
13
+ exports.extractMessages = void 0;
14
+ const node_assert_1 = __importDefault(require("node:assert"));
15
+ const node_path_1 = __importDefault(require("node:path"));
16
+ const application_1 = require("../application");
17
+ const browser_esbuild_1 = require("../browser-esbuild");
18
+ async function extractMessages(options, builderName, context, extractorConstructor) {
19
+ const messages = [];
20
+ // Setup the build options for the application based on the browserTarget option
21
+ const buildOptions = (await context.validateOptions(await context.getTargetOptions(options.browserTarget), builderName));
22
+ buildOptions.optimization = false;
23
+ buildOptions.sourceMap = { scripts: true, vendor: true };
24
+ let build;
25
+ if (builderName === '@angular-devkit/build-angular:application') {
26
+ build = application_1.buildApplicationInternal;
27
+ buildOptions.ssr = false;
28
+ buildOptions.appShell = false;
29
+ buildOptions.prerender = false;
30
+ }
31
+ else {
32
+ build = browser_esbuild_1.buildEsbuildBrowser;
33
+ }
34
+ // Build the application with the build options
35
+ let builderResult;
36
+ try {
37
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
38
+ for await (const result of build(buildOptions, context, { write: false })) {
39
+ builderResult = result;
40
+ break;
41
+ }
42
+ (0, node_assert_1.default)(builderResult !== undefined, 'Application builder did not provide a result.');
43
+ }
44
+ catch (err) {
45
+ builderResult = {
46
+ success: false,
47
+ error: err.message,
48
+ };
49
+ }
50
+ // Extract messages from each output JavaScript file.
51
+ // Output files are only present on a successful build.
52
+ if (builderResult.outputFiles) {
53
+ // Store the JS and JS map files for lookup during extraction
54
+ const files = new Map();
55
+ for (const outputFile of builderResult.outputFiles) {
56
+ if (outputFile.path.endsWith('.js')) {
57
+ files.set(outputFile.path, outputFile.text);
58
+ }
59
+ else if (outputFile.path.endsWith('.js.map')) {
60
+ files.set(outputFile.path, outputFile.text);
61
+ }
62
+ }
63
+ // Setup the localize message extractor based on the in-memory files
64
+ const extractor = setupLocalizeExtractor(extractorConstructor, files, context);
65
+ // Attempt extraction of all output JS files
66
+ for (const filePath of files.keys()) {
67
+ if (!filePath.endsWith('.js')) {
68
+ continue;
69
+ }
70
+ const fileMessages = extractor.extractMessages(filePath);
71
+ messages.push(...fileMessages);
72
+ }
73
+ }
74
+ return {
75
+ builderResult,
76
+ basePath: context.workspaceRoot,
77
+ messages,
78
+ // Legacy i18n identifiers are not supported with the new application builder
79
+ useLegacyIds: false,
80
+ };
81
+ }
82
+ exports.extractMessages = extractMessages;
83
+ function setupLocalizeExtractor(extractorConstructor, files, context) {
84
+ // Setup a virtual file system instance for the extractor
85
+ // * MessageExtractor itself uses readFile, relative and resolve
86
+ // * Internal SourceFileLoader (sourcemap support) uses dirname, exists, readFile, and resolve
87
+ const filesystem = {
88
+ readFile(path) {
89
+ // Output files are stored as relative to the workspace root
90
+ const requestedPath = node_path_1.default.relative(context.workspaceRoot, path);
91
+ const content = files.get(requestedPath);
92
+ if (content === undefined) {
93
+ throw new Error('Unknown file requested: ' + requestedPath);
94
+ }
95
+ return content;
96
+ },
97
+ relative(from, to) {
98
+ return node_path_1.default.relative(from, to);
99
+ },
100
+ resolve(...paths) {
101
+ return node_path_1.default.resolve(...paths);
102
+ },
103
+ exists(path) {
104
+ // Output files are stored as relative to the workspace root
105
+ const requestedPath = node_path_1.default.relative(context.workspaceRoot, path);
106
+ return files.has(requestedPath);
107
+ },
108
+ dirname(path) {
109
+ return node_path_1.default.dirname(path);
110
+ },
111
+ };
112
+ const logger = {
113
+ // level 2 is warnings
114
+ level: 2,
115
+ debug(...args) {
116
+ // eslint-disable-next-line no-console
117
+ console.debug(...args);
118
+ },
119
+ info(...args) {
120
+ context.logger.info(args.join(''));
121
+ },
122
+ warn(...args) {
123
+ context.logger.warn(args.join(''));
124
+ },
125
+ error(...args) {
126
+ context.logger.error(args.join(''));
127
+ },
128
+ };
129
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
130
+ const extractor = new extractorConstructor(filesystem, logger, {
131
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
132
+ basePath: context.workspaceRoot,
133
+ useSourceMaps: true,
134
+ });
135
+ return extractor;
136
+ }
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application-extraction.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/extract-i18n/application-extraction.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport type { ɵParsedMessage as LocalizeMessage } from '@angular/localize';\nimport type { MessageExtractor } from '@angular/localize/tools';\nimport type { BuilderContext, BuilderOutput } from '@angular-devkit/architect';\nimport assert from 'node:assert';\nimport nodePath from 'node:path';\nimport { buildApplicationInternal } from '../application';\nimport type { ApplicationBuilderInternalOptions } from '../application/options';\nimport { buildEsbuildBrowser } from '../browser-esbuild';\nimport type { NormalizedExtractI18nOptions } from './options';\n\nexport async function extractMessages(\n  options: NormalizedExtractI18nOptions,\n  builderName: string,\n  context: BuilderContext,\n  extractorConstructor: typeof MessageExtractor,\n): Promise<{\n  builderResult: BuilderOutput;\n  basePath: string;\n  messages: LocalizeMessage[];\n  useLegacyIds: boolean;\n}> {\n  const messages: LocalizeMessage[] = [];\n\n  // Setup the build options for the application based on the browserTarget option\n  const buildOptions = (await context.validateOptions(\n    await context.getTargetOptions(options.browserTarget),\n    builderName,\n  )) as unknown as ApplicationBuilderInternalOptions;\n  buildOptions.optimization = false;\n  buildOptions.sourceMap = { scripts: true, vendor: true };\n\n  let build;\n  if (builderName === '@angular-devkit/build-angular:application') {\n    build = buildApplicationInternal;\n\n    buildOptions.ssr = false;\n    buildOptions.appShell = false;\n    buildOptions.prerender = false;\n  } else {\n    build = buildEsbuildBrowser;\n  }\n\n  // Build the application with the build options\n  let builderResult;\n  try {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    for await (const result of build(buildOptions as any, context, { write: false })) {\n      builderResult = result;\n      break;\n    }\n\n    assert(builderResult !== undefined, 'Application builder did not provide a result.');\n  } catch (err) {\n    builderResult = {\n      success: false,\n      error: (err as Error).message,\n    };\n  }\n\n  // Extract messages from each output JavaScript file.\n  // Output files are only present on a successful build.\n  if (builderResult.outputFiles) {\n    // Store the JS and JS map files for lookup during extraction\n    const files = new Map<string, string>();\n    for (const outputFile of builderResult.outputFiles) {\n      if (outputFile.path.endsWith('.js')) {\n        files.set(outputFile.path, outputFile.text);\n      } else if (outputFile.path.endsWith('.js.map')) {\n        files.set(outputFile.path, outputFile.text);\n      }\n    }\n\n    // Setup the localize message extractor based on the in-memory files\n    const extractor = setupLocalizeExtractor(extractorConstructor, files, context);\n\n    // Attempt extraction of all output JS files\n    for (const filePath of files.keys()) {\n      if (!filePath.endsWith('.js')) {\n        continue;\n      }\n\n      const fileMessages = extractor.extractMessages(filePath);\n      messages.push(...fileMessages);\n    }\n  }\n\n  return {\n    builderResult,\n    basePath: context.workspaceRoot,\n    messages,\n    // Legacy i18n identifiers are not supported with the new application builder\n    useLegacyIds: false,\n  };\n}\n\nfunction setupLocalizeExtractor(\n  extractorConstructor: typeof MessageExtractor,\n  files: Map<string, string>,\n  context: BuilderContext,\n): MessageExtractor {\n  // Setup a virtual file system instance for the extractor\n  // * MessageExtractor itself uses readFile, relative and resolve\n  // * Internal SourceFileLoader (sourcemap support) uses dirname, exists, readFile, and resolve\n  const filesystem = {\n    readFile(path: string): string {\n      // Output files are stored as relative to the workspace root\n      const requestedPath = nodePath.relative(context.workspaceRoot, path);\n\n      const content = files.get(requestedPath);\n      if (content === undefined) {\n        throw new Error('Unknown file requested: ' + requestedPath);\n      }\n\n      return content;\n    },\n    relative(from: string, to: string): string {\n      return nodePath.relative(from, to);\n    },\n    resolve(...paths: string[]): string {\n      return nodePath.resolve(...paths);\n    },\n    exists(path: string): boolean {\n      // Output files are stored as relative to the workspace root\n      const requestedPath = nodePath.relative(context.workspaceRoot, path);\n\n      return files.has(requestedPath);\n    },\n    dirname(path: string): string {\n      return nodePath.dirname(path);\n    },\n  };\n\n  const logger = {\n    // level 2 is warnings\n    level: 2,\n    debug(...args: string[]): void {\n      // eslint-disable-next-line no-console\n      console.debug(...args);\n    },\n    info(...args: string[]): void {\n      context.logger.info(args.join(''));\n    },\n    warn(...args: string[]): void {\n      context.logger.warn(args.join(''));\n    },\n    error(...args: string[]): void {\n      context.logger.error(args.join(''));\n    },\n  };\n\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  const extractor = new extractorConstructor(filesystem as any, logger, {\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    basePath: context.workspaceRoot as any,\n    useSourceMaps: true,\n  });\n\n  return extractor;\n}\n"]}
@@ -0,0 +1,17 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ import type { BuilderContext, BuilderOutput } from '@angular-devkit/architect';
9
+ import type webpack from 'webpack';
10
+ import type { ExecutionTransformer } from '../../transforms';
11
+ import { Schema as ExtractI18nBuilderOptions } from './schema';
12
+ /**
13
+ * @experimental Direct usage of this function is considered experimental.
14
+ */
15
+ export declare function execute(options: ExtractI18nBuilderOptions, context: BuilderContext, transforms?: {
16
+ webpackConfiguration?: ExecutionTransformer<webpack.Configuration>;
17
+ }): Promise<BuilderOutput>;
@@ -0,0 +1,152 @@
1
+ "use strict";
2
+ /**
3
+ * @license
4
+ * Copyright Google LLC All Rights Reserved.
5
+ *
6
+ * Use of this source code is governed by an MIT-style license that can be
7
+ * found in the LICENSE file at https://angular.io/license
8
+ */
9
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ var desc = Object.getOwnPropertyDescriptor(m, k);
12
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
13
+ desc = { enumerable: true, get: function() { return m[k]; } };
14
+ }
15
+ Object.defineProperty(o, k2, desc);
16
+ }) : (function(o, m, k, k2) {
17
+ if (k2 === undefined) k2 = k;
18
+ o[k2] = m[k];
19
+ }));
20
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
21
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
22
+ }) : function(o, v) {
23
+ o["default"] = v;
24
+ });
25
+ var __importStar = (this && this.__importStar) || function (mod) {
26
+ if (mod && mod.__esModule) return mod;
27
+ var result = {};
28
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
29
+ __setModuleDefault(result, mod);
30
+ return result;
31
+ };
32
+ var __importDefault = (this && this.__importDefault) || function (mod) {
33
+ return (mod && mod.__esModule) ? mod : { "default": mod };
34
+ };
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.execute = void 0;
37
+ const node_fs_1 = __importDefault(require("node:fs"));
38
+ const node_path_1 = __importDefault(require("node:path"));
39
+ const load_esm_1 = require("../../utils/load-esm");
40
+ const purge_cache_1 = require("../../utils/purge-cache");
41
+ const version_1 = require("../../utils/version");
42
+ const options_1 = require("./options");
43
+ const schema_1 = require("./schema");
44
+ /**
45
+ * @experimental Direct usage of this function is considered experimental.
46
+ */
47
+ async function execute(options, context, transforms) {
48
+ // Determine project name from builder context target
49
+ const projectName = context.target?.project;
50
+ if (!projectName) {
51
+ context.logger.error(`The 'extract-i18n' builder requires a target to be specified.`);
52
+ return { success: false };
53
+ }
54
+ // Check Angular version.
55
+ (0, version_1.assertCompatibleAngularVersion)(context.workspaceRoot);
56
+ // Load the Angular localize package.
57
+ // The package is a peer dependency and might not be present
58
+ let localizeToolsModule;
59
+ try {
60
+ localizeToolsModule = await (0, load_esm_1.loadEsmModule)('@angular/localize/tools');
61
+ }
62
+ catch {
63
+ return {
64
+ success: false,
65
+ error: `i18n extraction requires the '@angular/localize' package.` +
66
+ ` You can add it by using 'ng add @angular/localize'.`,
67
+ };
68
+ }
69
+ // Normalize options
70
+ const normalizedOptions = await (0, options_1.normalizeOptions)(context, projectName, options);
71
+ const builderName = await context.getBuilderNameForTarget(normalizedOptions.browserTarget);
72
+ // Extract messages based on configured builder
73
+ let extractionResult;
74
+ if (builderName === '@angular-devkit/build-angular:application' ||
75
+ builderName === '@angular-devkit/build-angular:browser-esbuild') {
76
+ const { extractMessages } = await Promise.resolve().then(() => __importStar(require('./application-extraction')));
77
+ extractionResult = await extractMessages(normalizedOptions, builderName, context, localizeToolsModule.MessageExtractor);
78
+ }
79
+ else {
80
+ // Purge old build disk cache.
81
+ // Other build systems handle stale cache purging directly.
82
+ await (0, purge_cache_1.purgeStaleBuildCache)(context);
83
+ const { extractMessages } = await Promise.resolve().then(() => __importStar(require('./webpack-extraction')));
84
+ extractionResult = await extractMessages(normalizedOptions, builderName, context, transforms);
85
+ }
86
+ // Return the builder result if it failed
87
+ if (!extractionResult.builderResult.success) {
88
+ return extractionResult.builderResult;
89
+ }
90
+ // Perform duplicate message checks
91
+ const { checkDuplicateMessages } = localizeToolsModule;
92
+ // The filesystem is used to create a relative path for each file
93
+ // from the basePath. This relative path is then used in the error message.
94
+ const checkFileSystem = {
95
+ relative(from, to) {
96
+ return node_path_1.default.relative(from, to);
97
+ },
98
+ };
99
+ const diagnostics = checkDuplicateMessages(
100
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
101
+ checkFileSystem, extractionResult.messages, 'warning',
102
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
103
+ extractionResult.basePath);
104
+ if (diagnostics.messages.length > 0) {
105
+ context.logger.warn(diagnostics.formatDiagnostics(''));
106
+ }
107
+ // Serialize all extracted messages
108
+ const serializer = await createSerializer(localizeToolsModule, normalizedOptions.format, normalizedOptions.i18nOptions.sourceLocale, extractionResult.basePath, extractionResult.useLegacyIds, diagnostics);
109
+ const content = serializer.serialize(extractionResult.messages);
110
+ // Ensure directory exists
111
+ const outputPath = node_path_1.default.dirname(normalizedOptions.outFile);
112
+ if (!node_fs_1.default.existsSync(outputPath)) {
113
+ node_fs_1.default.mkdirSync(outputPath, { recursive: true });
114
+ }
115
+ // Write translation file
116
+ node_fs_1.default.writeFileSync(normalizedOptions.outFile, content);
117
+ if (normalizedOptions.progress) {
118
+ context.logger.info(`Extraction Complete. (Messages: ${extractionResult.messages.length})`);
119
+ }
120
+ return { success: true, outputPath: normalizedOptions.outFile };
121
+ }
122
+ exports.execute = execute;
123
+ async function createSerializer(localizeToolsModule, format, sourceLocale, basePath, useLegacyIds, diagnostics) {
124
+ const { XmbTranslationSerializer, LegacyMessageIdMigrationSerializer, ArbTranslationSerializer, Xliff1TranslationSerializer, Xliff2TranslationSerializer, SimpleJsonTranslationSerializer, } = localizeToolsModule;
125
+ switch (format) {
126
+ case schema_1.Format.Xmb:
127
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
128
+ return new XmbTranslationSerializer(basePath, useLegacyIds);
129
+ case schema_1.Format.Xlf:
130
+ case schema_1.Format.Xlif:
131
+ case schema_1.Format.Xliff:
132
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
133
+ return new Xliff1TranslationSerializer(sourceLocale, basePath, useLegacyIds, {});
134
+ case schema_1.Format.Xlf2:
135
+ case schema_1.Format.Xliff2:
136
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
137
+ return new Xliff2TranslationSerializer(sourceLocale, basePath, useLegacyIds, {});
138
+ case schema_1.Format.Json:
139
+ return new SimpleJsonTranslationSerializer(sourceLocale);
140
+ case schema_1.Format.LegacyMigrate:
141
+ return new LegacyMessageIdMigrationSerializer(diagnostics);
142
+ case schema_1.Format.Arb:
143
+ const fileSystem = {
144
+ relative(from, to) {
145
+ return node_path_1.default.relative(from, to);
146
+ },
147
+ };
148
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
149
+ return new ArbTranslationSerializer(sourceLocale, basePath, fileSystem);
150
+ }
151
+ }
152
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/extract-i18n/builder.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport type { Diagnostics } from '@angular/localize/tools';\nimport type { BuilderContext, BuilderOutput } from '@angular-devkit/architect';\nimport fs from 'node:fs';\nimport path from 'node:path';\nimport type webpack from 'webpack';\nimport type { ExecutionTransformer } from '../../transforms';\nimport { loadEsmModule } from '../../utils/load-esm';\nimport { purgeStaleBuildCache } from '../../utils/purge-cache';\nimport { assertCompatibleAngularVersion } from '../../utils/version';\nimport { normalizeOptions } from './options';\nimport { Schema as ExtractI18nBuilderOptions, Format } from './schema';\n\n/**\n * @experimental Direct usage of this function is considered experimental.\n */\nexport async function execute(\n  options: ExtractI18nBuilderOptions,\n  context: BuilderContext,\n  transforms?: {\n    webpackConfiguration?: ExecutionTransformer<webpack.Configuration>;\n  },\n): Promise<BuilderOutput> {\n  // Determine project name from builder context target\n  const projectName = context.target?.project;\n  if (!projectName) {\n    context.logger.error(`The 'extract-i18n' builder requires a target to be specified.`);\n\n    return { success: false };\n  }\n\n  // Check Angular version.\n  assertCompatibleAngularVersion(context.workspaceRoot);\n\n  // Load the Angular localize package.\n  // The package is a peer dependency and might not be present\n  let localizeToolsModule;\n  try {\n    localizeToolsModule = await loadEsmModule<typeof import('@angular/localize/tools')>(\n      '@angular/localize/tools',\n    );\n  } catch {\n    return {\n      success: false,\n      error:\n        `i18n extraction requires the '@angular/localize' package.` +\n        ` You can add it by using 'ng add @angular/localize'.`,\n    };\n  }\n\n  // Normalize options\n  const normalizedOptions = await normalizeOptions(context, projectName, options);\n  const builderName = await context.getBuilderNameForTarget(normalizedOptions.browserTarget);\n\n  // Extract messages based on configured builder\n  let extractionResult;\n  if (\n    builderName === '@angular-devkit/build-angular:application' ||\n    builderName === '@angular-devkit/build-angular:browser-esbuild'\n  ) {\n    const { extractMessages } = await import('./application-extraction');\n    extractionResult = await extractMessages(\n      normalizedOptions,\n      builderName,\n      context,\n      localizeToolsModule.MessageExtractor,\n    );\n  } else {\n    // Purge old build disk cache.\n    // Other build systems handle stale cache purging directly.\n    await purgeStaleBuildCache(context);\n\n    const { extractMessages } = await import('./webpack-extraction');\n    extractionResult = await extractMessages(normalizedOptions, builderName, context, transforms);\n  }\n\n  // Return the builder result if it failed\n  if (!extractionResult.builderResult.success) {\n    return extractionResult.builderResult;\n  }\n\n  // Perform duplicate message checks\n  const { checkDuplicateMessages } = localizeToolsModule;\n\n  // The filesystem is used to create a relative path for each file\n  // from the basePath.  This relative path is then used in the error message.\n  const checkFileSystem = {\n    relative(from: string, to: string): string {\n      return path.relative(from, to);\n    },\n  };\n  const diagnostics = checkDuplicateMessages(\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    checkFileSystem as any,\n    extractionResult.messages,\n    'warning',\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    extractionResult.basePath as any,\n  );\n  if (diagnostics.messages.length > 0) {\n    context.logger.warn(diagnostics.formatDiagnostics(''));\n  }\n\n  // Serialize all extracted messages\n  const serializer = await createSerializer(\n    localizeToolsModule,\n    normalizedOptions.format,\n    normalizedOptions.i18nOptions.sourceLocale,\n    extractionResult.basePath,\n    extractionResult.useLegacyIds,\n    diagnostics,\n  );\n  const content = serializer.serialize(extractionResult.messages);\n\n  // Ensure directory exists\n  const outputPath = path.dirname(normalizedOptions.outFile);\n  if (!fs.existsSync(outputPath)) {\n    fs.mkdirSync(outputPath, { recursive: true });\n  }\n\n  // Write translation file\n  fs.writeFileSync(normalizedOptions.outFile, content);\n\n  if (normalizedOptions.progress) {\n    context.logger.info(`Extraction Complete. (Messages: ${extractionResult.messages.length})`);\n  }\n\n  return { success: true, outputPath: normalizedOptions.outFile };\n}\n\nasync function createSerializer(\n  localizeToolsModule: typeof import('@angular/localize/tools'),\n  format: Format,\n  sourceLocale: string,\n  basePath: string,\n  useLegacyIds: boolean,\n  diagnostics: Diagnostics,\n) {\n  const {\n    XmbTranslationSerializer,\n    LegacyMessageIdMigrationSerializer,\n    ArbTranslationSerializer,\n    Xliff1TranslationSerializer,\n    Xliff2TranslationSerializer,\n    SimpleJsonTranslationSerializer,\n  } = localizeToolsModule;\n\n  switch (format) {\n    case Format.Xmb:\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      return new XmbTranslationSerializer(basePath as any, useLegacyIds);\n    case Format.Xlf:\n    case Format.Xlif:\n    case Format.Xliff:\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      return new Xliff1TranslationSerializer(sourceLocale, basePath as any, useLegacyIds, {});\n    case Format.Xlf2:\n    case Format.Xliff2:\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      return new Xliff2TranslationSerializer(sourceLocale, basePath as any, useLegacyIds, {});\n    case Format.Json:\n      return new SimpleJsonTranslationSerializer(sourceLocale);\n    case Format.LegacyMigrate:\n      return new LegacyMessageIdMigrationSerializer(diagnostics);\n    case Format.Arb:\n      const fileSystem = {\n        relative(from: string, to: string): string {\n          return path.relative(from, to);\n        },\n      };\n\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      return new ArbTranslationSerializer(sourceLocale, basePath as any, fileSystem as any);\n  }\n}\n"]}
@@ -5,18 +5,8 @@
5
5
  * Use of this source code is governed by an MIT-style license that can be
6
6
  * found in the LICENSE file at https://angular.io/license
7
7
  */
8
- import { BuilderContext } from '@angular-devkit/architect';
9
- import { BuildResult } from '@angular-devkit/build-webpack';
10
- import { JsonObject } from '@angular-devkit/core';
11
- import webpack from 'webpack';
12
- import { ExecutionTransformer } from '../../transforms';
13
- import { Schema } from './schema';
14
- export type ExtractI18nBuilderOptions = Schema;
15
- /**
16
- * @experimental Direct usage of this function is considered experimental.
17
- */
18
- export declare function execute(options: ExtractI18nBuilderOptions, context: BuilderContext, transforms?: {
19
- webpackConfiguration?: ExecutionTransformer<webpack.Configuration>;
20
- }): Promise<BuildResult>;
21
- declare const _default: import("../../../../architect/src/internal").Builder<Schema & JsonObject>;
8
+ import { execute } from './builder';
9
+ import type { Schema as ExtractI18nBuilderOptions } from './schema';
10
+ export { ExtractI18nBuilderOptions, execute };
11
+ declare const _default: import("../../../../architect/src/internal").Builder<ExtractI18nBuilderOptions & import("../../../../core/src").JsonObject>;
22
12
  export default _default;