UNPKG

@angular-devkit/build-angular

Version:
287 lines (286 loc) • 43 kB
"use strict"; /** * @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 */ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.serveWebpackBrowser = void 0; const build_webpack_1 = require("@angular-devkit/build-webpack"); const core_1 = require("@angular-devkit/core"); const path = __importStar(require("path")); const rxjs_1 = require("rxjs"); const url = __importStar(require("url")); const configs_1 = require("../../tools/webpack/configs"); const index_html_webpack_plugin_1 = require("../../tools/webpack/plugins/index-html-webpack-plugin"); const service_worker_plugin_1 = require("../../tools/webpack/plugins/service-worker-plugin"); const stats_1 = require("../../tools/webpack/utils/stats"); const utils_1 = require("../../utils"); const color_1 = require("../../utils/color"); const i18n_options_1 = require("../../utils/i18n-options"); const load_translations_1 = require("../../utils/load-translations"); const package_chunk_sort_1 = require("../../utils/package-chunk-sort"); const version_1 = require("../../utils/version"); const webpack_browser_config_1 = require("../../utils/webpack-browser-config"); const webpack_diagnostics_1 = require("../../utils/webpack-diagnostics"); const schema_1 = require("../browser/schema"); /** * Reusable implementation of the Angular Webpack development server builder. * @param options Dev Server options. * @param builderName The name of the builder used to build the application. * @param context The build context. * @param transforms A map of transforms that can be used to hook into some logic (such as * transforming webpack configuration before passing it to webpack). */ // eslint-disable-next-line max-lines-per-function function serveWebpackBrowser(options, builderName, context, transforms = {}) { // Check Angular version. const { logger, workspaceRoot } = context; (0, version_1.assertCompatibleAngularVersion)(workspaceRoot); async function setup() { if (options.hmr) { logger.warn(core_1.tags.stripIndents `NOTICE: Hot Module Replacement (HMR) is enabled for the dev server. See https://webpack.js.org/guides/hot-module-replacement for information on working with HMR for Webpack.`); } // Get the browser configuration from the target name. const rawBrowserOptions = (await context.getTargetOptions(options.browserTarget)); if (rawBrowserOptions.outputHashing && rawBrowserOptions.outputHashing !== schema_1.OutputHashing.None) { // Disable output hashing for dev build as this can cause memory leaks // See: https://github.com/webpack/webpack-dev-server/issues/377#issuecomment-241258405 rawBrowserOptions.outputHashing = schema_1.OutputHashing.None; logger.warn(`Warning: 'outputHashing' option is disabled when using the dev-server.`); } const browserOptions = (await context.validateOptions({ ...rawBrowserOptions, watch: options.watch, verbose: options.verbose, // In dev server we should not have budgets because of extra libs such as socks-js budgets: undefined, }, builderName)); const { styles, scripts } = (0, utils_1.normalizeOptimization)(browserOptions.optimization); if (scripts || styles.minify) { logger.error(core_1.tags.stripIndents ` **************************************************************************************** This is a simple server for use in testing or debugging Angular applications locally. It hasn't been reviewed for security issues. DON'T USE IT FOR PRODUCTION! **************************************************************************************** `); } const { config, i18n } = await (0, webpack_browser_config_1.generateI18nBrowserWebpackConfigFromContext)(browserOptions, context, (wco) => [(0, configs_1.getDevServerConfig)(wco), (0, configs_1.getCommonConfig)(wco), (0, configs_1.getStylesConfig)(wco)], options); if (!config.devServer) { throw new Error('Webpack Dev Server configuration was not set.'); } let locale; if (i18n.shouldInline) { // Dev-server only supports one locale locale = [...i18n.inlineLocales][0]; } else if (i18n.hasDefinedSourceLocale) { // use source locale if not localizing locale = i18n.sourceLocale; } let webpackConfig = config; // If a locale is defined, setup localization if (locale) { if (i18n.inlineLocales.size > 1) { throw new Error('The development server only supports localizing a single locale per build.'); } await setupLocalize(locale, i18n, browserOptions, webpackConfig, options.cacheOptions, context); } if (transforms.webpackConfiguration) { webpackConfig = await transforms.webpackConfiguration(webpackConfig); } webpackConfig.plugins ?? (webpackConfig.plugins = []); if (browserOptions.index) { const { scripts = [], styles = [], baseHref } = browserOptions; const entrypoints = (0, package_chunk_sort_1.generateEntryPoints)({ scripts, styles, // The below is needed as otherwise HMR for CSS will break. // styles.js and runtime.js needs to be loaded as a non-module scripts as otherwise `document.currentScript` will be null. // https://github.com/webpack-contrib/mini-css-extract-plugin/blob/90445dd1d81da0c10b9b0e8a17b417d0651816b8/src/hmr/hotModuleReplacement.js#L39 isHMREnabled: !!webpackConfig.devServer?.hot, }); webpackConfig.plugins.push(new index_html_webpack_plugin_1.IndexHtmlWebpackPlugin({ indexPath: path.resolve(workspaceRoot, (0, webpack_browser_config_1.getIndexInputFile)(browserOptions.index)), outputPath: (0, webpack_browser_config_1.getIndexOutputFile)(browserOptions.index), baseHref, entrypoints, deployUrl: browserOptions.deployUrl, sri: browserOptions.subresourceIntegrity, cache: options.cacheOptions, postTransform: transforms.indexHtml, optimization: (0, utils_1.normalizeOptimization)(browserOptions.optimization), crossOrigin: browserOptions.crossOrigin, lang: locale, })); } if (browserOptions.serviceWorker) { webpackConfig.plugins.push(new service_worker_plugin_1.ServiceWorkerPlugin({ baseHref: browserOptions.baseHref, root: context.workspaceRoot, projectRoot: options.projectRoot, ngswConfigPath: browserOptions.ngswConfigPath, })); } return { browserOptions, webpackConfig, }; } return (0, rxjs_1.from)(setup()).pipe((0, rxjs_1.switchMap)(({ browserOptions, webpackConfig }) => { return (0, build_webpack_1.runWebpackDevServer)(webpackConfig, context, { logging: transforms.logging || (0, stats_1.createWebpackLoggingCallback)(browserOptions, logger), webpackFactory: require('webpack'), webpackDevServerFactory: require('webpack-dev-server'), }).pipe((0, rxjs_1.concatMap)(async (buildEvent, index) => { const webpackRawStats = buildEvent.webpackStats; if (!webpackRawStats) { throw new Error('Webpack stats build result is required.'); } // Resolve serve address. const publicPath = webpackConfig.devServer?.devMiddleware?.publicPath; const serverAddress = url.format({ protocol: options.ssl ? 'https' : 'http', hostname: options.host === '0.0.0.0' ? 'localhost' : options.host, port: buildEvent.port, pathname: typeof publicPath === 'string' ? publicPath : undefined, }); if (index === 0) { logger.info('\n' + core_1.tags.oneLine ` ** Angular Live Development Server is listening on ${options.host}:${buildEvent.port}, open your browser on ${serverAddress} ** ` + '\n'); if (options.open) { const open = (await Promise.resolve().then(() => __importStar(require('open')))).default; await open(serverAddress); } } if (buildEvent.success) { logger.info(`\n${color_1.colors.greenBright(color_1.colors.symbols.check)} Compiled successfully.`); } else { logger.info(`\n${color_1.colors.redBright(color_1.colors.symbols.cross)} Failed to compile.`); } return { ...buildEvent, baseUrl: serverAddress, stats: (0, stats_1.generateBuildEventStats)(webpackRawStats, browserOptions), }; })); })); } exports.serveWebpackBrowser = serveWebpackBrowser; async function setupLocalize(locale, i18n, browserOptions, webpackConfig, cacheOptions, context) { const localeDescription = i18n.locales[locale]; // Modify main entrypoint to include locale data if (localeDescription?.dataPath && typeof webpackConfig.entry === 'object' && !Array.isArray(webpackConfig.entry) && webpackConfig.entry['main']) { if (Array.isArray(webpackConfig.entry['main'])) { webpackConfig.entry['main'].unshift(localeDescription.dataPath); } else { webpackConfig.entry['main'] = [ localeDescription.dataPath, webpackConfig.entry['main'], ]; } } let missingTranslationBehavior = browserOptions.i18nMissingTranslation || 'ignore'; let translation = localeDescription?.translation || {}; if (locale === i18n.sourceLocale) { missingTranslationBehavior = 'ignore'; translation = {}; } const i18nLoaderOptions = { locale, missingTranslationBehavior, translation: i18n.shouldInline ? translation : undefined, translationFiles: localeDescription?.files.map((file) => path.resolve(context.workspaceRoot, file.path)), }; const i18nRule = { test: /\.[cm]?[tj]sx?$/, enforce: 'post', use: [ { loader: require.resolve('../../tools/babel/webpack-loader'), options: { cacheDirectory: (cacheOptions.enabled && path.join(cacheOptions.path, 'babel-dev-server-i18n')) || false, cacheIdentifier: JSON.stringify({ locale, translationIntegrity: localeDescription?.files.map((file) => file.integrity), }), i18n: i18nLoaderOptions, }, }, ], }; // Get the rules and ensure the Webpack configuration is setup properly const rules = webpackConfig.module?.rules || []; if (!webpackConfig.module) { webpackConfig.module = { rules }; } else if (!webpackConfig.module.rules) { webpackConfig.module.rules = rules; } rules.push(i18nRule); // Add a plugin to reload translation files on rebuilds const loader = await (0, load_translations_1.createTranslationLoader)(); // eslint-disable-next-line @typescript-eslint/no-non-null-assertion webpackConfig.plugins.push({ apply: (compiler) => { compiler.hooks.thisCompilation.tap('build-angular', (compilation) => { if (i18n.shouldInline && i18nLoaderOptions.translation === undefined) { // Reload translations (0, i18n_options_1.loadTranslations)(locale, localeDescription, context.workspaceRoot, loader, { warn(message) { (0, webpack_diagnostics_1.addWarning)(compilation, message); }, error(message) { (0, webpack_diagnostics_1.addError)(compilation, message); }, }, undefined, browserOptions.i18nDuplicateTranslation); i18nLoaderOptions.translation = localeDescription.translation ?? {}; } compilation.hooks.finishModules.tap('build-angular', () => { // After loaders are finished, clear out the now unneeded translations i18nLoaderOptions.translation = undefined; }); }); }, }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webpack-server.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/dev-server/webpack-server.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 } from '@angular-devkit/architect';\nimport {\n  DevServerBuildOutput,\n  WebpackLoggingCallback,\n  runWebpackDevServer,\n} from '@angular-devkit/build-webpack';\nimport { json, tags } from '@angular-devkit/core';\nimport * as path from 'path';\nimport { Observable, concatMap, from, switchMap } from 'rxjs';\nimport * as url from 'url';\nimport webpack from 'webpack';\nimport webpackDevServer from 'webpack-dev-server';\nimport { getCommonConfig, getDevServerConfig, getStylesConfig } from '../../tools/webpack/configs';\nimport { IndexHtmlWebpackPlugin } from '../../tools/webpack/plugins/index-html-webpack-plugin';\nimport { ServiceWorkerPlugin } from '../../tools/webpack/plugins/service-worker-plugin';\nimport {\n  BuildEventStats,\n  createWebpackLoggingCallback,\n  generateBuildEventStats,\n} from '../../tools/webpack/utils/stats';\nimport { ExecutionTransformer } from '../../transforms';\nimport { normalizeOptimization } from '../../utils';\nimport { colors } from '../../utils/color';\nimport { I18nOptions, loadTranslations } from '../../utils/i18n-options';\nimport { IndexHtmlTransform } from '../../utils/index-file/index-html-generator';\nimport { createTranslationLoader } from '../../utils/load-translations';\nimport { NormalizedCachedOptions } from '../../utils/normalize-cache';\nimport { generateEntryPoints } from '../../utils/package-chunk-sort';\nimport { assertCompatibleAngularVersion } from '../../utils/version';\nimport {\n  generateI18nBrowserWebpackConfigFromContext,\n  getIndexInputFile,\n  getIndexOutputFile,\n} from '../../utils/webpack-browser-config';\nimport { addError, addWarning } from '../../utils/webpack-diagnostics';\nimport { Schema as BrowserBuilderSchema, OutputHashing } from '../browser/schema';\nimport { NormalizedDevServerOptions } from './options';\n\n/**\n * @experimental Direct usage of this type is considered experimental.\n */\nexport type DevServerBuilderOutput = DevServerBuildOutput & {\n  baseUrl: string;\n  stats: BuildEventStats;\n};\n\n/**\n * Reusable implementation of the Angular Webpack development server builder.\n * @param options Dev Server options.\n * @param builderName The name of the builder used to build the application.\n * @param context The build context.\n * @param transforms A map of transforms that can be used to hook into some logic (such as\n *     transforming webpack configuration before passing it to webpack).\n */\n// eslint-disable-next-line max-lines-per-function\nexport function serveWebpackBrowser(\n  options: NormalizedDevServerOptions,\n  builderName: string,\n  context: BuilderContext,\n  transforms: {\n    webpackConfiguration?: ExecutionTransformer<webpack.Configuration>;\n    logging?: WebpackLoggingCallback;\n    indexHtml?: IndexHtmlTransform;\n  } = {},\n): Observable<DevServerBuilderOutput> {\n  // Check Angular version.\n  const { logger, workspaceRoot } = context;\n  assertCompatibleAngularVersion(workspaceRoot);\n\n  async function setup(): Promise<{\n    browserOptions: BrowserBuilderSchema;\n    webpackConfig: webpack.Configuration;\n  }> {\n    if (options.hmr) {\n      logger.warn(tags.stripIndents`NOTICE: Hot Module Replacement (HMR) is enabled for the dev server.\n      See https://webpack.js.org/guides/hot-module-replacement for information on working with HMR for Webpack.`);\n    }\n\n    // Get the browser configuration from the target name.\n    const rawBrowserOptions = (await context.getTargetOptions(\n      options.browserTarget,\n    )) as json.JsonObject & BrowserBuilderSchema;\n\n    if (rawBrowserOptions.outputHashing && rawBrowserOptions.outputHashing !== OutputHashing.None) {\n      // Disable output hashing for dev build as this can cause memory leaks\n      // See: https://github.com/webpack/webpack-dev-server/issues/377#issuecomment-241258405\n      rawBrowserOptions.outputHashing = OutputHashing.None;\n      logger.warn(`Warning: 'outputHashing' option is disabled when using the dev-server.`);\n    }\n\n    const browserOptions = (await context.validateOptions(\n      {\n        ...rawBrowserOptions,\n        watch: options.watch,\n        verbose: options.verbose,\n        // In dev server we should not have budgets because of extra libs such as socks-js\n        budgets: undefined,\n      } as json.JsonObject & BrowserBuilderSchema,\n      builderName,\n    )) as json.JsonObject & BrowserBuilderSchema;\n\n    const { styles, scripts } = normalizeOptimization(browserOptions.optimization);\n    if (scripts || styles.minify) {\n      logger.error(tags.stripIndents`\n        ****************************************************************************************\n        This is a simple server for use in testing or debugging Angular applications locally.\n        It hasn't been reviewed for security issues.\n\n        DON'T USE IT FOR PRODUCTION!\n        ****************************************************************************************\n      `);\n    }\n\n    const { config, i18n } = await generateI18nBrowserWebpackConfigFromContext(\n      browserOptions,\n      context,\n      (wco) => [getDevServerConfig(wco), getCommonConfig(wco), getStylesConfig(wco)],\n      options,\n    );\n\n    if (!config.devServer) {\n      throw new Error('Webpack Dev Server configuration was not set.');\n    }\n\n    let locale: string | undefined;\n    if (i18n.shouldInline) {\n      // Dev-server only supports one locale\n      locale = [...i18n.inlineLocales][0];\n    } else if (i18n.hasDefinedSourceLocale) {\n      // use source locale if not localizing\n      locale = i18n.sourceLocale;\n    }\n\n    let webpackConfig = config;\n\n    // If a locale is defined, setup localization\n    if (locale) {\n      if (i18n.inlineLocales.size > 1) {\n        throw new Error(\n          'The development server only supports localizing a single locale per build.',\n        );\n      }\n\n      await setupLocalize(\n        locale,\n        i18n,\n        browserOptions,\n        webpackConfig,\n        options.cacheOptions,\n        context,\n      );\n    }\n\n    if (transforms.webpackConfiguration) {\n      webpackConfig = await transforms.webpackConfiguration(webpackConfig);\n    }\n\n    webpackConfig.plugins ??= [];\n\n    if (browserOptions.index) {\n      const { scripts = [], styles = [], baseHref } = browserOptions;\n      const entrypoints = generateEntryPoints({\n        scripts,\n        styles,\n        // The below is needed as otherwise HMR for CSS will break.\n        // styles.js and runtime.js needs to be loaded as a non-module scripts as otherwise `document.currentScript` will be null.\n        // https://github.com/webpack-contrib/mini-css-extract-plugin/blob/90445dd1d81da0c10b9b0e8a17b417d0651816b8/src/hmr/hotModuleReplacement.js#L39\n        isHMREnabled: !!webpackConfig.devServer?.hot,\n      });\n\n      webpackConfig.plugins.push(\n        new IndexHtmlWebpackPlugin({\n          indexPath: path.resolve(workspaceRoot, getIndexInputFile(browserOptions.index)),\n          outputPath: getIndexOutputFile(browserOptions.index),\n          baseHref,\n          entrypoints,\n          deployUrl: browserOptions.deployUrl,\n          sri: browserOptions.subresourceIntegrity,\n          cache: options.cacheOptions,\n          postTransform: transforms.indexHtml,\n          optimization: normalizeOptimization(browserOptions.optimization),\n          crossOrigin: browserOptions.crossOrigin,\n          lang: locale,\n        }),\n      );\n    }\n\n    if (browserOptions.serviceWorker) {\n      webpackConfig.plugins.push(\n        new ServiceWorkerPlugin({\n          baseHref: browserOptions.baseHref,\n          root: context.workspaceRoot,\n          projectRoot: options.projectRoot,\n          ngswConfigPath: browserOptions.ngswConfigPath,\n        }),\n      );\n    }\n\n    return {\n      browserOptions,\n      webpackConfig,\n    };\n  }\n\n  return from(setup()).pipe(\n    switchMap(({ browserOptions, webpackConfig }) => {\n      return runWebpackDevServer(webpackConfig, context, {\n        logging: transforms.logging || createWebpackLoggingCallback(browserOptions, logger),\n        webpackFactory: require('webpack') as typeof webpack,\n        webpackDevServerFactory: require('webpack-dev-server') as typeof webpackDevServer,\n      }).pipe(\n        concatMap(async (buildEvent, index) => {\n          const webpackRawStats = buildEvent.webpackStats;\n          if (!webpackRawStats) {\n            throw new Error('Webpack stats build result is required.');\n          }\n\n          // Resolve serve address.\n          const publicPath = webpackConfig.devServer?.devMiddleware?.publicPath;\n\n          const serverAddress = url.format({\n            protocol: options.ssl ? 'https' : 'http',\n            hostname: options.host === '0.0.0.0' ? 'localhost' : options.host,\n            port: buildEvent.port,\n            pathname: typeof publicPath === 'string' ? publicPath : undefined,\n          });\n\n          if (index === 0) {\n            logger.info(\n              '\\n' +\n                tags.oneLine`\n              **\n              Angular Live Development Server is listening on ${options.host}:${buildEvent.port},\n              open your browser on ${serverAddress}\n              **\n            ` +\n                '\\n',\n            );\n\n            if (options.open) {\n              const open = (await import('open')).default;\n              await open(serverAddress);\n            }\n          }\n\n          if (buildEvent.success) {\n            logger.info(`\\n${colors.greenBright(colors.symbols.check)} Compiled successfully.`);\n          } else {\n            logger.info(`\\n${colors.redBright(colors.symbols.cross)} Failed to compile.`);\n          }\n\n          return {\n            ...buildEvent,\n            baseUrl: serverAddress,\n            stats: generateBuildEventStats(webpackRawStats, browserOptions),\n          } as DevServerBuilderOutput;\n        }),\n      );\n    }),\n  );\n}\n\nasync function setupLocalize(\n  locale: string,\n  i18n: I18nOptions,\n  browserOptions: BrowserBuilderSchema,\n  webpackConfig: webpack.Configuration,\n  cacheOptions: NormalizedCachedOptions,\n  context: BuilderContext,\n) {\n  const localeDescription = i18n.locales[locale];\n\n  // Modify main entrypoint to include locale data\n  if (\n    localeDescription?.dataPath &&\n    typeof webpackConfig.entry === 'object' &&\n    !Array.isArray(webpackConfig.entry) &&\n    webpackConfig.entry['main']\n  ) {\n    if (Array.isArray(webpackConfig.entry['main'])) {\n      webpackConfig.entry['main'].unshift(localeDescription.dataPath);\n    } else {\n      webpackConfig.entry['main'] = [\n        localeDescription.dataPath,\n        webpackConfig.entry['main'] as string,\n      ];\n    }\n  }\n\n  let missingTranslationBehavior = browserOptions.i18nMissingTranslation || 'ignore';\n  let translation = localeDescription?.translation || {};\n\n  if (locale === i18n.sourceLocale) {\n    missingTranslationBehavior = 'ignore';\n    translation = {};\n  }\n\n  const i18nLoaderOptions = {\n    locale,\n    missingTranslationBehavior,\n    translation: i18n.shouldInline ? translation : undefined,\n    translationFiles: localeDescription?.files.map((file) =>\n      path.resolve(context.workspaceRoot, file.path),\n    ),\n  };\n\n  const i18nRule: webpack.RuleSetRule = {\n    test: /\\.[cm]?[tj]sx?$/,\n    enforce: 'post',\n    use: [\n      {\n        loader: require.resolve('../../tools/babel/webpack-loader'),\n        options: {\n          cacheDirectory:\n            (cacheOptions.enabled && path.join(cacheOptions.path, 'babel-dev-server-i18n')) ||\n            false,\n          cacheIdentifier: JSON.stringify({\n            locale,\n            translationIntegrity: localeDescription?.files.map((file) => file.integrity),\n          }),\n          i18n: i18nLoaderOptions,\n        },\n      },\n    ],\n  };\n\n  // Get the rules and ensure the Webpack configuration is setup properly\n  const rules = webpackConfig.module?.rules || [];\n  if (!webpackConfig.module) {\n    webpackConfig.module = { rules };\n  } else if (!webpackConfig.module.rules) {\n    webpackConfig.module.rules = rules;\n  }\n\n  rules.push(i18nRule);\n\n  // Add a plugin to reload translation files on rebuilds\n  const loader = await createTranslationLoader();\n  // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n  webpackConfig.plugins!.push({\n    apply: (compiler: webpack.Compiler) => {\n      compiler.hooks.thisCompilation.tap('build-angular', (compilation) => {\n        if (i18n.shouldInline && i18nLoaderOptions.translation === undefined) {\n          // Reload translations\n          loadTranslations(\n            locale,\n            localeDescription,\n            context.workspaceRoot,\n            loader,\n            {\n              warn(message) {\n                addWarning(compilation, message);\n              },\n              error(message) {\n                addError(compilation, message);\n              },\n            },\n            undefined,\n            browserOptions.i18nDuplicateTranslation,\n          );\n\n          i18nLoaderOptions.translation = localeDescription.translation ?? {};\n        }\n\n        compilation.hooks.finishModules.tap('build-angular', () => {\n          // After loaders are finished, clear out the now unneeded translations\n          i18nLoaderOptions.translation = undefined;\n        });\n      });\n    },\n  });\n}\n"]}