UNPKG

55.7 kBJavaScriptView Raw
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 */
9var __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}));
20var __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});
25var __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};
32Object.defineProperty(exports, "__esModule", { value: true });
33exports.buildWebpackBrowser = exports.BUILD_TIMEOUT = void 0;
34const architect_1 = require("@angular-devkit/architect");
35const build_webpack_1 = require("@angular-devkit/build-webpack");
36const fs = __importStar(require("fs"));
37const path = __importStar(require("path"));
38const rxjs_1 = require("rxjs");
39const operators_1 = require("rxjs/operators");
40const utils_1 = require("../../utils");
41const bundle_calculator_1 = require("../../utils/bundle-calculator");
42const color_1 = require("../../utils/color");
43const copy_assets_1 = require("../../utils/copy-assets");
44const error_1 = require("../../utils/error");
45const i18n_inlining_1 = require("../../utils/i18n-inlining");
46const index_html_generator_1 = require("../../utils/index-file/index-html-generator");
47const normalize_cache_1 = require("../../utils/normalize-cache");
48const output_paths_1 = require("../../utils/output-paths");
49const package_chunk_sort_1 = require("../../utils/package-chunk-sort");
50const purge_cache_1 = require("../../utils/purge-cache");
51const service_worker_1 = require("../../utils/service-worker");
52const spinner_1 = require("../../utils/spinner");
53const version_1 = require("../../utils/version");
54const webpack_browser_config_1 = require("../../utils/webpack-browser-config");
55const configs_1 = require("../../webpack/configs");
56const async_chunks_1 = require("../../webpack/utils/async-chunks");
57const helpers_1 = require("../../webpack/utils/helpers");
58const stats_1 = require("../../webpack/utils/stats");
59/**
60 * Maximum time in milliseconds for single build/rebuild
61 * This accounts for CI variability.
62 */
63exports.BUILD_TIMEOUT = 30000;
64async function initialize(options, context, webpackConfigurationTransform) {
65 const originalOutputPath = options.outputPath;
66 // Assets are processed directly by the builder except when watching
67 const adjustedOptions = options.watch ? options : { ...options, assets: [] };
68 const { config, projectRoot, projectSourceRoot, i18n } = await (0, webpack_browser_config_1.generateI18nBrowserWebpackConfigFromContext)(adjustedOptions, context, (wco) => [
69 (0, configs_1.getCommonConfig)(wco),
70 (0, configs_1.getStylesConfig)(wco),
71 ]);
72 let transformedConfig;
73 if (webpackConfigurationTransform) {
74 transformedConfig = await webpackConfigurationTransform(config);
75 }
76 if (options.deleteOutputPath) {
77 (0, utils_1.deleteOutputDir)(context.workspaceRoot, originalOutputPath);
78 }
79 return { config: transformedConfig || config, projectRoot, projectSourceRoot, i18n };
80}
81/**
82 * @experimental Direct usage of this function is considered experimental.
83 */
84// eslint-disable-next-line max-lines-per-function
85function buildWebpackBrowser(options, context, transforms = {}) {
86 var _a;
87 const projectName = (_a = context.target) === null || _a === void 0 ? void 0 : _a.project;
88 if (!projectName) {
89 throw new Error('The builder requires a target.');
90 }
91 const baseOutputPath = path.resolve(context.workspaceRoot, options.outputPath);
92 let outputPaths;
93 // Check Angular version.
94 (0, version_1.assertCompatibleAngularVersion)(context.workspaceRoot);
95 return (0, rxjs_1.from)(context.getProjectMetadata(projectName)).pipe((0, operators_1.switchMap)(async (projectMetadata) => {
96 var _a;
97 var _b;
98 // Purge old build disk cache.
99 await (0, purge_cache_1.purgeStaleBuildCache)(context);
100 // Initialize builder
101 const initialization = await initialize(options, context, transforms.webpackConfiguration);
102 // Add index file to watched files.
103 if (options.watch) {
104 const indexInputFile = path.join(context.workspaceRoot, (0, webpack_browser_config_1.getIndexInputFile)(options.index));
105 (_a = (_b = initialization.config).plugins) !== null && _a !== void 0 ? _a : (_b.plugins = []);
106 initialization.config.plugins.push({
107 apply: (compiler) => {
108 compiler.hooks.thisCompilation.tap('build-angular', (compilation) => {
109 compilation.fileDependencies.add(indexInputFile);
110 });
111 },
112 });
113 }
114 return {
115 ...initialization,
116 cacheOptions: (0, normalize_cache_1.normalizeCacheOptions)(projectMetadata, context.workspaceRoot),
117 };
118 }), (0, operators_1.switchMap)(
119 // eslint-disable-next-line max-lines-per-function
120 ({ config, projectRoot, projectSourceRoot, i18n, cacheOptions }) => {
121 const normalizedOptimization = (0, utils_1.normalizeOptimization)(options.optimization);
122 return (0, build_webpack_1.runWebpack)(config, context, {
123 webpackFactory: require('webpack'),
124 logging: transforms.logging ||
125 ((stats, config) => {
126 if (options.verbose) {
127 context.logger.info(stats.toString(config.stats));
128 }
129 }),
130 }).pipe((0, operators_1.concatMap)(
131 // eslint-disable-next-line max-lines-per-function
132 async (buildEvent) => {
133 var _a, _b, _c, _d, _e, _f, _g, _h;
134 const spinner = new spinner_1.Spinner();
135 spinner.enabled = options.progress !== false;
136 const { success, emittedFiles = [], outputPath: webpackOutputPath } = buildEvent;
137 const webpackRawStats = buildEvent.webpackStats;
138 if (!webpackRawStats) {
139 throw new Error('Webpack stats build result is required.');
140 }
141 // Fix incorrectly set `initial` value on chunks.
142 const extraEntryPoints = [
143 ...(0, helpers_1.normalizeExtraEntryPoints)(options.styles || [], 'styles'),
144 ...(0, helpers_1.normalizeExtraEntryPoints)(options.scripts || [], 'scripts'),
145 ];
146 const webpackStats = {
147 ...webpackRawStats,
148 chunks: (0, async_chunks_1.markAsyncChunksNonInitial)(webpackRawStats, extraEntryPoints),
149 };
150 if (!success) {
151 // If using bundle downleveling then there is only one build
152 // If it fails show any diagnostic messages and bail
153 if ((0, stats_1.statsHasWarnings)(webpackStats)) {
154 context.logger.warn((0, stats_1.statsWarningsToString)(webpackStats, { colors: true }));
155 }
156 if ((0, stats_1.statsHasErrors)(webpackStats)) {
157 context.logger.error((0, stats_1.statsErrorsToString)(webpackStats, { colors: true }));
158 }
159 return {
160 webpackStats: webpackRawStats,
161 output: { success: false },
162 };
163 }
164 else {
165 outputPaths = (0, output_paths_1.ensureOutputPaths)(baseOutputPath, i18n);
166 const scriptsEntryPointName = (0, helpers_1.normalizeExtraEntryPoints)(options.scripts || [], 'scripts').map((x) => x.bundleName);
167 if (i18n.shouldInline) {
168 const success = await (0, i18n_inlining_1.i18nInlineEmittedFiles)(context, emittedFiles, i18n, baseOutputPath, Array.from(outputPaths.values()), scriptsEntryPointName, webpackOutputPath, options.i18nMissingTranslation);
169 if (!success) {
170 return {
171 webpackStats: webpackRawStats,
172 output: { success: false },
173 };
174 }
175 }
176 // Check for budget errors and display them to the user.
177 const budgets = options.budgets;
178 let budgetFailures;
179 if (budgets === null || budgets === void 0 ? void 0 : budgets.length) {
180 budgetFailures = [...(0, bundle_calculator_1.checkBudgets)(budgets, webpackStats)];
181 for (const { severity, message } of budgetFailures) {
182 switch (severity) {
183 case bundle_calculator_1.ThresholdSeverity.Warning:
184 (_a = webpackStats.warnings) === null || _a === void 0 ? void 0 : _a.push({ message });
185 break;
186 case bundle_calculator_1.ThresholdSeverity.Error:
187 (_b = webpackStats.errors) === null || _b === void 0 ? void 0 : _b.push({ message });
188 break;
189 default:
190 assertNever(severity);
191 }
192 }
193 }
194 const buildSuccess = success && !(0, stats_1.statsHasErrors)(webpackStats);
195 if (buildSuccess) {
196 // Copy assets
197 if (!options.watch && ((_c = options.assets) === null || _c === void 0 ? void 0 : _c.length)) {
198 spinner.start('Copying assets...');
199 try {
200 await (0, copy_assets_1.copyAssets)((0, utils_1.normalizeAssetPatterns)(options.assets, context.workspaceRoot, projectRoot, projectSourceRoot), Array.from(outputPaths.values()), context.workspaceRoot);
201 spinner.succeed('Copying assets complete.');
202 }
203 catch (err) {
204 spinner.fail(color_1.colors.redBright('Copying of assets failed.'));
205 (0, error_1.assertIsError)(err);
206 return {
207 output: {
208 success: false,
209 error: 'Unable to copy assets: ' + err.message,
210 },
211 webpackStats: webpackRawStats,
212 };
213 }
214 }
215 if (options.index) {
216 spinner.start('Generating index html...');
217 const entrypoints = (0, package_chunk_sort_1.generateEntryPoints)({
218 scripts: (_d = options.scripts) !== null && _d !== void 0 ? _d : [],
219 styles: (_e = options.styles) !== null && _e !== void 0 ? _e : [],
220 });
221 const indexHtmlGenerator = new index_html_generator_1.IndexHtmlGenerator({
222 cache: cacheOptions,
223 indexPath: path.join(context.workspaceRoot, (0, webpack_browser_config_1.getIndexInputFile)(options.index)),
224 entrypoints,
225 deployUrl: options.deployUrl,
226 sri: options.subresourceIntegrity,
227 optimization: normalizedOptimization,
228 crossOrigin: options.crossOrigin,
229 postTransform: transforms.indexHtml,
230 });
231 let hasErrors = false;
232 for (const [locale, outputPath] of outputPaths.entries()) {
233 try {
234 const { content, warnings, errors } = await indexHtmlGenerator.process({
235 baseHref: (_f = getLocaleBaseHref(i18n, locale)) !== null && _f !== void 0 ? _f : options.baseHref,
236 // i18nLocale is used when Ivy is disabled
237 lang: locale || undefined,
238 outputPath,
239 files: mapEmittedFilesToFileInfo(emittedFiles),
240 });
241 if (warnings.length || errors.length) {
242 spinner.stop();
243 warnings.forEach((m) => context.logger.warn(m));
244 errors.forEach((m) => {
245 context.logger.error(m);
246 hasErrors = true;
247 });
248 spinner.start();
249 }
250 const indexOutput = path.join(outputPath, (0, webpack_browser_config_1.getIndexOutputFile)(options.index));
251 await fs.promises.mkdir(path.dirname(indexOutput), { recursive: true });
252 await fs.promises.writeFile(indexOutput, content);
253 }
254 catch (error) {
255 spinner.fail('Index html generation failed.');
256 (0, error_1.assertIsError)(error);
257 return {
258 webpackStats: webpackRawStats,
259 output: { success: false, error: error.message },
260 };
261 }
262 }
263 if (hasErrors) {
264 spinner.fail('Index html generation failed.');
265 return {
266 webpackStats: webpackRawStats,
267 output: { success: false },
268 };
269 }
270 else {
271 spinner.succeed('Index html generation complete.');
272 }
273 }
274 if (options.serviceWorker) {
275 spinner.start('Generating service worker...');
276 for (const [locale, outputPath] of outputPaths.entries()) {
277 try {
278 await (0, service_worker_1.augmentAppWithServiceWorker)(projectRoot, context.workspaceRoot, outputPath, (_h = (_g = getLocaleBaseHref(i18n, locale)) !== null && _g !== void 0 ? _g : options.baseHref) !== null && _h !== void 0 ? _h : '/', options.ngswConfigPath);
279 }
280 catch (error) {
281 spinner.fail('Service worker generation failed.');
282 (0, error_1.assertIsError)(error);
283 return {
284 webpackStats: webpackRawStats,
285 output: { success: false, error: error.message },
286 };
287 }
288 }
289 spinner.succeed('Service worker generation complete.');
290 }
291 }
292 (0, stats_1.webpackStatsLogger)(context.logger, webpackStats, config, budgetFailures);
293 return {
294 webpackStats: webpackRawStats,
295 output: { success: buildSuccess },
296 };
297 }
298 }), (0, operators_1.map)(({ output: event, webpackStats }) => ({
299 ...event,
300 stats: (0, stats_1.generateBuildEventStats)(webpackStats, options),
301 baseOutputPath,
302 outputPath: baseOutputPath,
303 outputPaths: (outputPaths && Array.from(outputPaths.values())) || [baseOutputPath],
304 outputs: (outputPaths &&
305 [...outputPaths.entries()].map(([locale, path]) => {
306 var _a;
307 return ({
308 locale,
309 path,
310 baseHref: (_a = getLocaleBaseHref(i18n, locale)) !== null && _a !== void 0 ? _a : options.baseHref,
311 });
312 })) || {
313 path: baseOutputPath,
314 baseHref: options.baseHref,
315 },
316 })));
317 }));
318 function getLocaleBaseHref(i18n, locale) {
319 var _a, _b;
320 if (i18n.locales[locale] && ((_a = i18n.locales[locale]) === null || _a === void 0 ? void 0 : _a.baseHref) !== '') {
321 return (0, utils_1.urlJoin)(options.baseHref || '', (_b = i18n.locales[locale].baseHref) !== null && _b !== void 0 ? _b : `/${locale}/`);
322 }
323 return undefined;
324 }
325}
326exports.buildWebpackBrowser = buildWebpackBrowser;
327function assertNever(input) {
328 throw new Error(`Unexpected call to assertNever() with input: ${JSON.stringify(input, null /* replacer */, 4 /* tabSize */)}`);
329}
330function mapEmittedFilesToFileInfo(files = []) {
331 const filteredFiles = [];
332 for (const { file, name, extension, initial } of files) {
333 if (name && initial) {
334 filteredFiles.push({ file, extension, name });
335 }
336 }
337 return filteredFiles;
338}
339exports.default = (0, architect_1.createBuilder)(buildWebpackBrowser);
340//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/browser/index.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 { BuilderContext, BuilderOutput, createBuilder } from '@angular-devkit/architect';\nimport { EmittedFiles, WebpackLoggingCallback, runWebpack } from '@angular-devkit/build-webpack';\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { Observable, from } from 'rxjs';\nimport { concatMap, map, switchMap } from 'rxjs/operators';\nimport webpack, { StatsCompilation } from 'webpack';\nimport { ExecutionTransformer } from '../../transforms';\nimport {\n  deleteOutputDir,\n  normalizeAssetPatterns,\n  normalizeOptimization,\n  urlJoin,\n} from '../../utils';\nimport {\n  BudgetCalculatorResult,\n  ThresholdSeverity,\n  checkBudgets,\n} from '../../utils/bundle-calculator';\nimport { colors } from '../../utils/color';\nimport { copyAssets } from '../../utils/copy-assets';\nimport { assertIsError } from '../../utils/error';\nimport { i18nInlineEmittedFiles } from '../../utils/i18n-inlining';\nimport { I18nOptions } from '../../utils/i18n-options';\nimport { FileInfo } from '../../utils/index-file/augment-index-html';\nimport {\n  IndexHtmlGenerator,\n  IndexHtmlTransform,\n} from '../../utils/index-file/index-html-generator';\nimport { normalizeCacheOptions } from '../../utils/normalize-cache';\nimport { ensureOutputPaths } from '../../utils/output-paths';\nimport { generateEntryPoints } from '../../utils/package-chunk-sort';\nimport { purgeStaleBuildCache } from '../../utils/purge-cache';\nimport { augmentAppWithServiceWorker } from '../../utils/service-worker';\nimport { Spinner } from '../../utils/spinner';\nimport { assertCompatibleAngularVersion } from '../../utils/version';\nimport {\n  generateI18nBrowserWebpackConfigFromContext,\n  getIndexInputFile,\n  getIndexOutputFile,\n} from '../../utils/webpack-browser-config';\nimport { getCommonConfig, getStylesConfig } from '../../webpack/configs';\nimport { markAsyncChunksNonInitial } from '../../webpack/utils/async-chunks';\nimport { normalizeExtraEntryPoints } from '../../webpack/utils/helpers';\nimport {\n  BuildEventStats,\n  generateBuildEventStats,\n  statsErrorsToString,\n  statsHasErrors,\n  statsHasWarnings,\n  statsWarningsToString,\n  webpackStatsLogger,\n} from '../../webpack/utils/stats';\nimport { Schema as BrowserBuilderSchema } from './schema';\n\n/**\n * @experimental Direct usage of this type is considered experimental.\n */\nexport type BrowserBuilderOutput = BuilderOutput & {\n  stats: BuildEventStats;\n\n  baseOutputPath: string;\n  /**\n   * @deprecated in version 14. Use 'outputs' instead.\n   */\n  outputPaths: string[];\n  /**\n   * @deprecated in version 9. Use 'outputs' instead.\n   */\n  outputPath: string;\n\n  outputs: {\n    locale?: string;\n    path: string;\n    baseHref?: string;\n  }[];\n};\n\n/**\n * Maximum time in milliseconds for single build/rebuild\n * This accounts for CI variability.\n */\nexport const BUILD_TIMEOUT = 30_000;\n\nasync function initialize(\n  options: BrowserBuilderSchema,\n  context: BuilderContext,\n  webpackConfigurationTransform?: ExecutionTransformer<webpack.Configuration>,\n): Promise<{\n  config: webpack.Configuration;\n  projectRoot: string;\n  projectSourceRoot?: string;\n  i18n: I18nOptions;\n}> {\n  const originalOutputPath = options.outputPath;\n\n  // Assets are processed directly by the builder except when watching\n  const adjustedOptions = options.watch ? options : { ...options, assets: [] };\n\n  const { config, projectRoot, projectSourceRoot, i18n } =\n    await generateI18nBrowserWebpackConfigFromContext(adjustedOptions, context, (wco) => [\n      getCommonConfig(wco),\n      getStylesConfig(wco),\n    ]);\n\n  let transformedConfig;\n  if (webpackConfigurationTransform) {\n    transformedConfig = await webpackConfigurationTransform(config);\n  }\n\n  if (options.deleteOutputPath) {\n    deleteOutputDir(context.workspaceRoot, originalOutputPath);\n  }\n\n  return { config: transformedConfig || config, projectRoot, projectSourceRoot, i18n };\n}\n\n/**\n * @experimental Direct usage of this function is considered experimental.\n */\n// eslint-disable-next-line max-lines-per-function\nexport function buildWebpackBrowser(\n  options: BrowserBuilderSchema,\n  context: BuilderContext,\n  transforms: {\n    webpackConfiguration?: ExecutionTransformer<webpack.Configuration>;\n    logging?: WebpackLoggingCallback;\n    indexHtml?: IndexHtmlTransform;\n  } = {},\n): Observable<BrowserBuilderOutput> {\n  const projectName = context.target?.project;\n  if (!projectName) {\n    throw new Error('The builder requires a target.');\n  }\n\n  const baseOutputPath = path.resolve(context.workspaceRoot, options.outputPath);\n  let outputPaths: undefined | Map<string, string>;\n\n  // Check Angular version.\n  assertCompatibleAngularVersion(context.workspaceRoot);\n\n  return from(context.getProjectMetadata(projectName)).pipe(\n    switchMap(async (projectMetadata) => {\n      // Purge old build disk cache.\n      await purgeStaleBuildCache(context);\n\n      // Initialize builder\n      const initialization = await initialize(options, context, transforms.webpackConfiguration);\n\n      // Add index file to watched files.\n      if (options.watch) {\n        const indexInputFile = path.join(context.workspaceRoot, getIndexInputFile(options.index));\n        initialization.config.plugins ??= [];\n        initialization.config.plugins.push({\n          apply: (compiler: webpack.Compiler) => {\n            compiler.hooks.thisCompilation.tap('build-angular', (compilation) => {\n              compilation.fileDependencies.add(indexInputFile);\n            });\n          },\n        });\n      }\n\n      return {\n        ...initialization,\n        cacheOptions: normalizeCacheOptions(projectMetadata, context.workspaceRoot),\n      };\n    }),\n    switchMap(\n      // eslint-disable-next-line max-lines-per-function\n      ({ config, projectRoot, projectSourceRoot, i18n, cacheOptions }) => {\n        const normalizedOptimization = normalizeOptimization(options.optimization);\n\n        return runWebpack(config, context, {\n          webpackFactory: require('webpack') as typeof webpack,\n          logging:\n            transforms.logging ||\n            ((stats, config) => {\n              if (options.verbose) {\n                context.logger.info(stats.toString(config.stats));\n              }\n            }),\n        }).pipe(\n          concatMap(\n            // eslint-disable-next-line max-lines-per-function\n            async (\n              buildEvent,\n            ): Promise<{ output: BuilderOutput; webpackStats: StatsCompilation }> => {\n              const spinner = new Spinner();\n              spinner.enabled = options.progress !== false;\n\n              const { success, emittedFiles = [], outputPath: webpackOutputPath } = buildEvent;\n              const webpackRawStats = buildEvent.webpackStats;\n              if (!webpackRawStats) {\n                throw new Error('Webpack stats build result is required.');\n              }\n\n              // Fix incorrectly set `initial` value on chunks.\n              const extraEntryPoints = [\n                ...normalizeExtraEntryPoints(options.styles || [], 'styles'),\n                ...normalizeExtraEntryPoints(options.scripts || [], 'scripts'),\n              ];\n\n              const webpackStats = {\n                ...webpackRawStats,\n                chunks: markAsyncChunksNonInitial(webpackRawStats, extraEntryPoints),\n              };\n\n              if (!success) {\n                // If using bundle downleveling then there is only one build\n                // If it fails show any diagnostic messages and bail\n                if (statsHasWarnings(webpackStats)) {\n                  context.logger.warn(statsWarningsToString(webpackStats, { colors: true }));\n                }\n                if (statsHasErrors(webpackStats)) {\n                  context.logger.error(statsErrorsToString(webpackStats, { colors: true }));\n                }\n\n                return {\n                  webpackStats: webpackRawStats,\n                  output: { success: false },\n                };\n              } else {\n                outputPaths = ensureOutputPaths(baseOutputPath, i18n);\n\n                const scriptsEntryPointName = normalizeExtraEntryPoints(\n                  options.scripts || [],\n                  'scripts',\n                ).map((x) => x.bundleName);\n\n                if (i18n.shouldInline) {\n                  const success = await i18nInlineEmittedFiles(\n                    context,\n                    emittedFiles,\n                    i18n,\n                    baseOutputPath,\n                    Array.from(outputPaths.values()),\n                    scriptsEntryPointName,\n                    webpackOutputPath,\n                    options.i18nMissingTranslation,\n                  );\n                  if (!success) {\n                    return {\n                      webpackStats: webpackRawStats,\n                      output: { success: false },\n                    };\n                  }\n                }\n\n                // Check for budget errors and display them to the user.\n                const budgets = options.budgets;\n                let budgetFailures: BudgetCalculatorResult[] | undefined;\n                if (budgets?.length) {\n                  budgetFailures = [...checkBudgets(budgets, webpackStats)];\n                  for (const { severity, message } of budgetFailures) {\n                    switch (severity) {\n                      case ThresholdSeverity.Warning:\n                        webpackStats.warnings?.push({ message });\n                        break;\n                      case ThresholdSeverity.Error:\n                        webpackStats.errors?.push({ message });\n                        break;\n                      default:\n                        assertNever(severity);\n                    }\n                  }\n                }\n\n                const buildSuccess = success && !statsHasErrors(webpackStats);\n                if (buildSuccess) {\n                  // Copy assets\n                  if (!options.watch && options.assets?.length) {\n                    spinner.start('Copying assets...');\n                    try {\n                      await copyAssets(\n                        normalizeAssetPatterns(\n                          options.assets,\n                          context.workspaceRoot,\n                          projectRoot,\n                          projectSourceRoot,\n                        ),\n                        Array.from(outputPaths.values()),\n                        context.workspaceRoot,\n                      );\n                      spinner.succeed('Copying assets complete.');\n                    } catch (err) {\n                      spinner.fail(colors.redBright('Copying of assets failed.'));\n                      assertIsError(err);\n\n                      return {\n                        output: {\n                          success: false,\n                          error: 'Unable to copy assets: ' + err.message,\n                        },\n                        webpackStats: webpackRawStats,\n                      };\n                    }\n                  }\n\n                  if (options.index) {\n                    spinner.start('Generating index html...');\n\n                    const entrypoints = generateEntryPoints({\n                      scripts: options.scripts ?? [],\n                      styles: options.styles ?? [],\n                    });\n\n                    const indexHtmlGenerator = new IndexHtmlGenerator({\n                      cache: cacheOptions,\n                      indexPath: path.join(context.workspaceRoot, getIndexInputFile(options.index)),\n                      entrypoints,\n                      deployUrl: options.deployUrl,\n                      sri: options.subresourceIntegrity,\n                      optimization: normalizedOptimization,\n                      crossOrigin: options.crossOrigin,\n                      postTransform: transforms.indexHtml,\n                    });\n\n                    let hasErrors = false;\n                    for (const [locale, outputPath] of outputPaths.entries()) {\n                      try {\n                        const { content, warnings, errors } = await indexHtmlGenerator.process({\n                          baseHref: getLocaleBaseHref(i18n, locale) ?? options.baseHref,\n                          // i18nLocale is used when Ivy is disabled\n                          lang: locale || undefined,\n                          outputPath,\n                          files: mapEmittedFilesToFileInfo(emittedFiles),\n                        });\n\n                        if (warnings.length || errors.length) {\n                          spinner.stop();\n                          warnings.forEach((m) => context.logger.warn(m));\n                          errors.forEach((m) => {\n                            context.logger.error(m);\n                            hasErrors = true;\n                          });\n                          spinner.start();\n                        }\n\n                        const indexOutput = path.join(\n                          outputPath,\n                          getIndexOutputFile(options.index),\n                        );\n                        await fs.promises.mkdir(path.dirname(indexOutput), { recursive: true });\n                        await fs.promises.writeFile(indexOutput, content);\n                      } catch (error) {\n                        spinner.fail('Index html generation failed.');\n                        assertIsError(error);\n\n                        return {\n                          webpackStats: webpackRawStats,\n                          output: { success: false, error: error.message },\n                        };\n                      }\n                    }\n\n                    if (hasErrors) {\n                      spinner.fail('Index html generation failed.');\n\n                      return {\n                        webpackStats: webpackRawStats,\n                        output: { success: false },\n                      };\n                    } else {\n                      spinner.succeed('Index html generation complete.');\n                    }\n                  }\n\n                  if (options.serviceWorker) {\n                    spinner.start('Generating service worker...');\n                    for (const [locale, outputPath] of outputPaths.entries()) {\n                      try {\n                        await augmentAppWithServiceWorker(\n                          projectRoot,\n                          context.workspaceRoot,\n                          outputPath,\n                          getLocaleBaseHref(i18n, locale) ?? options.baseHref ?? '/',\n                          options.ngswConfigPath,\n                        );\n                      } catch (error) {\n                        spinner.fail('Service worker generation failed.');\n                        assertIsError(error);\n\n                        return {\n                          webpackStats: webpackRawStats,\n                          output: { success: false, error: error.message },\n                        };\n                      }\n                    }\n\n                    spinner.succeed('Service worker generation complete.');\n                  }\n                }\n\n                webpackStatsLogger(context.logger, webpackStats, config, budgetFailures);\n\n                return {\n                  webpackStats: webpackRawStats,\n                  output: { success: buildSuccess },\n                };\n              }\n            },\n          ),\n          map(\n            ({ output: event, webpackStats }) =>\n              ({\n                ...event,\n                stats: generateBuildEventStats(webpackStats, options),\n                baseOutputPath,\n                outputPath: baseOutputPath,\n                outputPaths: (outputPaths && Array.from(outputPaths.values())) || [baseOutputPath],\n                outputs: (outputPaths &&\n                  [...outputPaths.entries()].map(([locale, path]) => ({\n                    locale,\n                    path,\n                    baseHref: getLocaleBaseHref(i18n, locale) ?? options.baseHref,\n                  }))) || {\n                  path: baseOutputPath,\n                  baseHref: options.baseHref,\n                },\n              } as BrowserBuilderOutput),\n          ),\n        );\n      },\n    ),\n  );\n\n  function getLocaleBaseHref(i18n: I18nOptions, locale: string): string | undefined {\n    if (i18n.locales[locale] && i18n.locales[locale]?.baseHref !== '') {\n      return urlJoin(options.baseHref || '', i18n.locales[locale].baseHref ?? `/${locale}/`);\n    }\n\n    return undefined;\n  }\n}\n\nfunction assertNever(input: never): never {\n  throw new Error(\n    `Unexpected call to assertNever() with input: ${JSON.stringify(\n      input,\n      null /* replacer */,\n      4 /* tabSize */,\n    )}`,\n  );\n}\n\nfunction mapEmittedFilesToFileInfo(files: EmittedFiles[] = []): FileInfo[] {\n  const filteredFiles: FileInfo[] = [];\n  for (const { file, name, extension, initial } of files) {\n    if (name && initial) {\n      filteredFiles.push({ file, extension, name });\n    }\n  }\n\n  return filteredFiles;\n}\n\nexport default createBuilder<BrowserBuilderSchema>(buildWebpackBrowser);\n"]}
\No newline at end of file