diff --git a/package.json b/package.json index 4d6f4a54..e2642b61 100644 --- a/package.json +++ b/package.json @@ -1,15 +1,15 @@ { "name": "@angular-devkit/build-angular", - "version": "17.0.0-next.5+sha-11449b1", + "version": "17.0.0-next.5+sha-c5f3ec7", "description": "Angular Webpack Build Facade", "main": "src/index.js", "typings": "src/index.d.ts", "builders": "builders.json", "dependencies": { "@ampproject/remapping": "2.2.1", - "@angular-devkit/architect": "github:angular/angular-devkit-architect-builds#11449b1b8", - "@angular-devkit/build-webpack": "github:angular/angular-devkit-build-webpack-builds#11449b1b8", - "@angular-devkit/core": "github:angular/angular-devkit-core-builds#11449b1b8", + "@angular-devkit/architect": "github:angular/angular-devkit-architect-builds#c5f3ec71f", + "@angular-devkit/build-webpack": "github:angular/angular-devkit-build-webpack-builds#c5f3ec71f", + "@angular-devkit/core": "github:angular/angular-devkit-core-builds#c5f3ec71f", "@babel/core": "7.22.17", "@babel/generator": "7.22.15", "@babel/helper-annotate-as-pure": "7.22.5", @@ -20,7 +20,7 @@ "@babel/preset-env": "7.22.15", "@babel/runtime": "7.22.15", "@discoveryjs/json-ext": "0.5.7", - "@ngtools/webpack": "github:angular/ngtools-webpack-builds#11449b1b8", + "@ngtools/webpack": "github:angular/ngtools-webpack-builds#c5f3ec71f", "@vitejs/plugin-basic-ssl": "1.0.1", "ansi-colors": "4.1.3", "autoprefixer": "10.4.15", diff --git a/src/builders/application/execute-build.js b/src/builders/application/execute-build.js index 9aa5bd75..eb499dbc 100644 --- a/src/builders/application/execute-build.js +++ b/src/builders/application/execute-build.js @@ -27,12 +27,18 @@ const environment_options_1 = require("../../utils/environment-options"); const prerender_1 = require("../../utils/server-rendering/prerender"); const service_worker_1 = require("../../utils/service-worker"); const supported_browsers_1 = require("../../utils/supported-browsers"); +const i18n_1 = require("./i18n"); // eslint-disable-next-line max-lines-per-function async function executeBuild(options, context, rebuildState) { const startTime = process.hrtime.bigint(); const { projectRoot, workspaceRoot, serviceWorker, optimizationOptions, serverEntryPoint, assets, indexHtmlOptions, cacheOptions, prerenderOptions, appShellOptions, ssrOptions, verbose, } = options; const browsers = (0, supported_browsers_1.getSupportedBrowsers)(projectRoot, context.logger); const target = (0, utils_1.transformSupportedBrowsersToTargets)(browsers); + // Load active translations if inlining + // TODO: Integrate into watch mode and only load changed translations + if (options.i18nOptions.shouldInline) { + await (0, i18n_1.loadActiveTranslations)(context, options.i18nOptions); + } // Reuse rebuild state or create new bundle contexts for code and global stylesheets let bundlerContexts = rebuildState?.rebuildContexts; const codeBundleCache = rebuildState?.codeBundleCache ?? @@ -89,11 +95,15 @@ async function executeBuild(options, context, rebuildState) { */ let indexContentOutputNoCssInlining; // Generate index HTML file - if (indexHtmlOptions) { - const { content, contentWithoutCriticalCssInlined, errors, warnings } = await (0, index_html_generator_1.generateIndexHtml)(initialFiles, executionResult, { + // If localization is enabled, index generation is handled in the inlining process. + // NOTE: Localization with SSR is not currently supported. + if (indexHtmlOptions && !options.i18nOptions.shouldInline) { + const { content, contentWithoutCriticalCssInlined, errors, warnings } = await (0, index_html_generator_1.generateIndexHtml)(initialFiles, executionResult.outputFiles, { ...options, optimizationOptions, - }); + }, + // Set lang attribute to the defined source locale if present + options.i18nOptions.hasDefinedSourceLocale ? options.i18nOptions.sourceLocale : undefined); indexContentOutputNoCssInlining = contentWithoutCriticalCssInlined; printWarningsAndErrorsToConsole(context, warnings, errors); executionResult.addOutputFile(indexHtmlOptions.output, content); @@ -144,6 +154,10 @@ async function executeBuild(options, context, rebuildState) { (0, utils_1.logBuildStats)(context, metafile, initialFiles, estimatedTransferSizes); const buildTime = Number(process.hrtime.bigint() - startTime) / 10 ** 9; context.logger.info(`Application bundle generation complete. [${buildTime.toFixed(3)} seconds]`); + // Perform i18n translation inlining if enabled + if (options.i18nOptions.shouldInline) { + await (0, i18n_1.inlineI18n)(options, executionResult, initialFiles); + } return executionResult; } exports.executeBuild = executeBuild; @@ -155,4 +169,4 @@ function printWarningsAndErrorsToConsole(context, warnings, errors) { context.logger.warn(warning); } } -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execute-build.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/application/execute-build.ts"],"names":[],"mappings":";AAAA;;;;;;GAMG;;;;;;AAGH,8DAAiC;AACjC,iFAA8E;AAC9E,yFAGqD;AACrD,yEAAqE;AACrE,2FAA6F;AAC7F,2EAA4E;AAC5E,uEAAsF;AACtF,qEAAoF;AACpF,mFAA6E;AAC7E,6EAAwE;AACxE,qDAKmC;AACnC,yDAAqD;AACrD,yEAA6D;AAC7D,sEAAwE;AACxE,+DAAgF;AAChF,uEAAsE;AAGtE,kDAAkD;AAC3C,KAAK,UAAU,YAAY,CAChC,OAA0C,EAC1C,OAAuB,EACvB,YAA2B;IAE3B,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;IAE1C,MAAM,EACJ,WAAW,EACX,aAAa,EACb,aAAa,EACb,mBAAmB,EACnB,gBAAgB,EAChB,MAAM,EACN,gBAAgB,EAChB,YAAY,EACZ,gBAAgB,EAChB,eAAe,EACf,UAAU,EACV,OAAO,GACR,GAAG,OAAO,CAAC;IAEZ,MAAM,QAAQ,GAAG,IAAA,yCAAoB,EAAC,WAAW,EAAE,OAAO,CAAC,MAAM,CAAC,CAAC;IACnE,MAAM,MAAM,GAAG,IAAA,2CAAmC,EAAC,QAAQ,CAAC,CAAC;IAE7D,oFAAoF;IACpF,IAAI,eAAe,GAAG,YAAY,EAAE,eAAe,CAAC;IACpD,MAAM,eAAe,GACnB,YAAY,EAAE,eAAe;QAC7B,IAAI,iCAAe,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IAC5E,IAAI,eAAe,KAAK,SAAS,EAAE;QACjC,eAAe,GAAG,EAAE,CAAC;QAErB,2BAA2B;QAC3B,eAAe,CAAC,IAAI,CAClB,IAAI,gCAAc,CAChB,aAAa,EACb,CAAC,CAAC,OAAO,CAAC,KAAK,EACf,IAAA,wDAA8B,EAAC,OAAO,EAAE,MAAM,EAAE,eAAe,CAAC,CACjE,CACF,CAAC;QAEF,qBAAqB;QACrB,IAAI,OAAO,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;YACnC,KAAK,MAAM,OAAO,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE;gBACnC,MAAM,aAAa,GAAG,IAAA,+CAA+B,EACnD,OAAO,EACP,MAAM,EACN,OAAO,EACP,eAAe,EAAE,eAAe,CACjC,CAAC;gBACF,IAAI,aAAa,EAAE;oBACjB,eAAe,CAAC,IAAI,CAClB,IAAI,gCAAc,CAAC,aAAa,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,aAAa,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,CACjF,CAAC;iBACH;aACF;SACF;QAED,iBAAiB;QACjB,IAAI,OAAO,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE;YACpC,KAAK,MAAM,OAAO,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,EAAE;gBACnC,MAAM,aAAa,GAAG,IAAA,iDAAgC,EAAC,OAAO,EAAE,OAAO,CAAC,CAAC;gBACzE,IAAI,aAAa,EAAE;oBACjB,eAAe,CAAC,IAAI,CAClB,IAAI,gCAAc,CAAC,aAAa,EAAE,CAAC,CAAC,OAAO,CAAC,KAAK,EAAE,aAAa,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,CACjF,CAAC;iBACH;aACF;SACF;QAED,0BAA0B;QAC1B,IAAI,gBAAgB,EAAE;YACpB,eAAe,CAAC,IAAI,CAClB,IAAI,gCAAc,CAChB,aAAa,EACb,CAAC,CAAC,OAAO,CAAC,KAAK,EACf,IAAA,uDAA6B,EAC3B,OAAO;YACP,sFAAsF;YACtF,sEAAsE;YACtE,CAAC,GAAG,MAAM,EAAE,WAAW,CAAC,EACxB,eAAe,CAChB,EACD,GAAG,EAAE,CAAC,KAAK,CACZ,CACF,CAAC;SACH;KACF;IAED,MAAM,cAAc,GAAG,MAAM,gCAAc,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;IAEvE,wDAAwD;IACxD,MAAM,IAAA,mBAAW,EAAC,OAAO,EAAE,cAAc,CAAC,CAAC;IAE3C,MAAM,eAAe,GAAG,IAAI,0CAAe,CAAC,eAAe,EAAE,eAAe,CAAC,CAAC;IAE9E,oCAAoC;IACpC,IAAI,cAAc,CAAC,MAAM,EAAE;QACzB,OAAO,eAAe,CAAC;KACxB;IAED,MAAM,EAAE,QAAQ,EAAE,YAAY,EAAE,WAAW,EAAE,GAAG,cAAc,CAAC;IAE/D,eAAe,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,WAAW,CAAC,CAAC;IAEjD,iEAAiE;IACjE,IAAI,mBAAmB,CAAC,OAAO,EAAE;QAC/B,MAAM,QAAQ,GAAG,IAAA,uCAAoB,EAAC,QAAQ,EAAE,OAAO,CAAC,2BAA2B,CAAC,CAAC;QACrF,MAAM,IAAA,mBAAW,EAAC,OAAO,EAAE,EAAE,QAAQ,EAAE,QAAQ,EAAE,CAAC,CAAC;KACpD;IAED;;;;OAIG;IACH,IAAI,+BAAmD,CAAC;IAExD,2BAA2B;IAC3B,IAAI,gBAAgB,EAAE;QACpB,MAAM,EAAE,OAAO,EAAE,gCAAgC,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,MAAM,IAAA,wCAAiB,EAC7F,YAAY,EACZ,eAAe,EACf;YACE,GAAG,OAAO;YACV,mBAAmB;SACpB,CACF,CAAC;QAEF,+BAA+B,GAAG,gCAAgC,CAAC;QACnE,+BAA+B,CAAC,OAAO,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAE3D,eAAe,CAAC,aAAa,CAAC,gBAAgB,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;QAEhE,IAAI,UAAU,EAAE;YACd,eAAe,CAAC,aAAa,CAAC,mBAAmB,EAAE,gCAAgC,CAAC,CAAC;SACtF;KACF;IAED,iCAAiC;IACjC,IAAI,gBAAgB,IAAI,eAAe,EAAE;QACvC,IAAA,qBAAM,EACJ,+BAA+B,EAC/B,4EAA4E,CAC7E,CAAC;QAEF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,MAAM,IAAA,0BAAc,EACvD,aAAa,EACb,eAAe,EACf,gBAAgB,EAChB,eAAe,CAAC,WAAW,EAC3B,+BAA+B,EAC/B,mBAAmB,CAAC,MAAM,CAAC,cAAc,EACzC,gCAAU,EACV,OAAO,CACR,CAAC;QAEF,+BAA+B,CAAC,OAAO,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAE3D,KAAK,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACpD,eAAe,CAAC,aAAa,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;SAC9C;KACF;IAED,cAAc;IACd,IAAI,MAAM,EAAE;QACV,8FAA8F;QAC9F,kGAAkG;QAClG,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,IAAA,wBAAU,EAAC,MAAM,EAAE,EAAE,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC;KACnF;IAED,4CAA4C;IAC5C,IAAI,OAAO,CAAC,KAAK,EAAE;QACjB,eAAe,CAAC,aAAa,CAAC,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;KAChF;IAED,+CAA+C;IAC/C,IAAI,OAAO,CAAC,eAAe,EAAE;QAC3B,eAAe,CAAC,aAAa,CAC3B,sBAAsB,EACtB,MAAM,IAAA,mCAAe,EAAC,QAAQ,EAAE,aAAa,CAAC,CAC/C,CAAC;KACH;IAED,sDAAsD;IACtD,IAAI,aAAa,EAAE;QACjB,IAAI;YACF,MAAM,mBAAmB,GAAG,MAAM,IAAA,mDAAkC,EAClE,aAAa,EACb,aAAa,EACb,OAAO,CAAC,QAAQ,IAAI,GAAG,EACvB,eAAe,CAAC,WAAW,EAC3B,eAAe,CAAC,UAAU,CAC3B,CAAC;YACF,eAAe,CAAC,aAAa,CAAC,WAAW,EAAE,mBAAmB,CAAC,QAAQ,CAAC,CAAC;YACzE,eAAe,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,mBAAmB,CAAC,UAAU,CAAC,CAAC;SACpE;QAAC,OAAO,KAAK,EAAE;YACd,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,KAAK,EAAE,CAAC,CAAC;YAE1E,OAAO,eAAe,CAAC;SACxB;KACF;IAED,6DAA6D;IAC7D,IAAI,sBAAsB,CAAC;IAC3B,IAAI,mBAAmB,CAAC,OAAO,IAAI,mBAAmB,CAAC,MAAM,CAAC,MAAM,EAAE;QACpE,sBAAsB,GAAG,MAAM,IAAA,uCAA+B,EAAC,eAAe,CAAC,WAAW,CAAC,CAAC;KAC7F;IAED,IAAA,qBAAa,EAAC,OAAO,EAAE,QAAQ,EAAE,YAAY,EAAE,sBAAsB,CAAC,CAAC;IAEvE,MAAM,SAAS,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,SAAS,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;IACxE,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,4CAA4C,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC;IAEjG,OAAO,eAAe,CAAC;AACzB,CAAC;AAxND,oCAwNC;AAED,SAAS,+BAA+B,CACtC,OAAuB,EACvB,QAAkB,EAClB,MAAgB;IAEhB,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE;QAC1B,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;KAC7B;IACD,KAAK,MAAM,OAAO,IAAI,QAAQ,EAAE;QAC9B,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;KAC9B;AACH,CAAC","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 assert from 'node:assert';\nimport { SourceFileCache } from '../../tools/esbuild/angular/compiler-plugin';\nimport {\n  createBrowserCodeBundleOptions,\n  createServerCodeBundleOptions,\n} from '../../tools/esbuild/application-code-bundle';\nimport { BundlerContext } from '../../tools/esbuild/bundler-context';\nimport { ExecutionResult, RebuildState } from '../../tools/esbuild/bundler-execution-result';\nimport { checkCommonJSModules } from '../../tools/esbuild/commonjs-checker';\nimport { createGlobalScriptsBundleOptions } from '../../tools/esbuild/global-scripts';\nimport { createGlobalStylesBundleOptions } from '../../tools/esbuild/global-styles';\nimport { generateIndexHtml } from '../../tools/esbuild/index-html-generator';\nimport { extractLicenses } from '../../tools/esbuild/license-extractor';\nimport {\n  calculateEstimatedTransferSizes,\n  logBuildStats,\n  logMessages,\n  transformSupportedBrowsersToTargets,\n} from '../../tools/esbuild/utils';\nimport { copyAssets } from '../../utils/copy-assets';\nimport { maxWorkers } from '../../utils/environment-options';\nimport { prerenderPages } from '../../utils/server-rendering/prerender';\nimport { augmentAppWithServiceWorkerEsbuild } from '../../utils/service-worker';\nimport { getSupportedBrowsers } from '../../utils/supported-browsers';\nimport { NormalizedApplicationBuildOptions } from './options';\n\n// eslint-disable-next-line max-lines-per-function\nexport async function executeBuild(\n  options: NormalizedApplicationBuildOptions,\n  context: BuilderContext,\n  rebuildState?: RebuildState,\n): Promise<ExecutionResult> {\n  const startTime = process.hrtime.bigint();\n\n  const {\n    projectRoot,\n    workspaceRoot,\n    serviceWorker,\n    optimizationOptions,\n    serverEntryPoint,\n    assets,\n    indexHtmlOptions,\n    cacheOptions,\n    prerenderOptions,\n    appShellOptions,\n    ssrOptions,\n    verbose,\n  } = options;\n\n  const browsers = getSupportedBrowsers(projectRoot, context.logger);\n  const target = transformSupportedBrowsersToTargets(browsers);\n\n  // Reuse rebuild state or create new bundle contexts for code and global stylesheets\n  let bundlerContexts = rebuildState?.rebuildContexts;\n  const codeBundleCache =\n    rebuildState?.codeBundleCache ??\n    new SourceFileCache(cacheOptions.enabled ? cacheOptions.path : undefined);\n  if (bundlerContexts === undefined) {\n    bundlerContexts = [];\n\n    // Browser application code\n    bundlerContexts.push(\n      new BundlerContext(\n        workspaceRoot,\n        !!options.watch,\n        createBrowserCodeBundleOptions(options, target, codeBundleCache),\n      ),\n    );\n\n    // Global Stylesheets\n    if (options.globalStyles.length > 0) {\n      for (const initial of [true, false]) {\n        const bundleOptions = createGlobalStylesBundleOptions(\n          options,\n          target,\n          initial,\n          codeBundleCache?.loadResultCache,\n        );\n        if (bundleOptions) {\n          bundlerContexts.push(\n            new BundlerContext(workspaceRoot, !!options.watch, bundleOptions, () => initial),\n          );\n        }\n      }\n    }\n\n    // Global Scripts\n    if (options.globalScripts.length > 0) {\n      for (const initial of [true, false]) {\n        const bundleOptions = createGlobalScriptsBundleOptions(options, initial);\n        if (bundleOptions) {\n          bundlerContexts.push(\n            new BundlerContext(workspaceRoot, !!options.watch, bundleOptions, () => initial),\n          );\n        }\n      }\n    }\n\n    // Server application code\n    if (serverEntryPoint) {\n      bundlerContexts.push(\n        new BundlerContext(\n          workspaceRoot,\n          !!options.watch,\n          createServerCodeBundleOptions(\n            options,\n            // NOTE: earlier versions of Node.js are not supported due to unsafe promise patching.\n            // See: https://github.com/angular/angular/pull/50552#issue-1737967592\n            [...target, 'node18.13'],\n            codeBundleCache,\n          ),\n          () => false,\n        ),\n      );\n    }\n  }\n\n  const bundlingResult = await BundlerContext.bundleAll(bundlerContexts);\n\n  // Log all warnings and errors generated during bundling\n  await logMessages(context, bundlingResult);\n\n  const executionResult = new ExecutionResult(bundlerContexts, codeBundleCache);\n\n  // Return if the bundling has errors\n  if (bundlingResult.errors) {\n    return executionResult;\n  }\n\n  const { metafile, initialFiles, outputFiles } = bundlingResult;\n\n  executionResult.outputFiles.push(...outputFiles);\n\n  // Check metafile for CommonJS module usage if optimizing scripts\n  if (optimizationOptions.scripts) {\n    const messages = checkCommonJSModules(metafile, options.allowedCommonJsDependencies);\n    await logMessages(context, { warnings: messages });\n  }\n\n  /**\n   * Index HTML content without CSS inlining to be used for server rendering (AppShell, SSG and SSR).\n   *\n   * NOTE: we don't perform critical CSS inlining as this will be done during server rendering.\n   */\n  let indexContentOutputNoCssInlining: string | undefined;\n\n  // Generate index HTML file\n  if (indexHtmlOptions) {\n    const { content, contentWithoutCriticalCssInlined, errors, warnings } = await generateIndexHtml(\n      initialFiles,\n      executionResult,\n      {\n        ...options,\n        optimizationOptions,\n      },\n    );\n\n    indexContentOutputNoCssInlining = contentWithoutCriticalCssInlined;\n    printWarningsAndErrorsToConsole(context, warnings, errors);\n\n    executionResult.addOutputFile(indexHtmlOptions.output, content);\n\n    if (ssrOptions) {\n      executionResult.addOutputFile('index.server.html', contentWithoutCriticalCssInlined);\n    }\n  }\n\n  // Pre-render (SSG) and App-shell\n  if (prerenderOptions || appShellOptions) {\n    assert(\n      indexContentOutputNoCssInlining,\n      'The \"index\" option is required when using the \"ssg\" or \"appShell\" options.',\n    );\n\n    const { output, warnings, errors } = await prerenderPages(\n      workspaceRoot,\n      appShellOptions,\n      prerenderOptions,\n      executionResult.outputFiles,\n      indexContentOutputNoCssInlining,\n      optimizationOptions.styles.inlineCritical,\n      maxWorkers,\n      verbose,\n    );\n\n    printWarningsAndErrorsToConsole(context, warnings, errors);\n\n    for (const [path, content] of Object.entries(output)) {\n      executionResult.addOutputFile(path, content);\n    }\n  }\n\n  // Copy assets\n  if (assets) {\n    // The webpack copy assets helper is used with no base paths defined. This prevents the helper\n    // from directly writing to disk. This should eventually be replaced with a more optimized helper.\n    executionResult.assetFiles.push(...(await copyAssets(assets, [], workspaceRoot)));\n  }\n\n  // Write metafile if stats option is enabled\n  if (options.stats) {\n    executionResult.addOutputFile('stats.json', JSON.stringify(metafile, null, 2));\n  }\n\n  // Extract and write licenses for used packages\n  if (options.extractLicenses) {\n    executionResult.addOutputFile(\n      '3rdpartylicenses.txt',\n      await extractLicenses(metafile, workspaceRoot),\n    );\n  }\n\n  // Augment the application with service worker support\n  if (serviceWorker) {\n    try {\n      const serviceWorkerResult = await augmentAppWithServiceWorkerEsbuild(\n        workspaceRoot,\n        serviceWorker,\n        options.baseHref || '/',\n        executionResult.outputFiles,\n        executionResult.assetFiles,\n      );\n      executionResult.addOutputFile('ngsw.json', serviceWorkerResult.manifest);\n      executionResult.assetFiles.push(...serviceWorkerResult.assetFiles);\n    } catch (error) {\n      context.logger.error(error instanceof Error ? error.message : `${error}`);\n\n      return executionResult;\n    }\n  }\n\n  // Calculate estimated transfer size if scripts are optimized\n  let estimatedTransferSizes;\n  if (optimizationOptions.scripts || optimizationOptions.styles.minify) {\n    estimatedTransferSizes = await calculateEstimatedTransferSizes(executionResult.outputFiles);\n  }\n\n  logBuildStats(context, metafile, initialFiles, estimatedTransferSizes);\n\n  const buildTime = Number(process.hrtime.bigint() - startTime) / 10 ** 9;\n  context.logger.info(`Application bundle generation complete. [${buildTime.toFixed(3)} seconds]`);\n\n  return executionResult;\n}\n\nfunction printWarningsAndErrorsToConsole(\n  context: BuilderContext,\n  warnings: string[],\n  errors: string[],\n): void {\n  for (const error of errors) {\n    context.logger.error(error);\n  }\n  for (const warning of warnings) {\n    context.logger.warn(warning);\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execute-build.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/application/execute-build.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 assert from 'node:assert';\nimport { SourceFileCache } from '../../tools/esbuild/angular/compiler-plugin';\nimport {\n  createBrowserCodeBundleOptions,\n  createServerCodeBundleOptions,\n} from '../../tools/esbuild/application-code-bundle';\nimport { BundlerContext } from '../../tools/esbuild/bundler-context';\nimport { ExecutionResult, RebuildState } from '../../tools/esbuild/bundler-execution-result';\nimport { checkCommonJSModules } from '../../tools/esbuild/commonjs-checker';\nimport { createGlobalScriptsBundleOptions } from '../../tools/esbuild/global-scripts';\nimport { createGlobalStylesBundleOptions } from '../../tools/esbuild/global-styles';\nimport { generateIndexHtml } from '../../tools/esbuild/index-html-generator';\nimport { extractLicenses } from '../../tools/esbuild/license-extractor';\nimport {\n  calculateEstimatedTransferSizes,\n  logBuildStats,\n  logMessages,\n  transformSupportedBrowsersToTargets,\n} from '../../tools/esbuild/utils';\nimport { copyAssets } from '../../utils/copy-assets';\nimport { maxWorkers } from '../../utils/environment-options';\nimport { prerenderPages } from '../../utils/server-rendering/prerender';\nimport { augmentAppWithServiceWorkerEsbuild } from '../../utils/service-worker';\nimport { getSupportedBrowsers } from '../../utils/supported-browsers';\nimport { inlineI18n, loadActiveTranslations } from './i18n';\nimport { NormalizedApplicationBuildOptions } from './options';\n\n// eslint-disable-next-line max-lines-per-function\nexport async function executeBuild(\n  options: NormalizedApplicationBuildOptions,\n  context: BuilderContext,\n  rebuildState?: RebuildState,\n): Promise<ExecutionResult> {\n  const startTime = process.hrtime.bigint();\n\n  const {\n    projectRoot,\n    workspaceRoot,\n    serviceWorker,\n    optimizationOptions,\n    serverEntryPoint,\n    assets,\n    indexHtmlOptions,\n    cacheOptions,\n    prerenderOptions,\n    appShellOptions,\n    ssrOptions,\n    verbose,\n  } = options;\n\n  const browsers = getSupportedBrowsers(projectRoot, context.logger);\n  const target = transformSupportedBrowsersToTargets(browsers);\n\n  // Load active translations if inlining\n  // TODO: Integrate into watch mode and only load changed translations\n  if (options.i18nOptions.shouldInline) {\n    await loadActiveTranslations(context, options.i18nOptions);\n  }\n\n  // Reuse rebuild state or create new bundle contexts for code and global stylesheets\n  let bundlerContexts = rebuildState?.rebuildContexts;\n  const codeBundleCache =\n    rebuildState?.codeBundleCache ??\n    new SourceFileCache(cacheOptions.enabled ? cacheOptions.path : undefined);\n  if (bundlerContexts === undefined) {\n    bundlerContexts = [];\n\n    // Browser application code\n    bundlerContexts.push(\n      new BundlerContext(\n        workspaceRoot,\n        !!options.watch,\n        createBrowserCodeBundleOptions(options, target, codeBundleCache),\n      ),\n    );\n\n    // Global Stylesheets\n    if (options.globalStyles.length > 0) {\n      for (const initial of [true, false]) {\n        const bundleOptions = createGlobalStylesBundleOptions(\n          options,\n          target,\n          initial,\n          codeBundleCache?.loadResultCache,\n        );\n        if (bundleOptions) {\n          bundlerContexts.push(\n            new BundlerContext(workspaceRoot, !!options.watch, bundleOptions, () => initial),\n          );\n        }\n      }\n    }\n\n    // Global Scripts\n    if (options.globalScripts.length > 0) {\n      for (const initial of [true, false]) {\n        const bundleOptions = createGlobalScriptsBundleOptions(options, initial);\n        if (bundleOptions) {\n          bundlerContexts.push(\n            new BundlerContext(workspaceRoot, !!options.watch, bundleOptions, () => initial),\n          );\n        }\n      }\n    }\n\n    // Server application code\n    if (serverEntryPoint) {\n      bundlerContexts.push(\n        new BundlerContext(\n          workspaceRoot,\n          !!options.watch,\n          createServerCodeBundleOptions(\n            options,\n            // NOTE: earlier versions of Node.js are not supported due to unsafe promise patching.\n            // See: https://github.com/angular/angular/pull/50552#issue-1737967592\n            [...target, 'node18.13'],\n            codeBundleCache,\n          ),\n          () => false,\n        ),\n      );\n    }\n  }\n\n  const bundlingResult = await BundlerContext.bundleAll(bundlerContexts);\n\n  // Log all warnings and errors generated during bundling\n  await logMessages(context, bundlingResult);\n\n  const executionResult = new ExecutionResult(bundlerContexts, codeBundleCache);\n\n  // Return if the bundling has errors\n  if (bundlingResult.errors) {\n    return executionResult;\n  }\n\n  const { metafile, initialFiles, outputFiles } = bundlingResult;\n\n  executionResult.outputFiles.push(...outputFiles);\n\n  // Check metafile for CommonJS module usage if optimizing scripts\n  if (optimizationOptions.scripts) {\n    const messages = checkCommonJSModules(metafile, options.allowedCommonJsDependencies);\n    await logMessages(context, { warnings: messages });\n  }\n\n  /**\n   * Index HTML content without CSS inlining to be used for server rendering (AppShell, SSG and SSR).\n   *\n   * NOTE: we don't perform critical CSS inlining as this will be done during server rendering.\n   */\n  let indexContentOutputNoCssInlining: string | undefined;\n\n  // Generate index HTML file\n  // If localization is enabled, index generation is handled in the inlining process.\n  // NOTE: Localization with SSR is not currently supported.\n  if (indexHtmlOptions && !options.i18nOptions.shouldInline) {\n    const { content, contentWithoutCriticalCssInlined, errors, warnings } = await generateIndexHtml(\n      initialFiles,\n      executionResult.outputFiles,\n      {\n        ...options,\n        optimizationOptions,\n      },\n      // Set lang attribute to the defined source locale if present\n      options.i18nOptions.hasDefinedSourceLocale ? options.i18nOptions.sourceLocale : undefined,\n    );\n\n    indexContentOutputNoCssInlining = contentWithoutCriticalCssInlined;\n    printWarningsAndErrorsToConsole(context, warnings, errors);\n\n    executionResult.addOutputFile(indexHtmlOptions.output, content);\n\n    if (ssrOptions) {\n      executionResult.addOutputFile('index.server.html', contentWithoutCriticalCssInlined);\n    }\n  }\n\n  // Pre-render (SSG) and App-shell\n  if (prerenderOptions || appShellOptions) {\n    assert(\n      indexContentOutputNoCssInlining,\n      'The \"index\" option is required when using the \"ssg\" or \"appShell\" options.',\n    );\n\n    const { output, warnings, errors } = await prerenderPages(\n      workspaceRoot,\n      appShellOptions,\n      prerenderOptions,\n      executionResult.outputFiles,\n      indexContentOutputNoCssInlining,\n      optimizationOptions.styles.inlineCritical,\n      maxWorkers,\n      verbose,\n    );\n\n    printWarningsAndErrorsToConsole(context, warnings, errors);\n\n    for (const [path, content] of Object.entries(output)) {\n      executionResult.addOutputFile(path, content);\n    }\n  }\n\n  // Copy assets\n  if (assets) {\n    // The webpack copy assets helper is used with no base paths defined. This prevents the helper\n    // from directly writing to disk. This should eventually be replaced with a more optimized helper.\n    executionResult.assetFiles.push(...(await copyAssets(assets, [], workspaceRoot)));\n  }\n\n  // Write metafile if stats option is enabled\n  if (options.stats) {\n    executionResult.addOutputFile('stats.json', JSON.stringify(metafile, null, 2));\n  }\n\n  // Extract and write licenses for used packages\n  if (options.extractLicenses) {\n    executionResult.addOutputFile(\n      '3rdpartylicenses.txt',\n      await extractLicenses(metafile, workspaceRoot),\n    );\n  }\n\n  // Augment the application with service worker support\n  if (serviceWorker) {\n    try {\n      const serviceWorkerResult = await augmentAppWithServiceWorkerEsbuild(\n        workspaceRoot,\n        serviceWorker,\n        options.baseHref || '/',\n        executionResult.outputFiles,\n        executionResult.assetFiles,\n      );\n      executionResult.addOutputFile('ngsw.json', serviceWorkerResult.manifest);\n      executionResult.assetFiles.push(...serviceWorkerResult.assetFiles);\n    } catch (error) {\n      context.logger.error(error instanceof Error ? error.message : `${error}`);\n\n      return executionResult;\n    }\n  }\n\n  // Calculate estimated transfer size if scripts are optimized\n  let estimatedTransferSizes;\n  if (optimizationOptions.scripts || optimizationOptions.styles.minify) {\n    estimatedTransferSizes = await calculateEstimatedTransferSizes(executionResult.outputFiles);\n  }\n\n  logBuildStats(context, metafile, initialFiles, estimatedTransferSizes);\n\n  const buildTime = Number(process.hrtime.bigint() - startTime) / 10 ** 9;\n  context.logger.info(`Application bundle generation complete. [${buildTime.toFixed(3)} seconds]`);\n\n  // Perform i18n translation inlining if enabled\n  if (options.i18nOptions.shouldInline) {\n    await inlineI18n(options, executionResult, initialFiles);\n  }\n\n  return executionResult;\n}\n\nfunction printWarningsAndErrorsToConsole(\n  context: BuilderContext,\n  warnings: string[],\n  errors: string[],\n): void {\n  for (const error of errors) {\n    context.logger.error(error);\n  }\n  for (const warning of warnings) {\n    context.logger.warn(warning);\n  }\n}\n"]} \ No newline at end of file diff --git a/src/builders/application/i18n.d.ts b/src/builders/application/i18n.d.ts new file mode 100644 index 00000000..842298b1 --- /dev/null +++ b/src/builders/application/i18n.d.ts @@ -0,0 +1,25 @@ +/** + * @license + * Copyright Google LLC All Rights Reserved. + * + * Use of this source code is governed by an MIT-style license that can be + * found in the LICENSE file at https://angular.io/license + */ +import { BuilderContext } from '@angular-devkit/architect'; +import { InitialFileRecord } from '../../tools/esbuild/bundler-context'; +import { ExecutionResult } from '../../tools/esbuild/bundler-execution-result'; +import { NormalizedApplicationBuildOptions } from './options'; +/** + * Inlines all active locales as specified by the application build options into all + * application JavaScript files created during the build. + * @param options The normalized application builder options used to create the build. + * @param executionResult The result of an executed build. + * @param initialFiles A map containing initial file information for the executed build. + */ +export declare function inlineI18n(options: NormalizedApplicationBuildOptions, executionResult: ExecutionResult, initialFiles: Map): Promise; +/** + * Loads all active translations using the translation loaders from the `@angular/localize` package. + * @param context The architect builder context for the current build. + * @param i18n The normalized i18n options to use. + */ +export declare function loadActiveTranslations(context: BuilderContext, i18n: NormalizedApplicationBuildOptions['i18nOptions']): Promise; diff --git a/src/builders/application/i18n.js b/src/builders/application/i18n.js new file mode 100644 index 00000000..97b49571 --- /dev/null +++ b/src/builders/application/i18n.js @@ -0,0 +1,110 @@ +"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 + */ +Object.defineProperty(exports, "__esModule", { value: true }); +exports.loadActiveTranslations = exports.inlineI18n = void 0; +const node_path_1 = require("node:path"); +const i18n_inliner_1 = require("../../tools/esbuild/i18n-inliner"); +const index_html_generator_1 = require("../../tools/esbuild/index-html-generator"); +const utils_1 = require("../../tools/esbuild/utils"); +const environment_options_1 = require("../../utils/environment-options"); +const i18n_options_1 = require("../../utils/i18n-options"); +const load_translations_1 = require("../../utils/load-translations"); +const url_1 = require("../../utils/url"); +/** + * Inlines all active locales as specified by the application build options into all + * application JavaScript files created during the build. + * @param options The normalized application builder options used to create the build. + * @param executionResult The result of an executed build. + * @param initialFiles A map containing initial file information for the executed build. + */ +async function inlineI18n(options, executionResult, initialFiles) { + // Create the multi-threaded inliner with common options and the files generated from the build. + const inliner = new i18n_inliner_1.I18nInliner({ + missingTranslation: options.i18nOptions.missingTranslationBehavior ?? 'warning', + outputFiles: executionResult.outputFiles, + shouldOptimize: options.optimizationOptions.scripts, + }, environment_options_1.maxWorkers); + // For each active locale, use the inliner to process the output files of the build. + const updatedOutputFiles = []; + const updatedAssetFiles = []; + try { + for (const locale of options.i18nOptions.inlineLocales) { + // A locale specific set of files is returned from the inliner. + const localeOutputFiles = await inliner.inlineForLocale(locale, options.i18nOptions.locales[locale].translation); + // Generate locale specific index HTML files + if (options.indexHtmlOptions) { + const { content, errors, warnings } = await (0, index_html_generator_1.generateIndexHtml)(initialFiles, localeOutputFiles, { + ...options, + baseHref: getLocaleBaseHref(options.baseHref, options.i18nOptions, locale) ?? options.baseHref, + }, locale); + localeOutputFiles.push((0, utils_1.createOutputFileFromText)(options.indexHtmlOptions.output, content)); + } + // Update directory with locale base + if (options.i18nOptions.flatOutput !== true) { + localeOutputFiles.forEach((file) => { + file.path = (0, node_path_1.join)(locale, file.path); + }); + for (const assetFile of executionResult.assetFiles) { + updatedAssetFiles.push({ + source: assetFile.source, + destination: (0, node_path_1.join)(locale, assetFile.destination), + }); + } + } + updatedOutputFiles.push(...localeOutputFiles); + } + } + finally { + await inliner.close(); + } + // Update the result with all localized files + executionResult.outputFiles = updatedOutputFiles; + // Assets are only changed if not using the flat output option + if (options.i18nOptions.flatOutput !== true) { + executionResult.assetFiles = updatedAssetFiles; + } +} +exports.inlineI18n = inlineI18n; +function getLocaleBaseHref(baseHref, i18n, locale) { + if (i18n.flatOutput) { + return undefined; + } + if (i18n.locales[locale] && i18n.locales[locale].baseHref !== '') { + return (0, url_1.urlJoin)(baseHref || '', i18n.locales[locale].baseHref ?? `/${locale}/`); + } + return undefined; +} +/** + * Loads all active translations using the translation loaders from the `@angular/localize` package. + * @param context The architect builder context for the current build. + * @param i18n The normalized i18n options to use. + */ +async function loadActiveTranslations(context, i18n) { + // Load locale data and translations (if present) + let loader; + for (const [locale, desc] of Object.entries(i18n.locales)) { + if (!i18n.inlineLocales.has(locale) && locale !== i18n.sourceLocale) { + continue; + } + if (!desc.files.length) { + continue; + } + loader ??= await (0, load_translations_1.createTranslationLoader)(); + (0, i18n_options_1.loadTranslations)(locale, desc, context.workspaceRoot, loader, { + warn(message) { + context.logger.warn(message); + }, + error(message) { + throw new Error(message); + }, + }, undefined, i18n.duplicateTranslationBehavior); + } +} +exports.loadActiveTranslations = loadActiveTranslations; +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/application/i18n.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 { join } from 'node:path';\nimport { InitialFileRecord } from '../../tools/esbuild/bundler-context';\nimport { ExecutionResult } from '../../tools/esbuild/bundler-execution-result';\nimport { I18nInliner } from '../../tools/esbuild/i18n-inliner';\nimport { generateIndexHtml } from '../../tools/esbuild/index-html-generator';\nimport { createOutputFileFromText } from '../../tools/esbuild/utils';\nimport { maxWorkers } from '../../utils/environment-options';\nimport { loadTranslations } from '../../utils/i18n-options';\nimport { createTranslationLoader } from '../../utils/load-translations';\nimport { urlJoin } from '../../utils/url';\nimport { NormalizedApplicationBuildOptions } from './options';\n\n/**\n * Inlines all active locales as specified by the application build options into all\n * application JavaScript files created during the build.\n * @param options The normalized application builder options used to create the build.\n * @param executionResult The result of an executed build.\n * @param initialFiles A map containing initial file information for the executed build.\n */\nexport async function inlineI18n(\n  options: NormalizedApplicationBuildOptions,\n  executionResult: ExecutionResult,\n  initialFiles: Map<string, InitialFileRecord>,\n): Promise<void> {\n  // Create the multi-threaded inliner with common options and the files generated from the build.\n  const inliner = new I18nInliner(\n    {\n      missingTranslation: options.i18nOptions.missingTranslationBehavior ?? 'warning',\n      outputFiles: executionResult.outputFiles,\n      shouldOptimize: options.optimizationOptions.scripts,\n    },\n    maxWorkers,\n  );\n\n  // For each active locale, use the inliner to process the output files of the build.\n  const updatedOutputFiles = [];\n  const updatedAssetFiles = [];\n  try {\n    for (const locale of options.i18nOptions.inlineLocales) {\n      // A locale specific set of files is returned from the inliner.\n      const localeOutputFiles = await inliner.inlineForLocale(\n        locale,\n        options.i18nOptions.locales[locale].translation,\n      );\n\n      // Generate locale specific index HTML files\n      if (options.indexHtmlOptions) {\n        const { content, errors, warnings } = await generateIndexHtml(\n          initialFiles,\n          localeOutputFiles,\n          {\n            ...options,\n            baseHref:\n              getLocaleBaseHref(options.baseHref, options.i18nOptions, locale) ?? options.baseHref,\n          },\n          locale,\n        );\n\n        localeOutputFiles.push(createOutputFileFromText(options.indexHtmlOptions.output, content));\n      }\n\n      // Update directory with locale base\n      if (options.i18nOptions.flatOutput !== true) {\n        localeOutputFiles.forEach((file) => {\n          file.path = join(locale, file.path);\n        });\n\n        for (const assetFile of executionResult.assetFiles) {\n          updatedAssetFiles.push({\n            source: assetFile.source,\n            destination: join(locale, assetFile.destination),\n          });\n        }\n      }\n\n      updatedOutputFiles.push(...localeOutputFiles);\n    }\n  } finally {\n    await inliner.close();\n  }\n\n  // Update the result with all localized files\n  executionResult.outputFiles = updatedOutputFiles;\n\n  // Assets are only changed if not using the flat output option\n  if (options.i18nOptions.flatOutput !== true) {\n    executionResult.assetFiles = updatedAssetFiles;\n  }\n}\n\nfunction getLocaleBaseHref(\n  baseHref: string | undefined,\n  i18n: NormalizedApplicationBuildOptions['i18nOptions'],\n  locale: string,\n): string | undefined {\n  if (i18n.flatOutput) {\n    return undefined;\n  }\n\n  if (i18n.locales[locale] && i18n.locales[locale].baseHref !== '') {\n    return urlJoin(baseHref || '', i18n.locales[locale].baseHref ?? `/${locale}/`);\n  }\n\n  return undefined;\n}\n\n/**\n * Loads all active translations using the translation loaders from the `@angular/localize` package.\n * @param context The architect builder context for the current build.\n * @param i18n The normalized i18n options to use.\n */\nexport async function loadActiveTranslations(\n  context: BuilderContext,\n  i18n: NormalizedApplicationBuildOptions['i18nOptions'],\n) {\n  // Load locale data and translations (if present)\n  let loader;\n  for (const [locale, desc] of Object.entries(i18n.locales)) {\n    if (!i18n.inlineLocales.has(locale) && locale !== i18n.sourceLocale) {\n      continue;\n    }\n\n    if (!desc.files.length) {\n      continue;\n    }\n\n    loader ??= await createTranslationLoader();\n\n    loadTranslations(\n      locale,\n      desc,\n      context.workspaceRoot,\n      loader,\n      {\n        warn(message) {\n          context.logger.warn(message);\n        },\n        error(message) {\n          throw new Error(message);\n        },\n      },\n      undefined,\n      i18n.duplicateTranslationBehavior,\n    );\n  }\n}\n"]} \ No newline at end of file diff --git a/src/builders/application/index.js b/src/builders/application/index.js index a5afec75..2d8b1b21 100644 --- a/src/builders/application/index.js +++ b/src/builders/application/index.js @@ -26,6 +26,17 @@ async function* buildApplicationInternal(options, context, infrastructureSetting return; } const normalizedOptions = await (0, options_1.normalizeOptions)(context, projectName, options); + // Warn about prerender/ssr not yet supporting localize + if (normalizedOptions.i18nOptions.shouldInline && + (normalizedOptions.prerenderOptions || + normalizedOptions.ssrOptions || + normalizedOptions.appShellOptions)) { + context.logger.warn(`Prerendering, App Shell, and SSR are not yet supported with the 'localize' option and will be disabled for this build.`); + normalizedOptions.prerenderOptions = + normalizedOptions.ssrOptions = + normalizedOptions.appShellOptions = + undefined; + } yield* (0, build_action_1.runEsBuildBuildAction)((rebuildState) => (0, execute_build_1.executeBuild)(normalizedOptions, context, rebuildState), { watch: normalizedOptions.watch, poll: normalizedOptions.poll, @@ -46,4 +57,4 @@ function buildApplication(options, context) { } exports.buildApplication = buildApplication; exports.default = (0, architect_1.createBuilder)(buildApplication); -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hbmd1bGFyX2RldmtpdC9idWlsZF9hbmd1bGFyL3NyYy9idWlsZGVycy9hcHBsaWNhdGlvbi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUFFSCx5REFBeUY7QUFFekYseURBQStEO0FBQy9ELGlEQUFxRTtBQUNyRSxpREFBdUQ7QUFDdkQsbURBQStDO0FBQy9DLHVDQUFnRjtBQUd6RSxLQUFLLFNBQVMsQ0FBQyxDQUFDLHdCQUF3QixDQUM3QyxPQUEwQyxFQUMxQyxPQUF1QixFQUN2QixzQkFFQztJQU9ELHlCQUF5QjtJQUN6QixJQUFBLHdDQUE4QixFQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUV0RCw4QkFBOEI7SUFDOUIsTUFBTSxJQUFBLGtDQUFvQixFQUFDLE9BQU8sQ0FBQyxDQUFDO0lBRXBDLHFEQUFxRDtJQUNyRCxNQUFNLFdBQVcsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQztJQUM1QyxJQUFJLENBQUMsV0FBVyxFQUFFO1FBQ2hCLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLDhEQUE4RCxDQUFDLENBQUM7UUFFckYsT0FBTztLQUNSO0lBRUQsTUFBTSxpQkFBaUIsR0FBRyxNQUFNLElBQUEsMEJBQWdCLEVBQUMsT0FBTyxFQUFFLFdBQVcsRUFBRSxPQUFPLENBQUMsQ0FBQztJQUNoRixLQUFLLENBQUMsQ0FBQyxJQUFBLG9DQUFxQixFQUMxQixDQUFDLFlBQVksRUFBRSxFQUFFLENBQUMsSUFBQSw0QkFBWSxFQUFDLGlCQUFpQixFQUFFLE9BQU8sRUFBRSxZQUFZLENBQUMsRUFDeEU7UUFDRSxLQUFLLEVBQUUsaUJBQWlCLENBQUMsS0FBSztRQUM5QixJQUFJLEVBQUUsaUJBQWlCLENBQUMsSUFBSTtRQUM1QixnQkFBZ0IsRUFBRSxpQkFBaUIsQ0FBQyxnQkFBZ0I7UUFDcEQsWUFBWSxFQUFFLGlCQUFpQixDQUFDLFlBQVk7UUFDNUMsVUFBVSxFQUFFLGlCQUFpQixDQUFDLFVBQVU7UUFDeEMsT0FBTyxFQUFFLGlCQUFpQixDQUFDLE9BQU87UUFDbEMsV0FBVyxFQUFFLGlCQUFpQixDQUFDLFdBQVc7UUFDMUMsYUFBYSxFQUFFLGlCQUFpQixDQUFDLGFBQWE7UUFDOUMsUUFBUSxFQUFFLGlCQUFpQixDQUFDLFFBQVE7UUFDcEMsaUJBQWlCLEVBQUUsc0JBQXNCLEVBQUUsS0FBSztRQUNoRCxNQUFNLEVBQUUsT0FBTyxDQUFDLE1BQU07S0FDdkIsQ0FDRixDQUFDO0FBQ0osQ0FBQztBQTNDRCw0REEyQ0M7QUFFRCxTQUFnQixnQkFBZ0IsQ0FDOUIsT0FBa0MsRUFDbEMsT0FBdUI7SUFPdkIsT0FBTyx3QkFBd0IsQ0FBQyxPQUFPLEVBQUUsT0FBTyxDQUFDLENBQUM7QUFDcEQsQ0FBQztBQVZELDRDQVVDO0FBRUQsa0JBQWUsSUFBQSx5QkFBYSxFQUFDLGdCQUFnQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHsgQnVpbGRlckNvbnRleHQsIEJ1aWxkZXJPdXRwdXQsIGNyZWF0ZUJ1aWxkZXIgfSBmcm9tICdAYW5ndWxhci1kZXZraXQvYXJjaGl0ZWN0JztcbmltcG9ydCB0eXBlIHsgT3V0cHV0RmlsZSB9IGZyb20gJ2VzYnVpbGQnO1xuaW1wb3J0IHsgcHVyZ2VTdGFsZUJ1aWxkQ2FjaGUgfSBmcm9tICcuLi8uLi91dGlscy9wdXJnZS1jYWNoZSc7XG5pbXBvcnQgeyBhc3NlcnRDb21wYXRpYmxlQW5ndWxhclZlcnNpb24gfSBmcm9tICcuLi8uLi91dGlscy92ZXJzaW9uJztcbmltcG9ydCB7IHJ1bkVzQnVpbGRCdWlsZEFjdGlvbiB9IGZyb20gJy4vYnVpbGQtYWN0aW9uJztcbmltcG9ydCB7IGV4ZWN1dGVCdWlsZCB9IGZyb20gJy4vZXhlY3V0ZS1idWlsZCc7XG5pbXBvcnQgeyBBcHBsaWNhdGlvbkJ1aWxkZXJJbnRlcm5hbE9wdGlvbnMsIG5vcm1hbGl6ZU9wdGlvbnMgfSBmcm9tICcuL29wdGlvbnMnO1xuaW1wb3J0IHsgU2NoZW1hIGFzIEFwcGxpY2F0aW9uQnVpbGRlck9wdGlvbnMgfSBmcm9tICcuL3NjaGVtYSc7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiogYnVpbGRBcHBsaWNhdGlvbkludGVybmFsKFxuICBvcHRpb25zOiBBcHBsaWNhdGlvbkJ1aWxkZXJJbnRlcm5hbE9wdGlvbnMsXG4gIGNvbnRleHQ6IEJ1aWxkZXJDb250ZXh0LFxuICBpbmZyYXN0cnVjdHVyZVNldHRpbmdzPzoge1xuICAgIHdyaXRlPzogYm9vbGVhbjtcbiAgfSxcbik6IEFzeW5jSXRlcmFibGU8XG4gIEJ1aWxkZXJPdXRwdXQgJiB7XG4gICAgb3V0cHV0RmlsZXM/OiBPdXRwdXRGaWxlW107XG4gICAgYXNzZXRGaWxlcz86IHsgc291cmNlOiBzdHJpbmc7IGRlc3RpbmF0aW9uOiBzdHJpbmcgfVtdO1xuICB9XG4+IHtcbiAgLy8gQ2hlY2sgQW5ndWxhciB2ZXJzaW9uLlxuICBhc3NlcnRDb21wYXRpYmxlQW5ndWxhclZlcnNpb24oY29udGV4dC53b3Jrc3BhY2VSb290KTtcblxuICAvLyBQdXJnZSBvbGQgYnVpbGQgZGlzayBjYWNoZS5cbiAgYXdhaXQgcHVyZ2VTdGFsZUJ1aWxkQ2FjaGUoY29udGV4dCk7XG5cbiAgLy8gRGV0ZXJtaW5lIHByb2plY3QgbmFtZSBmcm9tIGJ1aWxkZXIgY29udGV4dCB0YXJnZXRcbiAgY29uc3QgcHJvamVjdE5hbWUgPSBjb250ZXh0LnRhcmdldD8ucHJvamVjdDtcbiAgaWYgKCFwcm9qZWN0TmFtZSkge1xuICAgIGNvbnRleHQubG9nZ2VyLmVycm9yKGBUaGUgJ2FwcGxpY2F0aW9uJyBidWlsZGVyIHJlcXVpcmVzIGEgdGFyZ2V0IHRvIGJlIHNwZWNpZmllZC5gKTtcblxuICAgIHJldHVybjtcbiAgfVxuXG4gIGNvbnN0IG5vcm1hbGl6ZWRPcHRpb25zID0gYXdhaXQgbm9ybWFsaXplT3B0aW9ucyhjb250ZXh0LCBwcm9qZWN0TmFtZSwgb3B0aW9ucyk7XG4gIHlpZWxkKiBydW5Fc0J1aWxkQnVpbGRBY3Rpb24oXG4gICAgKHJlYnVpbGRTdGF0ZSkgPT4gZXhlY3V0ZUJ1aWxkKG5vcm1hbGl6ZWRPcHRpb25zLCBjb250ZXh0LCByZWJ1aWxkU3RhdGUpLFxuICAgIHtcbiAgICAgIHdhdGNoOiBub3JtYWxpemVkT3B0aW9ucy53YXRjaCxcbiAgICAgIHBvbGw6IG5vcm1hbGl6ZWRPcHRpb25zLnBvbGwsXG4gICAgICBkZWxldGVPdXRwdXRQYXRoOiBub3JtYWxpemVkT3B0aW9ucy5kZWxldGVPdXRwdXRQYXRoLFxuICAgICAgY2FjaGVPcHRpb25zOiBub3JtYWxpemVkT3B0aW9ucy5jYWNoZU9wdGlvbnMsXG4gICAgICBvdXRwdXRQYXRoOiBub3JtYWxpemVkT3B0aW9ucy5vdXRwdXRQYXRoLFxuICAgICAgdmVyYm9zZTogbm9ybWFsaXplZE9wdGlvbnMudmVyYm9zZSxcbiAgICAgIHByb2plY3RSb290OiBub3JtYWxpemVkT3B0aW9ucy5wcm9qZWN0Um9vdCxcbiAgICAgIHdvcmtzcGFjZVJvb3Q6IG5vcm1hbGl6ZWRPcHRpb25zLndvcmtzcGFjZVJvb3QsXG4gICAgICBwcm9ncmVzczogbm9ybWFsaXplZE9wdGlvbnMucHJvZ3Jlc3MsXG4gICAgICB3cml0ZVRvRmlsZVN5c3RlbTogaW5mcmFzdHJ1Y3R1cmVTZXR0aW5ncz8ud3JpdGUsXG4gICAgICBsb2dnZXI6IGNvbnRleHQubG9nZ2VyLFxuICAgIH0sXG4gICk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBidWlsZEFwcGxpY2F0aW9uKFxuICBvcHRpb25zOiBBcHBsaWNhdGlvbkJ1aWxkZXJPcHRpb25zLFxuICBjb250ZXh0OiBCdWlsZGVyQ29udGV4dCxcbik6IEFzeW5jSXRlcmFibGU8XG4gIEJ1aWxkZXJPdXRwdXQgJiB7XG4gICAgb3V0cHV0RmlsZXM/OiBPdXRwdXRGaWxlW107XG4gICAgYXNzZXRGaWxlcz86IHsgc291cmNlOiBzdHJpbmc7IGRlc3RpbmF0aW9uOiBzdHJpbmcgfVtdO1xuICB9XG4+IHtcbiAgcmV0dXJuIGJ1aWxkQXBwbGljYXRpb25JbnRlcm5hbChvcHRpb25zLCBjb250ZXh0KTtcbn1cblxuZXhwb3J0IGRlZmF1bHQgY3JlYXRlQnVpbGRlcihidWlsZEFwcGxpY2F0aW9uKTtcbiJdfQ== \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hbmd1bGFyX2RldmtpdC9idWlsZF9hbmd1bGFyL3NyYy9idWlsZGVycy9hcHBsaWNhdGlvbi9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUFFSCx5REFBeUY7QUFFekYseURBQStEO0FBQy9ELGlEQUFxRTtBQUNyRSxpREFBdUQ7QUFDdkQsbURBQStDO0FBQy9DLHVDQUFnRjtBQUd6RSxLQUFLLFNBQVMsQ0FBQyxDQUFDLHdCQUF3QixDQUM3QyxPQUEwQyxFQUMxQyxPQUF1QixFQUN2QixzQkFFQztJQU9ELHlCQUF5QjtJQUN6QixJQUFBLHdDQUE4QixFQUFDLE9BQU8sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUV0RCw4QkFBOEI7SUFDOUIsTUFBTSxJQUFBLGtDQUFvQixFQUFDLE9BQU8sQ0FBQyxDQUFDO0lBRXBDLHFEQUFxRDtJQUNyRCxNQUFNLFdBQVcsR0FBRyxPQUFPLENBQUMsTUFBTSxFQUFFLE9BQU8sQ0FBQztJQUM1QyxJQUFJLENBQUMsV0FBVyxFQUFFO1FBQ2hCLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLDhEQUE4RCxDQUFDLENBQUM7UUFFckYsT0FBTztLQUNSO0lBRUQsTUFBTSxpQkFBaUIsR0FBRyxNQUFNLElBQUEsMEJBQWdCLEVBQUMsT0FBTyxFQUFFLFdBQVcsRUFBRSxPQUFPLENBQUMsQ0FBQztJQUVoRix1REFBdUQ7SUFDdkQsSUFDRSxpQkFBaUIsQ0FBQyxXQUFXLENBQUMsWUFBWTtRQUMxQyxDQUFDLGlCQUFpQixDQUFDLGdCQUFnQjtZQUNqQyxpQkFBaUIsQ0FBQyxVQUFVO1lBQzVCLGlCQUFpQixDQUFDLGVBQWUsQ0FBQyxFQUNwQztRQUNBLE9BQU8sQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUNqQix3SEFBd0gsQ0FDekgsQ0FBQztRQUNGLGlCQUFpQixDQUFDLGdCQUFnQjtZQUNoQyxpQkFBaUIsQ0FBQyxVQUFVO2dCQUM1QixpQkFBaUIsQ0FBQyxlQUFlO29CQUMvQixTQUFTLENBQUM7S0FDZjtJQUVELEtBQUssQ0FBQyxDQUFDLElBQUEsb0NBQXFCLEVBQzFCLENBQUMsWUFBWSxFQUFFLEVBQUUsQ0FBQyxJQUFBLDRCQUFZLEVBQUMsaUJBQWlCLEVBQUUsT0FBTyxFQUFFLFlBQVksQ0FBQyxFQUN4RTtRQUNFLEtBQUssRUFBRSxpQkFBaUIsQ0FBQyxLQUFLO1FBQzlCLElBQUksRUFBRSxpQkFBaUIsQ0FBQyxJQUFJO1FBQzVCLGdCQUFnQixFQUFFLGlCQUFpQixDQUFDLGdCQUFnQjtRQUNwRCxZQUFZLEVBQUUsaUJBQWlCLENBQUMsWUFBWTtRQUM1QyxVQUFVLEVBQUUsaUJBQWlCLENBQUMsVUFBVTtRQUN4QyxPQUFPLEVBQUUsaUJBQWlCLENBQUMsT0FBTztRQUNsQyxXQUFXLEVBQUUsaUJBQWlCLENBQUMsV0FBVztRQUMxQyxhQUFhLEVBQUUsaUJBQWlCLENBQUMsYUFBYTtRQUM5QyxRQUFRLEVBQUUsaUJBQWlCLENBQUMsUUFBUTtRQUNwQyxpQkFBaUIsRUFBRSxzQkFBc0IsRUFBRSxLQUFLO1FBQ2hELE1BQU0sRUFBRSxPQUFPLENBQUMsTUFBTTtLQUN2QixDQUNGLENBQUM7QUFDSixDQUFDO0FBNURELDREQTREQztBQUVELFNBQWdCLGdCQUFnQixDQUM5QixPQUFrQyxFQUNsQyxPQUF1QjtJQU92QixPQUFPLHdCQUF3QixDQUFDLE9BQU8sRUFBRSxPQUFPLENBQUMsQ0FBQztBQUNwRCxDQUFDO0FBVkQsNENBVUM7QUFFRCxrQkFBZSxJQUFBLHlCQUFhLEVBQUMsZ0JBQWdCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG5pbXBvcnQgeyBCdWlsZGVyQ29udGV4dCwgQnVpbGRlck91dHB1dCwgY3JlYXRlQnVpbGRlciB9IGZyb20gJ0Bhbmd1bGFyLWRldmtpdC9hcmNoaXRlY3QnO1xuaW1wb3J0IHR5cGUgeyBPdXRwdXRGaWxlIH0gZnJvbSAnZXNidWlsZCc7XG5pbXBvcnQgeyBwdXJnZVN0YWxlQnVpbGRDYWNoZSB9IGZyb20gJy4uLy4uL3V0aWxzL3B1cmdlLWNhY2hlJztcbmltcG9ydCB7IGFzc2VydENvbXBhdGlibGVBbmd1bGFyVmVyc2lvbiB9IGZyb20gJy4uLy4uL3V0aWxzL3ZlcnNpb24nO1xuaW1wb3J0IHsgcnVuRXNCdWlsZEJ1aWxkQWN0aW9uIH0gZnJvbSAnLi9idWlsZC1hY3Rpb24nO1xuaW1wb3J0IHsgZXhlY3V0ZUJ1aWxkIH0gZnJvbSAnLi9leGVjdXRlLWJ1aWxkJztcbmltcG9ydCB7IEFwcGxpY2F0aW9uQnVpbGRlckludGVybmFsT3B0aW9ucywgbm9ybWFsaXplT3B0aW9ucyB9IGZyb20gJy4vb3B0aW9ucyc7XG5pbXBvcnQgeyBTY2hlbWEgYXMgQXBwbGljYXRpb25CdWlsZGVyT3B0aW9ucyB9IGZyb20gJy4vc2NoZW1hJztcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uKiBidWlsZEFwcGxpY2F0aW9uSW50ZXJuYWwoXG4gIG9wdGlvbnM6IEFwcGxpY2F0aW9uQnVpbGRlckludGVybmFsT3B0aW9ucyxcbiAgY29udGV4dDogQnVpbGRlckNvbnRleHQsXG4gIGluZnJhc3RydWN0dXJlU2V0dGluZ3M/OiB7XG4gICAgd3JpdGU/OiBib29sZWFuO1xuICB9LFxuKTogQXN5bmNJdGVyYWJsZTxcbiAgQnVpbGRlck91dHB1dCAmIHtcbiAgICBvdXRwdXRGaWxlcz86IE91dHB1dEZpbGVbXTtcbiAgICBhc3NldEZpbGVzPzogeyBzb3VyY2U6IHN0cmluZzsgZGVzdGluYXRpb246IHN0cmluZyB9W107XG4gIH1cbj4ge1xuICAvLyBDaGVjayBBbmd1bGFyIHZlcnNpb24uXG4gIGFzc2VydENvbXBhdGlibGVBbmd1bGFyVmVyc2lvbihjb250ZXh0LndvcmtzcGFjZVJvb3QpO1xuXG4gIC8vIFB1cmdlIG9sZCBidWlsZCBkaXNrIGNhY2hlLlxuICBhd2FpdCBwdXJnZVN0YWxlQnVpbGRDYWNoZShjb250ZXh0KTtcblxuICAvLyBEZXRlcm1pbmUgcHJvamVjdCBuYW1lIGZyb20gYnVpbGRlciBjb250ZXh0IHRhcmdldFxuICBjb25zdCBwcm9qZWN0TmFtZSA9IGNvbnRleHQudGFyZ2V0Py5wcm9qZWN0O1xuICBpZiAoIXByb2plY3ROYW1lKSB7XG4gICAgY29udGV4dC5sb2dnZXIuZXJyb3IoYFRoZSAnYXBwbGljYXRpb24nIGJ1aWxkZXIgcmVxdWlyZXMgYSB0YXJnZXQgdG8gYmUgc3BlY2lmaWVkLmApO1xuXG4gICAgcmV0dXJuO1xuICB9XG5cbiAgY29uc3Qgbm9ybWFsaXplZE9wdGlvbnMgPSBhd2FpdCBub3JtYWxpemVPcHRpb25zKGNvbnRleHQsIHByb2plY3ROYW1lLCBvcHRpb25zKTtcblxuICAvLyBXYXJuIGFib3V0IHByZXJlbmRlci9zc3Igbm90IHlldCBzdXBwb3J0aW5nIGxvY2FsaXplXG4gIGlmIChcbiAgICBub3JtYWxpemVkT3B0aW9ucy5pMThuT3B0aW9ucy5zaG91bGRJbmxpbmUgJiZcbiAgICAobm9ybWFsaXplZE9wdGlvbnMucHJlcmVuZGVyT3B0aW9ucyB8fFxuICAgICAgbm9ybWFsaXplZE9wdGlvbnMuc3NyT3B0aW9ucyB8fFxuICAgICAgbm9ybWFsaXplZE9wdGlvbnMuYXBwU2hlbGxPcHRpb25zKVxuICApIHtcbiAgICBjb250ZXh0LmxvZ2dlci53YXJuKFxuICAgICAgYFByZXJlbmRlcmluZywgQXBwIFNoZWxsLCBhbmQgU1NSIGFyZSBub3QgeWV0IHN1cHBvcnRlZCB3aXRoIHRoZSAnbG9jYWxpemUnIG9wdGlvbiBhbmQgd2lsbCBiZSBkaXNhYmxlZCBmb3IgdGhpcyBidWlsZC5gLFxuICAgICk7XG4gICAgbm9ybWFsaXplZE9wdGlvbnMucHJlcmVuZGVyT3B0aW9ucyA9XG4gICAgICBub3JtYWxpemVkT3B0aW9ucy5zc3JPcHRpb25zID1cbiAgICAgIG5vcm1hbGl6ZWRPcHRpb25zLmFwcFNoZWxsT3B0aW9ucyA9XG4gICAgICAgIHVuZGVmaW5lZDtcbiAgfVxuXG4gIHlpZWxkKiBydW5Fc0J1aWxkQnVpbGRBY3Rpb24oXG4gICAgKHJlYnVpbGRTdGF0ZSkgPT4gZXhlY3V0ZUJ1aWxkKG5vcm1hbGl6ZWRPcHRpb25zLCBjb250ZXh0LCByZWJ1aWxkU3RhdGUpLFxuICAgIHtcbiAgICAgIHdhdGNoOiBub3JtYWxpemVkT3B0aW9ucy53YXRjaCxcbiAgICAgIHBvbGw6IG5vcm1hbGl6ZWRPcHRpb25zLnBvbGwsXG4gICAgICBkZWxldGVPdXRwdXRQYXRoOiBub3JtYWxpemVkT3B0aW9ucy5kZWxldGVPdXRwdXRQYXRoLFxuICAgICAgY2FjaGVPcHRpb25zOiBub3JtYWxpemVkT3B0aW9ucy5jYWNoZU9wdGlvbnMsXG4gICAgICBvdXRwdXRQYXRoOiBub3JtYWxpemVkT3B0aW9ucy5vdXRwdXRQYXRoLFxuICAgICAgdmVyYm9zZTogbm9ybWFsaXplZE9wdGlvbnMudmVyYm9zZSxcbiAgICAgIHByb2plY3RSb290OiBub3JtYWxpemVkT3B0aW9ucy5wcm9qZWN0Um9vdCxcbiAgICAgIHdvcmtzcGFjZVJvb3Q6IG5vcm1hbGl6ZWRPcHRpb25zLndvcmtzcGFjZVJvb3QsXG4gICAgICBwcm9ncmVzczogbm9ybWFsaXplZE9wdGlvbnMucHJvZ3Jlc3MsXG4gICAgICB3cml0ZVRvRmlsZVN5c3RlbTogaW5mcmFzdHJ1Y3R1cmVTZXR0aW5ncz8ud3JpdGUsXG4gICAgICBsb2dnZXI6IGNvbnRleHQubG9nZ2VyLFxuICAgIH0sXG4gICk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBidWlsZEFwcGxpY2F0aW9uKFxuICBvcHRpb25zOiBBcHBsaWNhdGlvbkJ1aWxkZXJPcHRpb25zLFxuICBjb250ZXh0OiBCdWlsZGVyQ29udGV4dCxcbik6IEFzeW5jSXRlcmFibGU8XG4gIEJ1aWxkZXJPdXRwdXQgJiB7XG4gICAgb3V0cHV0RmlsZXM/OiBPdXRwdXRGaWxlW107XG4gICAgYXNzZXRGaWxlcz86IHsgc291cmNlOiBzdHJpbmc7IGRlc3RpbmF0aW9uOiBzdHJpbmcgfVtdO1xuICB9XG4+IHtcbiAgcmV0dXJuIGJ1aWxkQXBwbGljYXRpb25JbnRlcm5hbChvcHRpb25zLCBjb250ZXh0KTtcbn1cblxuZXhwb3J0IGRlZmF1bHQgY3JlYXRlQnVpbGRlcihidWlsZEFwcGxpY2F0aW9uKTtcbiJdfQ== \ No newline at end of file diff --git a/src/builders/application/options.d.ts b/src/builders/application/options.d.ts index ec19884f..f2631ed1 100644 --- a/src/builders/application/options.d.ts +++ b/src/builders/application/options.d.ts @@ -25,6 +25,11 @@ interface InternalOptions { * Currently used by the dev-server to support prebundling. */ externalPackages?: boolean; + /** + * Forces the output from the localize post-processing to not create nested directories per locale output. + * This is only used by the development server which currently only supports a single locale per build. + */ + forceI18nFlatOutput?: boolean; } /** Full set of options for `application` builder. */ export type ApplicationBuilderInternalOptions = Omit & { diff --git a/src/builders/application/options.js b/src/builders/application/options.js index 60b12a5e..cda3278f 100644 --- a/src/builders/application/options.js +++ b/src/builders/application/options.js @@ -43,6 +43,9 @@ async function normalizeOptions(context, projectName, options) { const i18nOptions = (0, i18n_options_1.createI18nOptions)(projectMetadata, options.localize); i18nOptions.duplicateTranslationBehavior = options.i18nDuplicateTranslation; i18nOptions.missingTranslationBehavior = options.i18nMissingTranslation; + if (options.forceI18nFlatOutput) { + i18nOptions.flatOutput = true; + } const entryPoints = normalizeEntryPoints(workspaceRoot, options.browser, options.entryPoints); const tsconfig = node_path_1.default.join(workspaceRoot, options.tsConfig); const outputPath = normalizeDirectoryPath(node_path_1.default.join(workspaceRoot, options.outputPath)); @@ -260,4 +263,4 @@ function normalizeDirectoryPath(path) { } return path; } -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"options.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/application/options.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 { createRequire } from 'node:module';\nimport path from 'node:path';\nimport {\n  globalScriptsByBundleName,\n  normalizeGlobalStyles,\n} from '../../tools/webpack/utils/helpers';\nimport { normalizeAssetPatterns, normalizeOptimization, normalizeSourceMaps } from '../../utils';\nimport { I18nOptions, createI18nOptions } from '../../utils/i18n-options';\nimport { normalizeCacheOptions } from '../../utils/normalize-cache';\nimport { generateEntryPoints } from '../../utils/package-chunk-sort';\nimport { findTailwindConfigurationFile } from '../../utils/tailwind';\nimport { getIndexInputFile, getIndexOutputFile } from '../../utils/webpack-browser-config';\nimport { Schema as ApplicationBuilderOptions, I18NTranslation, OutputHashing } from './schema';\n\nexport type NormalizedApplicationBuildOptions = Awaited<ReturnType<typeof normalizeOptions>>;\n\n/** Internal options hidden from builder schema but available when invoked programmatically. */\ninterface InternalOptions {\n  /**\n   * Entry points to use for the compilation. Incompatible with `browser`, which must not be provided. May be relative or absolute paths.\n   * If given a relative path, it is resolved relative to the current workspace and will generate an output at the same relative location\n   * in the output directory. If given an absolute path, the output will be generated in the root of the output directory with the same base\n   * name.\n   */\n  entryPoints?: Set<string>;\n\n  /** File extension to use for the generated output files. */\n  outExtension?: 'js' | 'mjs';\n\n  /**\n   * Indicates whether all node packages should be marked as external.\n   * Currently used by the dev-server to support prebundling.\n   */\n  externalPackages?: boolean;\n}\n\n/** Full set of options for `application` builder. */\nexport type ApplicationBuilderInternalOptions = Omit<\n  ApplicationBuilderOptions & InternalOptions,\n  'browser'\n> & {\n  // `browser` can be `undefined` if `entryPoints` is used.\n  browser?: string;\n};\n\n/**\n * Normalize the user provided options by creating full paths for all path based options\n * and converting multi-form options into a single form that can be directly used\n * by the build process.\n *\n * @param context The context for current builder execution.\n * @param projectName The name of the project for the current execution.\n * @param options An object containing the options to use for the build.\n * @returns An object containing normalized options required to perform the build.\n */\n// eslint-disable-next-line max-lines-per-function\nexport async function normalizeOptions(\n  context: BuilderContext,\n  projectName: string,\n  options: ApplicationBuilderInternalOptions,\n) {\n  const workspaceRoot = context.workspaceRoot;\n  const projectMetadata = await context.getProjectMetadata(projectName);\n  const projectRoot = normalizeDirectoryPath(\n    path.join(workspaceRoot, (projectMetadata.root as string | undefined) ?? ''),\n  );\n  const projectSourceRoot = normalizeDirectoryPath(\n    path.join(workspaceRoot, (projectMetadata.sourceRoot as string | undefined) ?? 'src'),\n  );\n\n  // Gather persistent caching option and provide a project specific cache location\n  const cacheOptions = normalizeCacheOptions(projectMetadata, workspaceRoot);\n  cacheOptions.path = path.join(cacheOptions.path, projectName);\n\n  const i18nOptions: I18nOptions & {\n    duplicateTranslationBehavior?: I18NTranslation;\n    missingTranslationBehavior?: I18NTranslation;\n  } = createI18nOptions(projectMetadata, options.localize);\n  i18nOptions.duplicateTranslationBehavior = options.i18nDuplicateTranslation;\n  i18nOptions.missingTranslationBehavior = options.i18nMissingTranslation;\n\n  const entryPoints = normalizeEntryPoints(workspaceRoot, options.browser, options.entryPoints);\n  const tsconfig = path.join(workspaceRoot, options.tsConfig);\n  const outputPath = normalizeDirectoryPath(path.join(workspaceRoot, options.outputPath));\n  const optimizationOptions = normalizeOptimization(options.optimization);\n  const sourcemapOptions = normalizeSourceMaps(options.sourceMap ?? false);\n  const assets = options.assets?.length\n    ? normalizeAssetPatterns(options.assets, workspaceRoot, projectRoot, projectSourceRoot)\n    : undefined;\n\n  const outputNames = {\n    bundles:\n      options.outputHashing === OutputHashing.All || options.outputHashing === OutputHashing.Bundles\n        ? '[name]-[hash]'\n        : '[name]',\n    media:\n      'media/' +\n      (options.outputHashing === OutputHashing.All || options.outputHashing === OutputHashing.Media\n        ? '[name]-[hash]'\n        : '[name]'),\n  };\n\n  let fileReplacements: Record<string, string> | undefined;\n  if (options.fileReplacements) {\n    for (const replacement of options.fileReplacements) {\n      fileReplacements ??= {};\n      fileReplacements[path.join(workspaceRoot, replacement.replace)] = path.join(\n        workspaceRoot,\n        replacement.with,\n      );\n    }\n  }\n\n  const globalStyles: { name: string; files: string[]; initial: boolean }[] = [];\n  if (options.styles?.length) {\n    const { entryPoints: stylesheetEntrypoints, noInjectNames } = normalizeGlobalStyles(\n      options.styles || [],\n    );\n    for (const [name, files] of Object.entries(stylesheetEntrypoints)) {\n      globalStyles.push({ name, files, initial: !noInjectNames.includes(name) });\n    }\n  }\n\n  const globalScripts: { name: string; files: string[]; initial: boolean }[] = [];\n  if (options.scripts?.length) {\n    for (const { bundleName, paths, inject } of globalScriptsByBundleName(options.scripts)) {\n      globalScripts.push({ name: bundleName, files: paths, initial: inject });\n    }\n  }\n\n  let tailwindConfiguration: { file: string; package: string } | undefined;\n  const tailwindConfigurationPath = await findTailwindConfigurationFile(workspaceRoot, projectRoot);\n  if (tailwindConfigurationPath) {\n    // Create a node resolver at the project root as a directory\n    const resolver = createRequire(projectRoot + '/');\n    try {\n      tailwindConfiguration = {\n        file: tailwindConfigurationPath,\n        package: resolver.resolve('tailwindcss'),\n      };\n    } catch {\n      const relativeTailwindConfigPath = path.relative(workspaceRoot, tailwindConfigurationPath);\n      context.logger.warn(\n        `Tailwind CSS configuration file found (${relativeTailwindConfigPath})` +\n          ` but the 'tailwindcss' package is not installed.` +\n          ` To enable Tailwind CSS, please install the 'tailwindcss' package.`,\n      );\n    }\n  }\n\n  let indexHtmlOptions;\n  // index can never have a value of `true` but in the schema it's of type `boolean`.\n  if (typeof options.index !== 'boolean') {\n    indexHtmlOptions = {\n      input: path.join(workspaceRoot, getIndexInputFile(options.index)),\n      // The output file will be created within the configured output path\n      output: getIndexOutputFile(options.index),\n      // TODO: Use existing information from above to create the insertion order\n      insertionOrder: generateEntryPoints({\n        scripts: options.scripts ?? [],\n        styles: options.styles ?? [],\n      }),\n    };\n  }\n\n  let serverEntryPoint: string | undefined;\n  if (options.server) {\n    serverEntryPoint = path.join(workspaceRoot, options.server);\n  } else if (options.server === '') {\n    throw new Error('`server` option cannot be an empty string.');\n  }\n\n  let prerenderOptions;\n  if (options.prerender) {\n    const { discoverRoutes = true, routesFile = undefined } =\n      options.prerender === true ? {} : options.prerender;\n\n    prerenderOptions = {\n      discoverRoutes,\n      routesFile: routesFile && path.join(workspaceRoot, routesFile),\n    };\n  }\n\n  let ssrOptions;\n  if (options.ssr === true) {\n    ssrOptions = {};\n  } else if (typeof options.ssr === 'string') {\n    ssrOptions = {\n      entry: path.join(workspaceRoot, options.ssr),\n    };\n  }\n\n  let appShellOptions;\n  if (options.appShell) {\n    appShellOptions = {\n      route: 'shell',\n    };\n  }\n\n  // Initial options to keep\n  const {\n    allowedCommonJsDependencies,\n    aot,\n    baseHref,\n    crossOrigin,\n    externalDependencies,\n    extractLicenses,\n    inlineStyleLanguage = 'css',\n    outExtension,\n    serviceWorker,\n    poll,\n    polyfills,\n    preserveSymlinks,\n    statsJson,\n    stylePreprocessorOptions,\n    subresourceIntegrity,\n    verbose,\n    watch,\n    progress = true,\n    externalPackages,\n    deleteOutputPath,\n  } = options;\n\n  // Return all the normalized options\n  return {\n    advancedOptimizations: !!aot,\n    allowedCommonJsDependencies,\n    baseHref,\n    cacheOptions,\n    crossOrigin,\n    deleteOutputPath,\n    externalDependencies,\n    extractLicenses,\n    inlineStyleLanguage,\n    jit: !aot,\n    stats: !!statsJson,\n    polyfills: polyfills === undefined || Array.isArray(polyfills) ? polyfills : [polyfills],\n    poll,\n    progress,\n    externalPackages,\n    // If not explicitly set, default to the Node.js process argument\n    preserveSymlinks: preserveSymlinks ?? process.execArgv.includes('--preserve-symlinks'),\n    stylePreprocessorOptions,\n    subresourceIntegrity,\n    serverEntryPoint,\n    prerenderOptions,\n    appShellOptions,\n    ssrOptions,\n    verbose,\n    watch,\n    workspaceRoot,\n    entryPoints,\n    optimizationOptions,\n    outputPath,\n    outExtension,\n    sourcemapOptions,\n    tsconfig,\n    projectRoot,\n    assets,\n    outputNames,\n    fileReplacements,\n    globalStyles,\n    globalScripts,\n    serviceWorker:\n      typeof serviceWorker === 'string' ? path.join(workspaceRoot, serviceWorker) : undefined,\n    indexHtmlOptions,\n    tailwindConfiguration,\n    i18nOptions,\n  };\n}\n\n/**\n * Normalize entry point options. To maintain compatibility with the legacy browser builder, we need a single `browser`\n * option which defines a single entry point. However, we also want to support multiple entry points as an internal option.\n * The two options are mutually exclusive and if `browser` is provided it will be used as the sole entry point.\n * If `entryPoints` are provided, they will be used as the set of entry points.\n *\n * @param workspaceRoot Path to the root of the Angular workspace.\n * @param browser The `browser` option pointing at the application entry point. While required per the schema file, it may be omitted by\n *     programmatic usages of `browser-esbuild`.\n * @param entryPoints Set of entry points to use if provided.\n * @returns An object mapping entry point names to their file paths.\n */\nfunction normalizeEntryPoints(\n  workspaceRoot: string,\n  browser: string | undefined,\n  entryPoints: Set<string> = new Set(),\n): Record<string, string> {\n  if (browser === '') {\n    throw new Error('`browser` option cannot be an empty string.');\n  }\n\n  // `browser` and `entryPoints` are mutually exclusive.\n  if (browser && entryPoints.size > 0) {\n    throw new Error('Only one of `browser` or `entryPoints` may be provided.');\n  }\n  if (!browser && entryPoints.size === 0) {\n    // Schema should normally reject this case, but programmatic usages of the builder might make this mistake.\n    throw new Error('Either `browser` or at least one `entryPoints` value must be provided.');\n  }\n\n  // Schema types force `browser` to always be provided, but it may be omitted when the builder is invoked programmatically.\n  if (browser) {\n    // Use `browser` alone.\n    return { 'main': path.join(workspaceRoot, browser) };\n  } else {\n    // Use `entryPoints` alone.\n    const entryPointPaths: Record<string, string> = {};\n    for (const entryPoint of entryPoints) {\n      const parsedEntryPoint = path.parse(entryPoint);\n\n      // Use the input file path without an extension as the \"name\" of the entry point dictating its output location.\n      // Relative entry points are generated at the same relative path in the output directory.\n      // Absolute entry points are always generated with the same file name in the root of the output directory. This includes absolute\n      // paths pointing at files actually within the workspace root.\n      const entryPointName = path.isAbsolute(entryPoint)\n        ? parsedEntryPoint.name\n        : path.join(parsedEntryPoint.dir, parsedEntryPoint.name);\n\n      // Get the full file path to the entry point input.\n      const entryPointPath = path.isAbsolute(entryPoint)\n        ? entryPoint\n        : path.join(workspaceRoot, entryPoint);\n\n      // Check for conflicts with previous entry points.\n      const existingEntryPointPath = entryPointPaths[entryPointName];\n      if (existingEntryPointPath) {\n        throw new Error(\n          `\\`${existingEntryPointPath}\\` and \\`${entryPointPath}\\` both output to the same location \\`${entryPointName}\\`.` +\n            ' Rename or move one of the files to fix the conflict.',\n        );\n      }\n\n      entryPointPaths[entryPointName] = entryPointPath;\n    }\n\n    return entryPointPaths;\n  }\n}\n\n/**\n * Normalize a directory path string.\n * Currently only removes a trailing slash if present.\n * @param path A path string.\n * @returns A normalized path string.\n */\nfunction normalizeDirectoryPath(path: string): string {\n  const last = path[path.length - 1];\n  if (last === '/' || last === '\\\\') {\n    return path.slice(0, -1);\n  }\n\n  return path;\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"options.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/application/options.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 { createRequire } from 'node:module';\nimport path from 'node:path';\nimport {\n  globalScriptsByBundleName,\n  normalizeGlobalStyles,\n} from '../../tools/webpack/utils/helpers';\nimport { normalizeAssetPatterns, normalizeOptimization, normalizeSourceMaps } from '../../utils';\nimport { I18nOptions, createI18nOptions } from '../../utils/i18n-options';\nimport { normalizeCacheOptions } from '../../utils/normalize-cache';\nimport { generateEntryPoints } from '../../utils/package-chunk-sort';\nimport { findTailwindConfigurationFile } from '../../utils/tailwind';\nimport { getIndexInputFile, getIndexOutputFile } from '../../utils/webpack-browser-config';\nimport { Schema as ApplicationBuilderOptions, I18NTranslation, OutputHashing } from './schema';\n\nexport type NormalizedApplicationBuildOptions = Awaited<ReturnType<typeof normalizeOptions>>;\n\n/** Internal options hidden from builder schema but available when invoked programmatically. */\ninterface InternalOptions {\n  /**\n   * Entry points to use for the compilation. Incompatible with `browser`, which must not be provided. May be relative or absolute paths.\n   * If given a relative path, it is resolved relative to the current workspace and will generate an output at the same relative location\n   * in the output directory. If given an absolute path, the output will be generated in the root of the output directory with the same base\n   * name.\n   */\n  entryPoints?: Set<string>;\n\n  /** File extension to use for the generated output files. */\n  outExtension?: 'js' | 'mjs';\n\n  /**\n   * Indicates whether all node packages should be marked as external.\n   * Currently used by the dev-server to support prebundling.\n   */\n  externalPackages?: boolean;\n\n  /**\n   * Forces the output from the localize post-processing to not create nested directories per locale output.\n   * This is only used by the development server which currently only supports a single locale per build.\n   */\n  forceI18nFlatOutput?: boolean;\n}\n\n/** Full set of options for `application` builder. */\nexport type ApplicationBuilderInternalOptions = Omit<\n  ApplicationBuilderOptions & InternalOptions,\n  'browser'\n> & {\n  // `browser` can be `undefined` if `entryPoints` is used.\n  browser?: string;\n};\n\n/**\n * Normalize the user provided options by creating full paths for all path based options\n * and converting multi-form options into a single form that can be directly used\n * by the build process.\n *\n * @param context The context for current builder execution.\n * @param projectName The name of the project for the current execution.\n * @param options An object containing the options to use for the build.\n * @returns An object containing normalized options required to perform the build.\n */\n// eslint-disable-next-line max-lines-per-function\nexport async function normalizeOptions(\n  context: BuilderContext,\n  projectName: string,\n  options: ApplicationBuilderInternalOptions,\n) {\n  const workspaceRoot = context.workspaceRoot;\n  const projectMetadata = await context.getProjectMetadata(projectName);\n  const projectRoot = normalizeDirectoryPath(\n    path.join(workspaceRoot, (projectMetadata.root as string | undefined) ?? ''),\n  );\n  const projectSourceRoot = normalizeDirectoryPath(\n    path.join(workspaceRoot, (projectMetadata.sourceRoot as string | undefined) ?? 'src'),\n  );\n\n  // Gather persistent caching option and provide a project specific cache location\n  const cacheOptions = normalizeCacheOptions(projectMetadata, workspaceRoot);\n  cacheOptions.path = path.join(cacheOptions.path, projectName);\n\n  const i18nOptions: I18nOptions & {\n    duplicateTranslationBehavior?: I18NTranslation;\n    missingTranslationBehavior?: I18NTranslation;\n  } = createI18nOptions(projectMetadata, options.localize);\n  i18nOptions.duplicateTranslationBehavior = options.i18nDuplicateTranslation;\n  i18nOptions.missingTranslationBehavior = options.i18nMissingTranslation;\n  if (options.forceI18nFlatOutput) {\n    i18nOptions.flatOutput = true;\n  }\n\n  const entryPoints = normalizeEntryPoints(workspaceRoot, options.browser, options.entryPoints);\n  const tsconfig = path.join(workspaceRoot, options.tsConfig);\n  const outputPath = normalizeDirectoryPath(path.join(workspaceRoot, options.outputPath));\n  const optimizationOptions = normalizeOptimization(options.optimization);\n  const sourcemapOptions = normalizeSourceMaps(options.sourceMap ?? false);\n  const assets = options.assets?.length\n    ? normalizeAssetPatterns(options.assets, workspaceRoot, projectRoot, projectSourceRoot)\n    : undefined;\n\n  const outputNames = {\n    bundles:\n      options.outputHashing === OutputHashing.All || options.outputHashing === OutputHashing.Bundles\n        ? '[name]-[hash]'\n        : '[name]',\n    media:\n      'media/' +\n      (options.outputHashing === OutputHashing.All || options.outputHashing === OutputHashing.Media\n        ? '[name]-[hash]'\n        : '[name]'),\n  };\n\n  let fileReplacements: Record<string, string> | undefined;\n  if (options.fileReplacements) {\n    for (const replacement of options.fileReplacements) {\n      fileReplacements ??= {};\n      fileReplacements[path.join(workspaceRoot, replacement.replace)] = path.join(\n        workspaceRoot,\n        replacement.with,\n      );\n    }\n  }\n\n  const globalStyles: { name: string; files: string[]; initial: boolean }[] = [];\n  if (options.styles?.length) {\n    const { entryPoints: stylesheetEntrypoints, noInjectNames } = normalizeGlobalStyles(\n      options.styles || [],\n    );\n    for (const [name, files] of Object.entries(stylesheetEntrypoints)) {\n      globalStyles.push({ name, files, initial: !noInjectNames.includes(name) });\n    }\n  }\n\n  const globalScripts: { name: string; files: string[]; initial: boolean }[] = [];\n  if (options.scripts?.length) {\n    for (const { bundleName, paths, inject } of globalScriptsByBundleName(options.scripts)) {\n      globalScripts.push({ name: bundleName, files: paths, initial: inject });\n    }\n  }\n\n  let tailwindConfiguration: { file: string; package: string } | undefined;\n  const tailwindConfigurationPath = await findTailwindConfigurationFile(workspaceRoot, projectRoot);\n  if (tailwindConfigurationPath) {\n    // Create a node resolver at the project root as a directory\n    const resolver = createRequire(projectRoot + '/');\n    try {\n      tailwindConfiguration = {\n        file: tailwindConfigurationPath,\n        package: resolver.resolve('tailwindcss'),\n      };\n    } catch {\n      const relativeTailwindConfigPath = path.relative(workspaceRoot, tailwindConfigurationPath);\n      context.logger.warn(\n        `Tailwind CSS configuration file found (${relativeTailwindConfigPath})` +\n          ` but the 'tailwindcss' package is not installed.` +\n          ` To enable Tailwind CSS, please install the 'tailwindcss' package.`,\n      );\n    }\n  }\n\n  let indexHtmlOptions;\n  // index can never have a value of `true` but in the schema it's of type `boolean`.\n  if (typeof options.index !== 'boolean') {\n    indexHtmlOptions = {\n      input: path.join(workspaceRoot, getIndexInputFile(options.index)),\n      // The output file will be created within the configured output path\n      output: getIndexOutputFile(options.index),\n      // TODO: Use existing information from above to create the insertion order\n      insertionOrder: generateEntryPoints({\n        scripts: options.scripts ?? [],\n        styles: options.styles ?? [],\n      }),\n    };\n  }\n\n  let serverEntryPoint: string | undefined;\n  if (options.server) {\n    serverEntryPoint = path.join(workspaceRoot, options.server);\n  } else if (options.server === '') {\n    throw new Error('`server` option cannot be an empty string.');\n  }\n\n  let prerenderOptions;\n  if (options.prerender) {\n    const { discoverRoutes = true, routesFile = undefined } =\n      options.prerender === true ? {} : options.prerender;\n\n    prerenderOptions = {\n      discoverRoutes,\n      routesFile: routesFile && path.join(workspaceRoot, routesFile),\n    };\n  }\n\n  let ssrOptions;\n  if (options.ssr === true) {\n    ssrOptions = {};\n  } else if (typeof options.ssr === 'string') {\n    ssrOptions = {\n      entry: path.join(workspaceRoot, options.ssr),\n    };\n  }\n\n  let appShellOptions;\n  if (options.appShell) {\n    appShellOptions = {\n      route: 'shell',\n    };\n  }\n\n  // Initial options to keep\n  const {\n    allowedCommonJsDependencies,\n    aot,\n    baseHref,\n    crossOrigin,\n    externalDependencies,\n    extractLicenses,\n    inlineStyleLanguage = 'css',\n    outExtension,\n    serviceWorker,\n    poll,\n    polyfills,\n    preserveSymlinks,\n    statsJson,\n    stylePreprocessorOptions,\n    subresourceIntegrity,\n    verbose,\n    watch,\n    progress = true,\n    externalPackages,\n    deleteOutputPath,\n  } = options;\n\n  // Return all the normalized options\n  return {\n    advancedOptimizations: !!aot,\n    allowedCommonJsDependencies,\n    baseHref,\n    cacheOptions,\n    crossOrigin,\n    deleteOutputPath,\n    externalDependencies,\n    extractLicenses,\n    inlineStyleLanguage,\n    jit: !aot,\n    stats: !!statsJson,\n    polyfills: polyfills === undefined || Array.isArray(polyfills) ? polyfills : [polyfills],\n    poll,\n    progress,\n    externalPackages,\n    // If not explicitly set, default to the Node.js process argument\n    preserveSymlinks: preserveSymlinks ?? process.execArgv.includes('--preserve-symlinks'),\n    stylePreprocessorOptions,\n    subresourceIntegrity,\n    serverEntryPoint,\n    prerenderOptions,\n    appShellOptions,\n    ssrOptions,\n    verbose,\n    watch,\n    workspaceRoot,\n    entryPoints,\n    optimizationOptions,\n    outputPath,\n    outExtension,\n    sourcemapOptions,\n    tsconfig,\n    projectRoot,\n    assets,\n    outputNames,\n    fileReplacements,\n    globalStyles,\n    globalScripts,\n    serviceWorker:\n      typeof serviceWorker === 'string' ? path.join(workspaceRoot, serviceWorker) : undefined,\n    indexHtmlOptions,\n    tailwindConfiguration,\n    i18nOptions,\n  };\n}\n\n/**\n * Normalize entry point options. To maintain compatibility with the legacy browser builder, we need a single `browser`\n * option which defines a single entry point. However, we also want to support multiple entry points as an internal option.\n * The two options are mutually exclusive and if `browser` is provided it will be used as the sole entry point.\n * If `entryPoints` are provided, they will be used as the set of entry points.\n *\n * @param workspaceRoot Path to the root of the Angular workspace.\n * @param browser The `browser` option pointing at the application entry point. While required per the schema file, it may be omitted by\n *     programmatic usages of `browser-esbuild`.\n * @param entryPoints Set of entry points to use if provided.\n * @returns An object mapping entry point names to their file paths.\n */\nfunction normalizeEntryPoints(\n  workspaceRoot: string,\n  browser: string | undefined,\n  entryPoints: Set<string> = new Set(),\n): Record<string, string> {\n  if (browser === '') {\n    throw new Error('`browser` option cannot be an empty string.');\n  }\n\n  // `browser` and `entryPoints` are mutually exclusive.\n  if (browser && entryPoints.size > 0) {\n    throw new Error('Only one of `browser` or `entryPoints` may be provided.');\n  }\n  if (!browser && entryPoints.size === 0) {\n    // Schema should normally reject this case, but programmatic usages of the builder might make this mistake.\n    throw new Error('Either `browser` or at least one `entryPoints` value must be provided.');\n  }\n\n  // Schema types force `browser` to always be provided, but it may be omitted when the builder is invoked programmatically.\n  if (browser) {\n    // Use `browser` alone.\n    return { 'main': path.join(workspaceRoot, browser) };\n  } else {\n    // Use `entryPoints` alone.\n    const entryPointPaths: Record<string, string> = {};\n    for (const entryPoint of entryPoints) {\n      const parsedEntryPoint = path.parse(entryPoint);\n\n      // Use the input file path without an extension as the \"name\" of the entry point dictating its output location.\n      // Relative entry points are generated at the same relative path in the output directory.\n      // Absolute entry points are always generated with the same file name in the root of the output directory. This includes absolute\n      // paths pointing at files actually within the workspace root.\n      const entryPointName = path.isAbsolute(entryPoint)\n        ? parsedEntryPoint.name\n        : path.join(parsedEntryPoint.dir, parsedEntryPoint.name);\n\n      // Get the full file path to the entry point input.\n      const entryPointPath = path.isAbsolute(entryPoint)\n        ? entryPoint\n        : path.join(workspaceRoot, entryPoint);\n\n      // Check for conflicts with previous entry points.\n      const existingEntryPointPath = entryPointPaths[entryPointName];\n      if (existingEntryPointPath) {\n        throw new Error(\n          `\\`${existingEntryPointPath}\\` and \\`${entryPointPath}\\` both output to the same location \\`${entryPointName}\\`.` +\n            ' Rename or move one of the files to fix the conflict.',\n        );\n      }\n\n      entryPointPaths[entryPointName] = entryPointPath;\n    }\n\n    return entryPointPaths;\n  }\n}\n\n/**\n * Normalize a directory path string.\n * Currently only removes a trailing slash if present.\n * @param path A path string.\n * @returns A normalized path string.\n */\nfunction normalizeDirectoryPath(path: string): string {\n  const last = path[path.length - 1];\n  if (last === '/' || last === '\\\\') {\n    return path.slice(0, -1);\n  }\n\n  return path;\n}\n"]} \ No newline at end of file diff --git a/src/builders/browser-esbuild/builder-status-warnings.js b/src/builders/browser-esbuild/builder-status-warnings.js index de61d091..535471ae 100644 --- a/src/builders/browser-esbuild/builder-status-warnings.js +++ b/src/builders/browser-esbuild/builder-status-warnings.js @@ -10,11 +10,6 @@ Object.defineProperty(exports, "__esModule", { value: true }); exports.logBuilderStatusWarnings = void 0; const UNSUPPORTED_OPTIONS = [ 'budgets', - // * i18n support - 'localize', - // The following two have no effect when localize is not enabled - // 'i18nDuplicateTranslation', - // 'i18nMissingTranslation', // * Deprecated 'deployUrl', // * Always enabled with esbuild @@ -50,4 +45,4 @@ function logBuilderStatusWarnings(options, context) { } } exports.logBuilderStatusWarnings = logBuilderStatusWarnings; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGRlci1zdGF0dXMtd2FybmluZ3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hbmd1bGFyX2RldmtpdC9idWlsZF9hbmd1bGFyL3NyYy9idWlsZGVycy9icm93c2VyLWVzYnVpbGQvYnVpbGRlci1zdGF0dXMtd2FybmluZ3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBS0gsTUFBTSxtQkFBbUIsR0FBdUM7SUFDOUQsU0FBUztJQUVULGlCQUFpQjtJQUNqQixVQUFVO0lBQ1YsZ0VBQWdFO0lBQ2hFLDhCQUE4QjtJQUM5Qiw0QkFBNEI7SUFFNUIsZUFBZTtJQUNmLFdBQVc7SUFFWCxnQ0FBZ0M7SUFDaEMsaUJBQWlCO0lBRWpCLDhEQUE4RDtJQUM5RCxhQUFhO0lBQ2IsYUFBYTtJQUNiLHFCQUFxQjtJQUVyQixxQ0FBcUM7SUFDckMsbUJBQW1CO0NBQ3BCLENBQUM7QUFFRixTQUFnQix3QkFBd0IsQ0FBQyxPQUE4QixFQUFFLE9BQXVCO0lBQzlGLDZCQUE2QjtJQUM3QixLQUFLLE1BQU0saUJBQWlCLElBQUksbUJBQW1CLEVBQUU7UUFDbkQsTUFBTSxLQUFLLEdBQUksT0FBNEMsQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBRS9FLElBQUksS0FBSyxLQUFLLFNBQVMsSUFBSSxLQUFLLEtBQUssS0FBSyxFQUFFO1lBQzFDLFNBQVM7U0FDVjtRQUNELElBQUksS0FBSyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsSUFBSSxLQUFLLENBQUMsTUFBTSxLQUFLLENBQUMsRUFBRTtZQUM5QyxTQUFTO1NBQ1Y7UUFDRCxJQUFJLE9BQU8sS0FBSyxLQUFLLFFBQVEsSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7WUFDaEUsU0FBUztTQUNWO1FBRUQsSUFDRSxpQkFBaUIsS0FBSyxhQUFhO1lBQ25DLGlCQUFpQixLQUFLLGFBQWE7WUFDbkMsaUJBQWlCLEtBQUsscUJBQXFCO1lBQzNDLGlCQUFpQixLQUFLLFdBQVcsRUFDakM7WUFDQSxPQUFPLENBQUMsTUFBTSxDQUFDLElBQUksQ0FDakIsUUFBUSxpQkFBaUIsMkRBQTJELENBQ3JGLENBQUM7WUFDRixTQUFTO1NBQ1Y7UUFFRCxPQUFPLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxRQUFRLGlCQUFpQixnREFBZ0QsQ0FBQyxDQUFDO0tBQ2hHO0FBQ0gsQ0FBQztBQTdCRCw0REE2QkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHsgQnVpbGRlckNvbnRleHQgfSBmcm9tICdAYW5ndWxhci1kZXZraXQvYXJjaGl0ZWN0JztcbmltcG9ydCB7IFNjaGVtYSBhcyBCcm93c2VyQnVpbGRlck9wdGlvbnMgfSBmcm9tICcuL3NjaGVtYSc7XG5cbmNvbnN0IFVOU1VQUE9SVEVEX09QVElPTlM6IEFycmF5PGtleW9mIEJyb3dzZXJCdWlsZGVyT3B0aW9ucz4gPSBbXG4gICdidWRnZXRzJyxcblxuICAvLyAqIGkxOG4gc3VwcG9ydFxuICAnbG9jYWxpemUnLFxuICAvLyBUaGUgZm9sbG93aW5nIHR3byBoYXZlIG5vIGVmZmVjdCB3aGVuIGxvY2FsaXplIGlzIG5vdCBlbmFibGVkXG4gIC8vICdpMThuRHVwbGljYXRlVHJhbnNsYXRpb24nLFxuICAvLyAnaTE4bk1pc3NpbmdUcmFuc2xhdGlvbicsXG5cbiAgLy8gKiBEZXByZWNhdGVkXG4gICdkZXBsb3lVcmwnLFxuXG4gIC8vICogQWx3YXlzIGVuYWJsZWQgd2l0aCBlc2J1aWxkXG4gIC8vICdjb21tb25DaHVuaycsXG5cbiAgLy8gKiBVbnVzZWQgYnkgYnVpbGRlciBhbmQgd2lsbCBiZSByZW1vdmVkIGluIGEgZnV0dXJlIHJlbGVhc2VcbiAgJ25hbWVkQ2h1bmtzJyxcbiAgJ3ZlbmRvckNodW5rJyxcbiAgJ3Jlc291cmNlc091dHB1dFBhdGgnLFxuXG4gIC8vICogQ3VycmVudGx5IHVuc3VwcG9ydGVkIGJ5IGVzYnVpbGRcbiAgJ3dlYldvcmtlclRzQ29uZmlnJyxcbl07XG5cbmV4cG9ydCBmdW5jdGlvbiBsb2dCdWlsZGVyU3RhdHVzV2FybmluZ3Mob3B0aW9uczogQnJvd3NlckJ1aWxkZXJPcHRpb25zLCBjb250ZXh0OiBCdWlsZGVyQ29udGV4dCkge1xuICAvLyBWYWxpZGF0ZSBzdXBwb3J0ZWQgb3B0aW9uc1xuICBmb3IgKGNvbnN0IHVuc3VwcG9ydGVkT3B0aW9uIG9mIFVOU1VQUE9SVEVEX09QVElPTlMpIHtcbiAgICBjb25zdCB2YWx1ZSA9IChvcHRpb25zIGFzIHVua25vd24gYXMgQnJvd3NlckJ1aWxkZXJPcHRpb25zKVt1bnN1cHBvcnRlZE9wdGlvbl07XG5cbiAgICBpZiAodmFsdWUgPT09IHVuZGVmaW5lZCB8fCB2YWx1ZSA9PT0gZmFsc2UpIHtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cbiAgICBpZiAoQXJyYXkuaXNBcnJheSh2YWx1ZSkgJiYgdmFsdWUubGVuZ3RoID09PSAwKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcgJiYgT2JqZWN0LmtleXModmFsdWUpLmxlbmd0aCA9PT0gMCkge1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgaWYgKFxuICAgICAgdW5zdXBwb3J0ZWRPcHRpb24gPT09ICduYW1lZENodW5rcycgfHxcbiAgICAgIHVuc3VwcG9ydGVkT3B0aW9uID09PSAndmVuZG9yQ2h1bmsnIHx8XG4gICAgICB1bnN1cHBvcnRlZE9wdGlvbiA9PT0gJ3Jlc291cmNlc091dHB1dFBhdGgnIHx8XG4gICAgICB1bnN1cHBvcnRlZE9wdGlvbiA9PT0gJ2RlcGxveVVybCdcbiAgICApIHtcbiAgICAgIGNvbnRleHQubG9nZ2VyLndhcm4oXG4gICAgICAgIGBUaGUgJyR7dW5zdXBwb3J0ZWRPcHRpb259JyBvcHRpb24gaXMgbm90IHVzZWQgYnkgdGhpcyBidWlsZGVyIGFuZCB3aWxsIGJlIGlnbm9yZWQuYCxcbiAgICAgICk7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBjb250ZXh0LmxvZ2dlci53YXJuKGBUaGUgJyR7dW5zdXBwb3J0ZWRPcHRpb259JyBvcHRpb24gaXMgbm90IHlldCBzdXBwb3J0ZWQgYnkgdGhpcyBidWlsZGVyLmApO1xuICB9XG59XG4iXX0= \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGRlci1zdGF0dXMtd2FybmluZ3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9hbmd1bGFyX2RldmtpdC9idWlsZF9hbmd1bGFyL3NyYy9idWlsZGVycy9icm93c2VyLWVzYnVpbGQvYnVpbGRlci1zdGF0dXMtd2FybmluZ3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBS0gsTUFBTSxtQkFBbUIsR0FBdUM7SUFDOUQsU0FBUztJQUVULGVBQWU7SUFDZixXQUFXO0lBRVgsZ0NBQWdDO0lBQ2hDLGlCQUFpQjtJQUVqQiw4REFBOEQ7SUFDOUQsYUFBYTtJQUNiLGFBQWE7SUFDYixxQkFBcUI7SUFFckIscUNBQXFDO0lBQ3JDLG1CQUFtQjtDQUNwQixDQUFDO0FBRUYsU0FBZ0Isd0JBQXdCLENBQUMsT0FBOEIsRUFBRSxPQUF1QjtJQUM5Riw2QkFBNkI7SUFDN0IsS0FBSyxNQUFNLGlCQUFpQixJQUFJLG1CQUFtQixFQUFFO1FBQ25ELE1BQU0sS0FBSyxHQUFJLE9BQTRDLENBQUMsaUJBQWlCLENBQUMsQ0FBQztRQUUvRSxJQUFJLEtBQUssS0FBSyxTQUFTLElBQUksS0FBSyxLQUFLLEtBQUssRUFBRTtZQUMxQyxTQUFTO1NBQ1Y7UUFDRCxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLElBQUksS0FBSyxDQUFDLE1BQU0sS0FBSyxDQUFDLEVBQUU7WUFDOUMsU0FBUztTQUNWO1FBQ0QsSUFBSSxPQUFPLEtBQUssS0FBSyxRQUFRLElBQUksTUFBTSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1lBQ2hFLFNBQVM7U0FDVjtRQUVELElBQ0UsaUJBQWlCLEtBQUssYUFBYTtZQUNuQyxpQkFBaUIsS0FBSyxhQUFhO1lBQ25DLGlCQUFpQixLQUFLLHFCQUFxQjtZQUMzQyxpQkFBaUIsS0FBSyxXQUFXLEVBQ2pDO1lBQ0EsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQ2pCLFFBQVEsaUJBQWlCLDJEQUEyRCxDQUNyRixDQUFDO1lBQ0YsU0FBUztTQUNWO1FBRUQsT0FBTyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsUUFBUSxpQkFBaUIsZ0RBQWdELENBQUMsQ0FBQztLQUNoRztBQUNILENBQUM7QUE3QkQsNERBNkJDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbmltcG9ydCB7IEJ1aWxkZXJDb250ZXh0IH0gZnJvbSAnQGFuZ3VsYXItZGV2a2l0L2FyY2hpdGVjdCc7XG5pbXBvcnQgeyBTY2hlbWEgYXMgQnJvd3NlckJ1aWxkZXJPcHRpb25zIH0gZnJvbSAnLi9zY2hlbWEnO1xuXG5jb25zdCBVTlNVUFBPUlRFRF9PUFRJT05TOiBBcnJheTxrZXlvZiBCcm93c2VyQnVpbGRlck9wdGlvbnM+ID0gW1xuICAnYnVkZ2V0cycsXG5cbiAgLy8gKiBEZXByZWNhdGVkXG4gICdkZXBsb3lVcmwnLFxuXG4gIC8vICogQWx3YXlzIGVuYWJsZWQgd2l0aCBlc2J1aWxkXG4gIC8vICdjb21tb25DaHVuaycsXG5cbiAgLy8gKiBVbnVzZWQgYnkgYnVpbGRlciBhbmQgd2lsbCBiZSByZW1vdmVkIGluIGEgZnV0dXJlIHJlbGVhc2VcbiAgJ25hbWVkQ2h1bmtzJyxcbiAgJ3ZlbmRvckNodW5rJyxcbiAgJ3Jlc291cmNlc091dHB1dFBhdGgnLFxuXG4gIC8vICogQ3VycmVudGx5IHVuc3VwcG9ydGVkIGJ5IGVzYnVpbGRcbiAgJ3dlYldvcmtlclRzQ29uZmlnJyxcbl07XG5cbmV4cG9ydCBmdW5jdGlvbiBsb2dCdWlsZGVyU3RhdHVzV2FybmluZ3Mob3B0aW9uczogQnJvd3NlckJ1aWxkZXJPcHRpb25zLCBjb250ZXh0OiBCdWlsZGVyQ29udGV4dCkge1xuICAvLyBWYWxpZGF0ZSBzdXBwb3J0ZWQgb3B0aW9uc1xuICBmb3IgKGNvbnN0IHVuc3VwcG9ydGVkT3B0aW9uIG9mIFVOU1VQUE9SVEVEX09QVElPTlMpIHtcbiAgICBjb25zdCB2YWx1ZSA9IChvcHRpb25zIGFzIHVua25vd24gYXMgQnJvd3NlckJ1aWxkZXJPcHRpb25zKVt1bnN1cHBvcnRlZE9wdGlvbl07XG5cbiAgICBpZiAodmFsdWUgPT09IHVuZGVmaW5lZCB8fCB2YWx1ZSA9PT0gZmFsc2UpIHtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cbiAgICBpZiAoQXJyYXkuaXNBcnJheSh2YWx1ZSkgJiYgdmFsdWUubGVuZ3RoID09PSAwKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG4gICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ29iamVjdCcgJiYgT2JqZWN0LmtleXModmFsdWUpLmxlbmd0aCA9PT0gMCkge1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgaWYgKFxuICAgICAgdW5zdXBwb3J0ZWRPcHRpb24gPT09ICduYW1lZENodW5rcycgfHxcbiAgICAgIHVuc3VwcG9ydGVkT3B0aW9uID09PSAndmVuZG9yQ2h1bmsnIHx8XG4gICAgICB1bnN1cHBvcnRlZE9wdGlvbiA9PT0gJ3Jlc291cmNlc091dHB1dFBhdGgnIHx8XG4gICAgICB1bnN1cHBvcnRlZE9wdGlvbiA9PT0gJ2RlcGxveVVybCdcbiAgICApIHtcbiAgICAgIGNvbnRleHQubG9nZ2VyLndhcm4oXG4gICAgICAgIGBUaGUgJyR7dW5zdXBwb3J0ZWRPcHRpb259JyBvcHRpb24gaXMgbm90IHVzZWQgYnkgdGhpcyBidWlsZGVyIGFuZCB3aWxsIGJlIGlnbm9yZWQuYCxcbiAgICAgICk7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBjb250ZXh0LmxvZ2dlci53YXJuKGBUaGUgJyR7dW5zdXBwb3J0ZWRPcHRpb259JyBvcHRpb24gaXMgbm90IHlldCBzdXBwb3J0ZWQgYnkgdGhpcyBidWlsZGVyLmApO1xuICB9XG59XG4iXX0= \ No newline at end of file diff --git a/src/builders/dev-server/vite-server.js b/src/builders/dev-server/vite-server.js index 8e06700d..b5317ce3 100644 --- a/src/builders/dev-server/vite-server.js +++ b/src/builders/dev-server/vite-server.js @@ -40,6 +40,7 @@ const node_crypto_1 = require("node:crypto"); const promises_1 = require("node:fs/promises"); const node_path_1 = __importStar(require("node:path")); const javascript_transformer_1 = require("../../tools/esbuild/javascript-transformer"); +const i18n_locale_plugin_1 = require("../../tools/vite/i18n-locale-plugin"); const render_page_1 = require("../../utils/server-rendering/render-page"); const webpack_browser_config_1 = require("../../utils/webpack-browser-config"); const browser_esbuild_1 = require("../browser-esbuild"); @@ -63,6 +64,17 @@ async function* serveWithVite(serverOptions, builderName, context) { if (serverOptions.servePath === undefined && browserOptions.baseHref !== undefined) { serverOptions.servePath = browserOptions.baseHref; } + // The development server currently only supports a single locale when localizing. + // This matches the behavior of the Webpack-based development server but could be expanded in the future. + if (browserOptions.localize === true || + (Array.isArray(browserOptions.localize) && browserOptions.localize.length > 1)) { + context.logger.warn('Localization (`localize` option) has been disabled. The development server only supports localizing a single locale per build.'); + browserOptions.localize = false; + } + else if (browserOptions.localize) { + // When localization is enabled with a single locale, force a flat path to maintain behavior with the existing Webpack-based dev server. + browserOptions.forceI18nFlatOutput = true; + } // Setup the prebundling transformer that will be shared across Vite prebundling requests const prebundleTransformer = new javascript_transformer_1.JavaScriptTransformer( // Always enable JIT linking to support applications built with and without AOT. @@ -248,6 +260,7 @@ async function setupServer(serverOptions, outputFiles, assets, preserveSymlinks, external: prebundleExclude, }, plugins: [ + (0, i18n_locale_plugin_1.createAngularLocaleDataPlugin)(), { name: 'vite:angular-memory', // Ensures plugin hooks run before built-in Vite hooks @@ -452,4 +465,4 @@ function pathnameWithoutServePath(url, serverOptions) { } return pathname; } -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vite-server.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/dev-server/vite-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 type { BuilderContext } from '@angular-devkit/architect';\nimport type { json, logging } from '@angular-devkit/core';\nimport type { OutputFile } from 'esbuild';\nimport { lookup as lookupMimeType } from 'mrmime';\nimport assert from 'node:assert';\nimport { BinaryLike, createHash } from 'node:crypto';\nimport { readFile } from 'node:fs/promises';\nimport { ServerResponse } from 'node:http';\nimport type { AddressInfo } from 'node:net';\nimport path, { posix } from 'node:path';\nimport type { Connect, InlineConfig, ViteDevServer } from 'vite';\nimport { JavaScriptTransformer } from '../../tools/esbuild/javascript-transformer';\nimport { RenderOptions, renderPage } from '../../utils/server-rendering/render-page';\nimport { getIndexOutputFile } from '../../utils/webpack-browser-config';\nimport { buildEsbuildBrowser } from '../browser-esbuild';\nimport { Schema as BrowserBuilderOptions } from '../browser-esbuild/schema';\nimport { loadProxyConfiguration } from './load-proxy-config';\nimport type { NormalizedDevServerOptions } from './options';\nimport type { DevServerBuilderOutput } from './webpack-server';\n\ninterface OutputFileRecord {\n  contents: Uint8Array;\n  size: number;\n  hash?: Buffer;\n  updated: boolean;\n}\n\nconst SSG_MARKER_REGEXP = /ng-server-context=[\"']\\w*\\|?ssg\\|?\\w*[\"']/;\n\nfunction hashContent(contents: BinaryLike): Buffer {\n  // TODO: Consider xxhash\n  return createHash('sha256').update(contents).digest();\n}\n\nexport async function* serveWithVite(\n  serverOptions: NormalizedDevServerOptions,\n  builderName: string,\n  context: BuilderContext,\n): AsyncIterableIterator<DevServerBuilderOutput> {\n  // Get the browser configuration from the target name.\n  const rawBrowserOptions = (await context.getTargetOptions(\n    serverOptions.browserTarget,\n  )) as json.JsonObject & BrowserBuilderOptions;\n\n  const browserOptions = (await context.validateOptions(\n    {\n      ...rawBrowserOptions,\n      watch: serverOptions.watch,\n      poll: serverOptions.poll,\n      verbose: serverOptions.verbose,\n    } as json.JsonObject & BrowserBuilderOptions,\n    builderName,\n  )) as json.JsonObject & BrowserBuilderOptions;\n  // Set all packages as external to support Vite's prebundle caching\n  browserOptions.externalPackages = serverOptions.cacheOptions.enabled;\n\n  if (serverOptions.servePath === undefined && browserOptions.baseHref !== undefined) {\n    serverOptions.servePath = browserOptions.baseHref;\n  }\n\n  // Setup the prebundling transformer that will be shared across Vite prebundling requests\n  const prebundleTransformer = new JavaScriptTransformer(\n    // Always enable JIT linking to support applications built with and without AOT.\n    // In a development environment the additional scope information does not\n    // have a negative effect unlike production where final output size is relevant.\n    { sourcemap: true, jit: true },\n    1,\n  );\n\n  // Extract output index from options\n  // TODO: Provide this info from the build results\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  const htmlIndexPath = getIndexOutputFile(browserOptions.index as any);\n\n  // dynamically import Vite for ESM compatibility\n  const { createServer, normalizePath } = await import('vite');\n\n  let server: ViteDevServer | undefined;\n  let listeningAddress: AddressInfo | undefined;\n  const generatedFiles = new Map<string, OutputFileRecord>();\n  const assetFiles = new Map<string, string>();\n  // TODO: Switch this to an architect schedule call when infrastructure settings are supported\n  for await (const result of buildEsbuildBrowser(browserOptions, context, {\n    write: false,\n  })) {\n    assert(result.outputFiles, 'Builder did not provide result files.');\n\n    // Analyze result files for changes\n    analyzeResultFiles(normalizePath, htmlIndexPath, result.outputFiles, generatedFiles);\n\n    assetFiles.clear();\n    if (result.assetFiles) {\n      for (const asset of result.assetFiles) {\n        assetFiles.set('/' + normalizePath(asset.destination), asset.source);\n      }\n    }\n\n    if (server) {\n      handleUpdate(generatedFiles, server, serverOptions, context.logger);\n    } else {\n      // Setup server and start listening\n      const serverConfiguration = await setupServer(\n        serverOptions,\n        generatedFiles,\n        assetFiles,\n        browserOptions.preserveSymlinks,\n        browserOptions.externalDependencies,\n        !!browserOptions.ssr,\n        prebundleTransformer,\n      );\n\n      server = await createServer(serverConfiguration);\n\n      await server.listen();\n      listeningAddress = server.httpServer?.address() as AddressInfo;\n\n      // log connection information\n      server.printUrls();\n    }\n\n    // TODO: adjust output typings to reflect both development servers\n    yield { success: true, port: listeningAddress?.port } as unknown as DevServerBuilderOutput;\n  }\n\n  // Add cleanup logic via a builder teardown\n  let deferred: () => void;\n  context.addTeardown(async () => {\n    await server?.close();\n    await prebundleTransformer.close();\n    deferred?.();\n  });\n  await new Promise<void>((resolve) => (deferred = resolve));\n}\n\nfunction handleUpdate(\n  generatedFiles: Map<string, OutputFileRecord>,\n  server: ViteDevServer,\n  serverOptions: NormalizedDevServerOptions,\n  logger: logging.LoggerApi,\n): void {\n  const updatedFiles: string[] = [];\n\n  // Invalidate any updated files\n  for (const [file, record] of generatedFiles) {\n    if (record.updated) {\n      updatedFiles.push(file);\n      const updatedModules = server.moduleGraph.getModulesByFile(file);\n      updatedModules?.forEach((m) => server?.moduleGraph.invalidateModule(m));\n    }\n  }\n\n  if (!updatedFiles.length) {\n    return;\n  }\n\n  if (serverOptions.liveReload || serverOptions.hmr) {\n    if (updatedFiles.every((f) => f.endsWith('.css'))) {\n      const timestamp = Date.now();\n      server.ws.send({\n        type: 'update',\n        updates: updatedFiles.map((f) => {\n          const filePath = f.slice(1); // Remove leading slash.\n\n          return {\n            type: 'css-update',\n            timestamp,\n            path: filePath,\n            acceptedPath: filePath,\n          };\n        }),\n      });\n\n      logger.info('HMR update sent to client(s)...');\n\n      return;\n    }\n  }\n\n  // Send reload command to clients\n  if (serverOptions.liveReload) {\n    logger.info('Reloading client(s)...');\n\n    server.ws.send({\n      type: 'full-reload',\n      path: '*',\n    });\n  }\n}\n\nfunction analyzeResultFiles(\n  normalizePath: (id: string) => string,\n  htmlIndexPath: string,\n  resultFiles: OutputFile[],\n  generatedFiles: Map<string, OutputFileRecord>,\n) {\n  const seen = new Set<string>(['/index.html']);\n  for (const file of resultFiles) {\n    let filePath;\n    if (file.path === htmlIndexPath) {\n      // Convert custom index output path to standard index path for dev-server usage.\n      // This mimics the Webpack dev-server behavior.\n      filePath = '/index.html';\n    } else {\n      filePath = '/' + normalizePath(file.path);\n    }\n    seen.add(filePath);\n\n    // Skip analysis of sourcemaps\n    if (filePath.endsWith('.map')) {\n      generatedFiles.set(filePath, {\n        contents: file.contents,\n        size: file.contents.byteLength,\n        updated: false,\n      });\n\n      continue;\n    }\n\n    let fileHash: Buffer | undefined;\n    const existingRecord = generatedFiles.get(filePath);\n    if (existingRecord && existingRecord.size === file.contents.byteLength) {\n      // Only hash existing file when needed\n      if (existingRecord.hash === undefined) {\n        existingRecord.hash = hashContent(existingRecord.contents);\n      }\n\n      // Compare against latest result output\n      fileHash = hashContent(file.contents);\n      if (fileHash.equals(existingRecord.hash)) {\n        // Same file\n        existingRecord.updated = false;\n        continue;\n      }\n    }\n\n    generatedFiles.set(filePath, {\n      contents: file.contents,\n      size: file.contents.byteLength,\n      hash: fileHash,\n      updated: true,\n    });\n  }\n\n  // Clear stale output files\n  for (const file of generatedFiles.keys()) {\n    if (!seen.has(file)) {\n      generatedFiles.delete(file);\n    }\n  }\n}\n\n// eslint-disable-next-line max-lines-per-function\nexport async function setupServer(\n  serverOptions: NormalizedDevServerOptions,\n  outputFiles: Map<string, OutputFileRecord>,\n  assets: Map<string, string>,\n  preserveSymlinks: boolean | undefined,\n  prebundleExclude: string[] | undefined,\n  ssr: boolean,\n  prebundleTransformer: JavaScriptTransformer,\n): Promise<InlineConfig> {\n  const proxy = await loadProxyConfiguration(\n    serverOptions.workspaceRoot,\n    serverOptions.proxyConfig,\n    true,\n  );\n\n  // dynamically import Vite for ESM compatibility\n  const { normalizePath } = await import('vite');\n\n  const configuration: InlineConfig = {\n    configFile: false,\n    envFile: false,\n    cacheDir: path.join(serverOptions.cacheOptions.path, 'vite'),\n    root: serverOptions.workspaceRoot,\n    publicDir: false,\n    esbuild: false,\n    mode: 'development',\n    appType: 'spa',\n    css: {\n      devSourcemap: true,\n    },\n    base: serverOptions.servePath,\n    resolve: {\n      mainFields: ['es2020', 'browser', 'module', 'main'],\n      preserveSymlinks,\n    },\n    server: {\n      port: serverOptions.port,\n      strictPort: true,\n      host: serverOptions.host,\n      open: serverOptions.open,\n      headers: serverOptions.headers,\n      proxy,\n      // Currently does not appear to be a way to disable file watching directly so ignore all files\n      watch: {\n        ignored: ['**/*'],\n      },\n    },\n    ssr: {\n      // Exclude any provided dependencies (currently build defined externals)\n      external: prebundleExclude,\n    },\n    plugins: [\n      {\n        name: 'vite:angular-memory',\n        // Ensures plugin hooks run before built-in Vite hooks\n        enforce: 'pre',\n        async resolveId(source, importer) {\n          if (importer && source.startsWith('.')) {\n            // Remove query if present\n            const [importerFile] = importer.split('?', 1);\n\n            source = normalizePath(path.join(path.dirname(importerFile), source));\n          }\n\n          const [file] = source.split('?', 1);\n          if (outputFiles.has(file)) {\n            return source;\n          }\n        },\n        load(id) {\n          const [file] = id.split('?', 1);\n          const codeContents = outputFiles.get(file)?.contents;\n          if (codeContents === undefined) {\n            return;\n          }\n\n          const code = Buffer.from(codeContents).toString('utf-8');\n          const mapContents = outputFiles.get(file + '.map')?.contents;\n\n          return {\n            // Remove source map URL comments from the code if a sourcemap is present.\n            // Vite will inline and add an additional sourcemap URL for the sourcemap.\n            code: mapContents ? code.replace(/^\\/\\/# sourceMappingURL=[^\\r\\n]*/gm, '') : code,\n            map: mapContents && Buffer.from(mapContents).toString('utf-8'),\n          };\n        },\n        configureServer(server) {\n          // Assets and resources get handled first\n          server.middlewares.use(function angularAssetsMiddleware(req, res, next) {\n            if (req.url === undefined || res.writableEnded) {\n              return;\n            }\n\n            // Parse the incoming request.\n            // The base of the URL is unused but required to parse the URL.\n            const pathname = pathnameWithoutServePath(req.url, serverOptions);\n            const extension = path.extname(pathname);\n\n            // Rewrite all build assets to a vite raw fs URL\n            const assetSourcePath = assets.get(pathname);\n            if (assetSourcePath !== undefined) {\n              // The encoding needs to match what happens in the vite static middleware.\n              // ref: https://github.com/vitejs/vite/blob/d4f13bd81468961c8c926438e815ab6b1c82735e/packages/vite/src/node/server/middlewares/static.ts#L163\n              req.url = `/@fs/${encodeURI(assetSourcePath)}`;\n              next();\n\n              return;\n            }\n\n            // Resource files are handled directly.\n            // Global stylesheets (CSS files) are currently considered resources to workaround\n            // dev server sourcemap issues with stylesheets.\n            if (extension !== '.js' && extension !== '.html') {\n              const outputFile = outputFiles.get(pathname);\n              if (outputFile) {\n                const mimeType = lookupMimeType(extension);\n                if (mimeType) {\n                  res.setHeader('Content-Type', mimeType);\n                }\n                res.setHeader('Cache-Control', 'no-cache');\n                if (serverOptions.headers) {\n                  Object.entries(serverOptions.headers).forEach(([name, value]) =>\n                    res.setHeader(name, value),\n                  );\n                }\n                res.end(outputFile.contents);\n\n                return;\n              }\n            }\n\n            next();\n          });\n\n          // Returning a function, installs middleware after the main transform middleware but\n          // before the built-in HTML middleware\n          return () => {\n            function angularSSRMiddleware(\n              req: Connect.IncomingMessage,\n              res: ServerResponse,\n              next: Connect.NextFunction,\n            ) {\n              const url = req.originalUrl;\n              if (!url || url.endsWith('.html')) {\n                next();\n\n                return;\n              }\n\n              const potentialPrerendered = outputFiles.get(posix.join(url, 'index.html'))?.contents;\n              if (potentialPrerendered) {\n                const content = Buffer.from(potentialPrerendered).toString('utf-8');\n                if (SSG_MARKER_REGEXP.test(content)) {\n                  transformIndexHtmlAndAddHeaders(url, potentialPrerendered, res, next);\n\n                  return;\n                }\n              }\n\n              const rawHtml = outputFiles.get('/index.server.html')?.contents;\n              if (!rawHtml) {\n                next();\n\n                return;\n              }\n\n              transformIndexHtmlAndAddHeaders(url, rawHtml, res, next, async (html) => {\n                const { content } = await renderPage({\n                  document: html,\n                  route: pathnameWithoutServePath(url, serverOptions),\n                  serverContext: 'ssr',\n                  loadBundle: (path: string) =>\n                    server.ssrLoadModule(path.slice(1)) as ReturnType<\n                      NonNullable<RenderOptions['loadBundle']>\n                    >,\n                  // Files here are only needed for critical CSS inlining.\n                  outputFiles: {},\n                  // TODO: add support for critical css inlining.\n                  inlineCriticalCss: false,\n                });\n\n                return content;\n              });\n            }\n\n            if (ssr) {\n              server.middlewares.use(angularSSRMiddleware);\n            }\n\n            server.middlewares.use(function angularIndexMiddleware(req, res, next) {\n              if (!req.url) {\n                next();\n\n                return;\n              }\n\n              // Parse the incoming request.\n              // The base of the URL is unused but required to parse the URL.\n              const pathname = pathnameWithoutServePath(req.url, serverOptions);\n\n              if (pathname === '/' || pathname === `/index.html`) {\n                const rawHtml = outputFiles.get('/index.html')?.contents;\n                if (rawHtml) {\n                  transformIndexHtmlAndAddHeaders(req.url, rawHtml, res, next);\n\n                  return;\n                }\n              }\n\n              next();\n            });\n          };\n\n          function transformIndexHtmlAndAddHeaders(\n            url: string,\n            rawHtml: Uint8Array,\n            res: ServerResponse<import('http').IncomingMessage>,\n            next: Connect.NextFunction,\n            additionalTransformer?: (html: string) => Promise<string | undefined>,\n          ) {\n            server\n              .transformIndexHtml(url, Buffer.from(rawHtml).toString('utf-8'))\n              .then(async (processedHtml) => {\n                if (additionalTransformer) {\n                  const content = await additionalTransformer(processedHtml);\n                  if (!content) {\n                    next();\n\n                    return;\n                  }\n\n                  processedHtml = content;\n                }\n\n                res.setHeader('Content-Type', 'text/html');\n                res.setHeader('Cache-Control', 'no-cache');\n                if (serverOptions.headers) {\n                  Object.entries(serverOptions.headers).forEach(([name, value]) =>\n                    res.setHeader(name, value),\n                  );\n                }\n                res.end(processedHtml);\n              })\n              .catch((error) => next(error));\n          }\n        },\n      },\n    ],\n    optimizeDeps: {\n      // Only enable with caching since it causes prebundle dependencies to be cached\n      disabled: !serverOptions.cacheOptions.enabled,\n      // Exclude any provided dependencies (currently build defined externals)\n      exclude: prebundleExclude,\n      // Skip automatic file-based entry point discovery\n      entries: [],\n      // Add an esbuild plugin to run the Angular linker on dependencies\n      esbuildOptions: {\n        plugins: [\n          {\n            name: 'angular-vite-optimize-deps',\n            setup(build) {\n              build.onLoad({ filter: /\\.[cm]?js$/ }, async (args) => {\n                return {\n                  contents: await prebundleTransformer.transformFile(args.path),\n                  loader: 'js',\n                };\n              });\n            },\n          },\n        ],\n      },\n    },\n  };\n\n  if (serverOptions.ssl) {\n    if (serverOptions.sslCert && serverOptions.sslKey) {\n      // server configuration is defined above\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      configuration.server!.https = {\n        cert: await readFile(serverOptions.sslCert),\n        key: await readFile(serverOptions.sslKey),\n      };\n    } else {\n      const { default: basicSslPlugin } = await import('@vitejs/plugin-basic-ssl');\n      configuration.plugins ??= [];\n      configuration.plugins.push(basicSslPlugin());\n    }\n  }\n\n  return configuration;\n}\n\nfunction pathnameWithoutServePath(url: string, serverOptions: NormalizedDevServerOptions): string {\n  const parsedUrl = new URL(url, 'http://localhost');\n  let pathname = decodeURIComponent(parsedUrl.pathname);\n  if (serverOptions.servePath && pathname.startsWith(serverOptions.servePath)) {\n    pathname = pathname.slice(serverOptions.servePath.length);\n    if (pathname[0] !== '/') {\n      pathname = '/' + pathname;\n    }\n  }\n\n  return pathname;\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"vite-server.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/dev-server/vite-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 type { BuilderContext } from '@angular-devkit/architect';\nimport type { json, logging } from '@angular-devkit/core';\nimport type { OutputFile } from 'esbuild';\nimport { lookup as lookupMimeType } from 'mrmime';\nimport assert from 'node:assert';\nimport { BinaryLike, createHash } from 'node:crypto';\nimport { readFile } from 'node:fs/promises';\nimport { ServerResponse } from 'node:http';\nimport type { AddressInfo } from 'node:net';\nimport path, { posix } from 'node:path';\nimport type { Connect, InlineConfig, ViteDevServer } from 'vite';\nimport { JavaScriptTransformer } from '../../tools/esbuild/javascript-transformer';\nimport { createAngularLocaleDataPlugin } from '../../tools/vite/i18n-locale-plugin';\nimport { RenderOptions, renderPage } from '../../utils/server-rendering/render-page';\nimport { getIndexOutputFile } from '../../utils/webpack-browser-config';\nimport { buildEsbuildBrowser } from '../browser-esbuild';\nimport { Schema as BrowserBuilderOptions } from '../browser-esbuild/schema';\nimport { loadProxyConfiguration } from './load-proxy-config';\nimport type { NormalizedDevServerOptions } from './options';\nimport type { DevServerBuilderOutput } from './webpack-server';\n\ninterface OutputFileRecord {\n  contents: Uint8Array;\n  size: number;\n  hash?: Buffer;\n  updated: boolean;\n}\n\nconst SSG_MARKER_REGEXP = /ng-server-context=[\"']\\w*\\|?ssg\\|?\\w*[\"']/;\n\nfunction hashContent(contents: BinaryLike): Buffer {\n  // TODO: Consider xxhash\n  return createHash('sha256').update(contents).digest();\n}\n\nexport async function* serveWithVite(\n  serverOptions: NormalizedDevServerOptions,\n  builderName: string,\n  context: BuilderContext,\n): AsyncIterableIterator<DevServerBuilderOutput> {\n  // Get the browser configuration from the target name.\n  const rawBrowserOptions = (await context.getTargetOptions(\n    serverOptions.browserTarget,\n  )) as json.JsonObject & BrowserBuilderOptions;\n\n  const browserOptions = (await context.validateOptions(\n    {\n      ...rawBrowserOptions,\n      watch: serverOptions.watch,\n      poll: serverOptions.poll,\n      verbose: serverOptions.verbose,\n    } as json.JsonObject & BrowserBuilderOptions,\n    builderName,\n  )) as json.JsonObject & BrowserBuilderOptions;\n  // Set all packages as external to support Vite's prebundle caching\n  browserOptions.externalPackages = serverOptions.cacheOptions.enabled;\n\n  if (serverOptions.servePath === undefined && browserOptions.baseHref !== undefined) {\n    serverOptions.servePath = browserOptions.baseHref;\n  }\n\n  // The development server currently only supports a single locale when localizing.\n  // This matches the behavior of the Webpack-based development server but could be expanded in the future.\n  if (\n    browserOptions.localize === true ||\n    (Array.isArray(browserOptions.localize) && browserOptions.localize.length > 1)\n  ) {\n    context.logger.warn(\n      'Localization (`localize` option) has been disabled. The development server only supports localizing a single locale per build.',\n    );\n    browserOptions.localize = false;\n  } else if (browserOptions.localize) {\n    // When localization is enabled with a single locale, force a flat path to maintain behavior with the existing Webpack-based dev server.\n    browserOptions.forceI18nFlatOutput = true;\n  }\n\n  // Setup the prebundling transformer that will be shared across Vite prebundling requests\n  const prebundleTransformer = new JavaScriptTransformer(\n    // Always enable JIT linking to support applications built with and without AOT.\n    // In a development environment the additional scope information does not\n    // have a negative effect unlike production where final output size is relevant.\n    { sourcemap: true, jit: true },\n    1,\n  );\n\n  // Extract output index from options\n  // TODO: Provide this info from the build results\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  const htmlIndexPath = getIndexOutputFile(browserOptions.index as any);\n\n  // dynamically import Vite for ESM compatibility\n  const { createServer, normalizePath } = await import('vite');\n\n  let server: ViteDevServer | undefined;\n  let listeningAddress: AddressInfo | undefined;\n  const generatedFiles = new Map<string, OutputFileRecord>();\n  const assetFiles = new Map<string, string>();\n  // TODO: Switch this to an architect schedule call when infrastructure settings are supported\n  for await (const result of buildEsbuildBrowser(browserOptions, context, {\n    write: false,\n  })) {\n    assert(result.outputFiles, 'Builder did not provide result files.');\n\n    // Analyze result files for changes\n    analyzeResultFiles(normalizePath, htmlIndexPath, result.outputFiles, generatedFiles);\n\n    assetFiles.clear();\n    if (result.assetFiles) {\n      for (const asset of result.assetFiles) {\n        assetFiles.set('/' + normalizePath(asset.destination), asset.source);\n      }\n    }\n\n    if (server) {\n      handleUpdate(generatedFiles, server, serverOptions, context.logger);\n    } else {\n      // Setup server and start listening\n      const serverConfiguration = await setupServer(\n        serverOptions,\n        generatedFiles,\n        assetFiles,\n        browserOptions.preserveSymlinks,\n        browserOptions.externalDependencies,\n        !!browserOptions.ssr,\n        prebundleTransformer,\n      );\n\n      server = await createServer(serverConfiguration);\n\n      await server.listen();\n      listeningAddress = server.httpServer?.address() as AddressInfo;\n\n      // log connection information\n      server.printUrls();\n    }\n\n    // TODO: adjust output typings to reflect both development servers\n    yield { success: true, port: listeningAddress?.port } as unknown as DevServerBuilderOutput;\n  }\n\n  // Add cleanup logic via a builder teardown\n  let deferred: () => void;\n  context.addTeardown(async () => {\n    await server?.close();\n    await prebundleTransformer.close();\n    deferred?.();\n  });\n  await new Promise<void>((resolve) => (deferred = resolve));\n}\n\nfunction handleUpdate(\n  generatedFiles: Map<string, OutputFileRecord>,\n  server: ViteDevServer,\n  serverOptions: NormalizedDevServerOptions,\n  logger: logging.LoggerApi,\n): void {\n  const updatedFiles: string[] = [];\n\n  // Invalidate any updated files\n  for (const [file, record] of generatedFiles) {\n    if (record.updated) {\n      updatedFiles.push(file);\n      const updatedModules = server.moduleGraph.getModulesByFile(file);\n      updatedModules?.forEach((m) => server?.moduleGraph.invalidateModule(m));\n    }\n  }\n\n  if (!updatedFiles.length) {\n    return;\n  }\n\n  if (serverOptions.liveReload || serverOptions.hmr) {\n    if (updatedFiles.every((f) => f.endsWith('.css'))) {\n      const timestamp = Date.now();\n      server.ws.send({\n        type: 'update',\n        updates: updatedFiles.map((f) => {\n          const filePath = f.slice(1); // Remove leading slash.\n\n          return {\n            type: 'css-update',\n            timestamp,\n            path: filePath,\n            acceptedPath: filePath,\n          };\n        }),\n      });\n\n      logger.info('HMR update sent to client(s)...');\n\n      return;\n    }\n  }\n\n  // Send reload command to clients\n  if (serverOptions.liveReload) {\n    logger.info('Reloading client(s)...');\n\n    server.ws.send({\n      type: 'full-reload',\n      path: '*',\n    });\n  }\n}\n\nfunction analyzeResultFiles(\n  normalizePath: (id: string) => string,\n  htmlIndexPath: string,\n  resultFiles: OutputFile[],\n  generatedFiles: Map<string, OutputFileRecord>,\n) {\n  const seen = new Set<string>(['/index.html']);\n  for (const file of resultFiles) {\n    let filePath;\n    if (file.path === htmlIndexPath) {\n      // Convert custom index output path to standard index path for dev-server usage.\n      // This mimics the Webpack dev-server behavior.\n      filePath = '/index.html';\n    } else {\n      filePath = '/' + normalizePath(file.path);\n    }\n    seen.add(filePath);\n\n    // Skip analysis of sourcemaps\n    if (filePath.endsWith('.map')) {\n      generatedFiles.set(filePath, {\n        contents: file.contents,\n        size: file.contents.byteLength,\n        updated: false,\n      });\n\n      continue;\n    }\n\n    let fileHash: Buffer | undefined;\n    const existingRecord = generatedFiles.get(filePath);\n    if (existingRecord && existingRecord.size === file.contents.byteLength) {\n      // Only hash existing file when needed\n      if (existingRecord.hash === undefined) {\n        existingRecord.hash = hashContent(existingRecord.contents);\n      }\n\n      // Compare against latest result output\n      fileHash = hashContent(file.contents);\n      if (fileHash.equals(existingRecord.hash)) {\n        // Same file\n        existingRecord.updated = false;\n        continue;\n      }\n    }\n\n    generatedFiles.set(filePath, {\n      contents: file.contents,\n      size: file.contents.byteLength,\n      hash: fileHash,\n      updated: true,\n    });\n  }\n\n  // Clear stale output files\n  for (const file of generatedFiles.keys()) {\n    if (!seen.has(file)) {\n      generatedFiles.delete(file);\n    }\n  }\n}\n\n// eslint-disable-next-line max-lines-per-function\nexport async function setupServer(\n  serverOptions: NormalizedDevServerOptions,\n  outputFiles: Map<string, OutputFileRecord>,\n  assets: Map<string, string>,\n  preserveSymlinks: boolean | undefined,\n  prebundleExclude: string[] | undefined,\n  ssr: boolean,\n  prebundleTransformer: JavaScriptTransformer,\n): Promise<InlineConfig> {\n  const proxy = await loadProxyConfiguration(\n    serverOptions.workspaceRoot,\n    serverOptions.proxyConfig,\n    true,\n  );\n\n  // dynamically import Vite for ESM compatibility\n  const { normalizePath } = await import('vite');\n\n  const configuration: InlineConfig = {\n    configFile: false,\n    envFile: false,\n    cacheDir: path.join(serverOptions.cacheOptions.path, 'vite'),\n    root: serverOptions.workspaceRoot,\n    publicDir: false,\n    esbuild: false,\n    mode: 'development',\n    appType: 'spa',\n    css: {\n      devSourcemap: true,\n    },\n    base: serverOptions.servePath,\n    resolve: {\n      mainFields: ['es2020', 'browser', 'module', 'main'],\n      preserveSymlinks,\n    },\n    server: {\n      port: serverOptions.port,\n      strictPort: true,\n      host: serverOptions.host,\n      open: serverOptions.open,\n      headers: serverOptions.headers,\n      proxy,\n      // Currently does not appear to be a way to disable file watching directly so ignore all files\n      watch: {\n        ignored: ['**/*'],\n      },\n    },\n    ssr: {\n      // Exclude any provided dependencies (currently build defined externals)\n      external: prebundleExclude,\n    },\n    plugins: [\n      createAngularLocaleDataPlugin(),\n      {\n        name: 'vite:angular-memory',\n        // Ensures plugin hooks run before built-in Vite hooks\n        enforce: 'pre',\n        async resolveId(source, importer) {\n          if (importer && source.startsWith('.')) {\n            // Remove query if present\n            const [importerFile] = importer.split('?', 1);\n\n            source = normalizePath(path.join(path.dirname(importerFile), source));\n          }\n\n          const [file] = source.split('?', 1);\n          if (outputFiles.has(file)) {\n            return source;\n          }\n        },\n        load(id) {\n          const [file] = id.split('?', 1);\n          const codeContents = outputFiles.get(file)?.contents;\n          if (codeContents === undefined) {\n            return;\n          }\n\n          const code = Buffer.from(codeContents).toString('utf-8');\n          const mapContents = outputFiles.get(file + '.map')?.contents;\n\n          return {\n            // Remove source map URL comments from the code if a sourcemap is present.\n            // Vite will inline and add an additional sourcemap URL for the sourcemap.\n            code: mapContents ? code.replace(/^\\/\\/# sourceMappingURL=[^\\r\\n]*/gm, '') : code,\n            map: mapContents && Buffer.from(mapContents).toString('utf-8'),\n          };\n        },\n        configureServer(server) {\n          // Assets and resources get handled first\n          server.middlewares.use(function angularAssetsMiddleware(req, res, next) {\n            if (req.url === undefined || res.writableEnded) {\n              return;\n            }\n\n            // Parse the incoming request.\n            // The base of the URL is unused but required to parse the URL.\n            const pathname = pathnameWithoutServePath(req.url, serverOptions);\n            const extension = path.extname(pathname);\n\n            // Rewrite all build assets to a vite raw fs URL\n            const assetSourcePath = assets.get(pathname);\n            if (assetSourcePath !== undefined) {\n              // The encoding needs to match what happens in the vite static middleware.\n              // ref: https://github.com/vitejs/vite/blob/d4f13bd81468961c8c926438e815ab6b1c82735e/packages/vite/src/node/server/middlewares/static.ts#L163\n              req.url = `/@fs/${encodeURI(assetSourcePath)}`;\n              next();\n\n              return;\n            }\n\n            // Resource files are handled directly.\n            // Global stylesheets (CSS files) are currently considered resources to workaround\n            // dev server sourcemap issues with stylesheets.\n            if (extension !== '.js' && extension !== '.html') {\n              const outputFile = outputFiles.get(pathname);\n              if (outputFile) {\n                const mimeType = lookupMimeType(extension);\n                if (mimeType) {\n                  res.setHeader('Content-Type', mimeType);\n                }\n                res.setHeader('Cache-Control', 'no-cache');\n                if (serverOptions.headers) {\n                  Object.entries(serverOptions.headers).forEach(([name, value]) =>\n                    res.setHeader(name, value),\n                  );\n                }\n                res.end(outputFile.contents);\n\n                return;\n              }\n            }\n\n            next();\n          });\n\n          // Returning a function, installs middleware after the main transform middleware but\n          // before the built-in HTML middleware\n          return () => {\n            function angularSSRMiddleware(\n              req: Connect.IncomingMessage,\n              res: ServerResponse,\n              next: Connect.NextFunction,\n            ) {\n              const url = req.originalUrl;\n              if (!url || url.endsWith('.html')) {\n                next();\n\n                return;\n              }\n\n              const potentialPrerendered = outputFiles.get(posix.join(url, 'index.html'))?.contents;\n              if (potentialPrerendered) {\n                const content = Buffer.from(potentialPrerendered).toString('utf-8');\n                if (SSG_MARKER_REGEXP.test(content)) {\n                  transformIndexHtmlAndAddHeaders(url, potentialPrerendered, res, next);\n\n                  return;\n                }\n              }\n\n              const rawHtml = outputFiles.get('/index.server.html')?.contents;\n              if (!rawHtml) {\n                next();\n\n                return;\n              }\n\n              transformIndexHtmlAndAddHeaders(url, rawHtml, res, next, async (html) => {\n                const { content } = await renderPage({\n                  document: html,\n                  route: pathnameWithoutServePath(url, serverOptions),\n                  serverContext: 'ssr',\n                  loadBundle: (path: string) =>\n                    server.ssrLoadModule(path.slice(1)) as ReturnType<\n                      NonNullable<RenderOptions['loadBundle']>\n                    >,\n                  // Files here are only needed for critical CSS inlining.\n                  outputFiles: {},\n                  // TODO: add support for critical css inlining.\n                  inlineCriticalCss: false,\n                });\n\n                return content;\n              });\n            }\n\n            if (ssr) {\n              server.middlewares.use(angularSSRMiddleware);\n            }\n\n            server.middlewares.use(function angularIndexMiddleware(req, res, next) {\n              if (!req.url) {\n                next();\n\n                return;\n              }\n\n              // Parse the incoming request.\n              // The base of the URL is unused but required to parse the URL.\n              const pathname = pathnameWithoutServePath(req.url, serverOptions);\n\n              if (pathname === '/' || pathname === `/index.html`) {\n                const rawHtml = outputFiles.get('/index.html')?.contents;\n                if (rawHtml) {\n                  transformIndexHtmlAndAddHeaders(req.url, rawHtml, res, next);\n\n                  return;\n                }\n              }\n\n              next();\n            });\n          };\n\n          function transformIndexHtmlAndAddHeaders(\n            url: string,\n            rawHtml: Uint8Array,\n            res: ServerResponse<import('http').IncomingMessage>,\n            next: Connect.NextFunction,\n            additionalTransformer?: (html: string) => Promise<string | undefined>,\n          ) {\n            server\n              .transformIndexHtml(url, Buffer.from(rawHtml).toString('utf-8'))\n              .then(async (processedHtml) => {\n                if (additionalTransformer) {\n                  const content = await additionalTransformer(processedHtml);\n                  if (!content) {\n                    next();\n\n                    return;\n                  }\n\n                  processedHtml = content;\n                }\n\n                res.setHeader('Content-Type', 'text/html');\n                res.setHeader('Cache-Control', 'no-cache');\n                if (serverOptions.headers) {\n                  Object.entries(serverOptions.headers).forEach(([name, value]) =>\n                    res.setHeader(name, value),\n                  );\n                }\n                res.end(processedHtml);\n              })\n              .catch((error) => next(error));\n          }\n        },\n      },\n    ],\n    optimizeDeps: {\n      // Only enable with caching since it causes prebundle dependencies to be cached\n      disabled: !serverOptions.cacheOptions.enabled,\n      // Exclude any provided dependencies (currently build defined externals)\n      exclude: prebundleExclude,\n      // Skip automatic file-based entry point discovery\n      entries: [],\n      // Add an esbuild plugin to run the Angular linker on dependencies\n      esbuildOptions: {\n        plugins: [\n          {\n            name: 'angular-vite-optimize-deps',\n            setup(build) {\n              build.onLoad({ filter: /\\.[cm]?js$/ }, async (args) => {\n                return {\n                  contents: await prebundleTransformer.transformFile(args.path),\n                  loader: 'js',\n                };\n              });\n            },\n          },\n        ],\n      },\n    },\n  };\n\n  if (serverOptions.ssl) {\n    if (serverOptions.sslCert && serverOptions.sslKey) {\n      // server configuration is defined above\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      configuration.server!.https = {\n        cert: await readFile(serverOptions.sslCert),\n        key: await readFile(serverOptions.sslKey),\n      };\n    } else {\n      const { default: basicSslPlugin } = await import('@vitejs/plugin-basic-ssl');\n      configuration.plugins ??= [];\n      configuration.plugins.push(basicSslPlugin());\n    }\n  }\n\n  return configuration;\n}\n\nfunction pathnameWithoutServePath(url: string, serverOptions: NormalizedDevServerOptions): string {\n  const parsedUrl = new URL(url, 'http://localhost');\n  let pathname = decodeURIComponent(parsedUrl.pathname);\n  if (serverOptions.servePath && pathname.startsWith(serverOptions.servePath)) {\n    pathname = pathname.slice(serverOptions.servePath.length);\n    if (pathname[0] !== '/') {\n      pathname = '/' + pathname;\n    }\n  }\n\n  return pathname;\n}\n"]} \ No newline at end of file diff --git a/src/builders/extract-i18n/application-extraction.js b/src/builders/extract-i18n/application-extraction.js index 2d396585..8d95cfa2 100644 --- a/src/builders/extract-i18n/application-extraction.js +++ b/src/builders/extract-i18n/application-extraction.js @@ -21,6 +21,7 @@ async function extractMessages(options, builderName, context, extractorConstruct const buildOptions = (await context.validateOptions(await context.getTargetOptions(options.browserTarget), builderName)); buildOptions.optimization = false; buildOptions.sourceMap = { scripts: true, vendor: true }; + buildOptions.localize = false; let build; if (builderName === '@angular-devkit/build-angular:application') { build = application_1.buildApplicationInternal; @@ -134,4 +135,4 @@ function setupLocalizeExtractor(extractorConstructor, files, context) { }); return extractor; } -//# 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"]} \ No newline at end of file +//# 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  buildOptions.localize = false;\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"]} \ No newline at end of file diff --git a/src/tools/esbuild/application-code-bundle.js b/src/tools/esbuild/application-code-bundle.js index 7eb15a5e..f797dfbb 100644 --- a/src/tools/esbuild/application-code-bundle.js +++ b/src/tools/esbuild/application-code-bundle.js @@ -12,11 +12,13 @@ var __importDefault = (this && this.__importDefault) || function (mod) { Object.defineProperty(exports, "__esModule", { value: true }); exports.createServerCodeBundleOptions = exports.createBrowserCodeBundleOptions = void 0; const node_assert_1 = __importDefault(require("node:assert")); +const node_crypto_1 = require("node:crypto"); const promises_1 = require("node:fs/promises"); const node_path_1 = require("node:path"); const environment_options_1 = require("../../utils/environment-options"); const compiler_plugin_1 = require("./angular/compiler-plugin"); const compiler_plugin_options_1 = require("./compiler-plugin-options"); +const i18n_locale_plugin_1 = require("./i18n-locale-plugin"); const rxjs_esm_resolution_plugin_1 = require("./rxjs-esm-resolution-plugin"); const sourcemap_ignorelist_plugin_1 = require("./sourcemap-ignorelist-plugin"); const utils_1 = require("./utils"); @@ -49,10 +51,42 @@ function createBrowserCodeBundleOptions(options, target, sourceFileCache) { buildOptions.packages = 'external'; } const polyfills = options.polyfills ? [...options.polyfills] : []; + // Angular JIT mode requires the runtime compiler if (jit) { polyfills.push('@angular/compiler'); } - if (polyfills?.length) { + // Add Angular's global locale data if i18n options are present. + // Locale data should go first so that project provided polyfill code can augment if needed. + let needLocaleDataPlugin = false; + if (options.i18nOptions.shouldInline) { + // When inlining, a placeholder is used to allow the post-processing step to inject the $localize locale identifier + polyfills.unshift('angular:locale/placeholder'); + buildOptions.plugins?.unshift((0, virtual_module_plugin_1.createVirtualModulePlugin)({ + namespace: 'angular:locale/placeholder', + entryPointOnly: false, + loadContent: () => ({ + contents: `(globalThis.$localize ??= {}).locale = "___NG_LOCALE_INSERT___";\n`, + loader: 'js', + resolveDir: workspaceRoot, + }), + })); + // Add locale data for all active locales + // TODO: Inject each individually within the inlining process itself + for (const locale of options.i18nOptions.inlineLocales) { + polyfills.unshift(`angular:locale/data:${locale}`); + } + needLocaleDataPlugin = true; + } + else if (options.i18nOptions.hasDefinedSourceLocale) { + // When not inlining and a source local is present, use the source locale data directly + polyfills.unshift(`angular:locale/data:${options.i18nOptions.sourceLocale}`); + needLocaleDataPlugin = true; + } + if (needLocaleDataPlugin) { + buildOptions.plugins?.push((0, i18n_locale_plugin_1.createAngularLocaleDataPlugin)()); + } + // Add polyfill entry point if polyfills are present + if (polyfills.length) { const namespace = 'angular:polyfills'; buildOptions.entryPoints = { ...buildOptions.entryPoints, @@ -181,6 +215,16 @@ function createServerCodeBundleOptions(options, target, sourceFileCache) { exports.createServerCodeBundleOptions = createServerCodeBundleOptions; function getEsBuildCommonOptions(options) { const { workspaceRoot, outExtension, optimizationOptions, sourcemapOptions, tsconfig, externalDependencies, outputNames, preserveSymlinks, jit, } = options; + // Ensure unique hashes for i18n translation changes when using post-process inlining. + // This hash value is added as a footer to each file and ensures that the output file names (with hashes) + // change when translation files have changed. If this is not done the post processed files may have + // different content but would retain identical production file names which would lead to browser caching problems. + let footer; + if (options.i18nOptions.shouldInline) { + // Update file hashes to include translation file content + const i18nHash = Object.values(options.i18nOptions.locales).reduce((data, locale) => data + locale.files.map((file) => file.integrity || '').join('|'), ''); + footer = { js: `/**i18n:${(0, node_crypto_1.createHash)('sha256').update(i18nHash).digest('hex')}*/` }; + } return { absWorkingDir: workspaceRoot, bundle: true, @@ -211,6 +255,7 @@ function getEsBuildCommonOptions(options) { ...(optimizationOptions.scripts ? { 'ngDevMode': 'false' } : undefined), 'ngJitMode': jit ? 'true' : 'false', }, + footer, }; } -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application-code-bundle.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/application-code-bundle.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 { BuildOptions } from 'esbuild';\nimport assert from 'node:assert';\nimport { readFile } from 'node:fs/promises';\nimport { join, relative } from 'node:path';\nimport type { NormalizedApplicationBuildOptions } from '../../builders/application/options';\nimport { allowMangle } from '../../utils/environment-options';\nimport { SourceFileCache, createCompilerPlugin } from './angular/compiler-plugin';\nimport { createCompilerPluginOptions } from './compiler-plugin-options';\nimport { createRxjsEsmResolutionPlugin } from './rxjs-esm-resolution-plugin';\nimport { createSourcemapIgnorelistPlugin } from './sourcemap-ignorelist-plugin';\nimport { getFeatureSupport } from './utils';\nimport { createVirtualModulePlugin } from './virtual-module-plugin';\n\nexport function createBrowserCodeBundleOptions(\n  options: NormalizedApplicationBuildOptions,\n  target: string[],\n  sourceFileCache?: SourceFileCache,\n): BuildOptions {\n  const { workspaceRoot, entryPoints, outputNames, jit } = options;\n\n  const { pluginOptions, styleOptions } = createCompilerPluginOptions(\n    options,\n    target,\n    sourceFileCache,\n  );\n\n  const buildOptions: BuildOptions = {\n    ...getEsBuildCommonOptions(options),\n    platform: 'browser',\n    // Note: `es2015` is needed for RxJS v6. If not specified, `module` would\n    // match and the ES5 distribution would be bundled and ends up breaking at\n    // runtime with the RxJS testing library.\n    // More details: https://github.com/angular/angular-cli/issues/25405.\n    mainFields: ['es2020', 'es2015', 'browser', 'module', 'main'],\n    entryNames: outputNames.bundles,\n    entryPoints,\n    target,\n    supported: getFeatureSupport(target),\n    plugins: [\n      createSourcemapIgnorelistPlugin(),\n      createCompilerPlugin(\n        // JS/TS options\n        pluginOptions,\n        // Component stylesheet options\n        styleOptions,\n      ),\n    ],\n  };\n\n  if (options.externalPackages) {\n    buildOptions.packages = 'external';\n  }\n\n  const polyfills = options.polyfills ? [...options.polyfills] : [];\n  if (jit) {\n    polyfills.push('@angular/compiler');\n  }\n\n  if (polyfills?.length) {\n    const namespace = 'angular:polyfills';\n    buildOptions.entryPoints = {\n      ...buildOptions.entryPoints,\n      'polyfills': namespace,\n    };\n\n    buildOptions.plugins?.unshift(\n      createVirtualModulePlugin({\n        namespace,\n        loadContent: () => ({\n          contents: polyfills.map((file) => `import '${file.replace(/\\\\/g, '/')}';`).join('\\n'),\n          loader: 'js',\n          resolveDir: workspaceRoot,\n        }),\n      }),\n    );\n  }\n\n  return buildOptions;\n}\n\n/**\n * Create an esbuild 'build' options object for the server bundle.\n * @param options The builder's user-provider normalized options.\n * @returns An esbuild BuildOptions object.\n */\nexport function createServerCodeBundleOptions(\n  options: NormalizedApplicationBuildOptions,\n  target: string[],\n  sourceFileCache: SourceFileCache,\n): BuildOptions {\n  const { jit, serverEntryPoint, workspaceRoot, ssrOptions } = options;\n\n  assert(\n    serverEntryPoint,\n    'createServerCodeBundleOptions should not be called without a defined serverEntryPoint.',\n  );\n\n  const { pluginOptions, styleOptions } = createCompilerPluginOptions(\n    options,\n    target,\n    sourceFileCache,\n  );\n\n  const mainServerNamespace = 'angular:main-server';\n  const ssrEntryNamespace = 'angular:ssr-entry';\n\n  const entryPoints: Record<string, string> = {\n    'main.server': mainServerNamespace,\n  };\n\n  const ssrEntryPoint = ssrOptions?.entry;\n  if (ssrEntryPoint) {\n    entryPoints['server'] = ssrEntryNamespace;\n  }\n\n  const buildOptions: BuildOptions = {\n    ...getEsBuildCommonOptions(options),\n    platform: 'node',\n    // TODO: Invesigate why enabling `splitting` in JIT mode causes an \"'@angular/compiler' is not available\" error.\n    splitting: !jit,\n    outExtension: { '.js': '.mjs' },\n    // Note: `es2015` is needed for RxJS v6. If not specified, `module` would\n    // match and the ES5 distribution would be bundled and ends up breaking at\n    // runtime with the RxJS testing library.\n    // More details: https://github.com/angular/angular-cli/issues/25405.\n    mainFields: ['es2020', 'es2015', 'module', 'main'],\n    entryNames: '[name]',\n    target,\n    banner: {\n      // Note: Needed as esbuild does not provide require shims / proxy from ESModules.\n      // See: https://github.com/evanw/esbuild/issues/1921.\n      js: [\n        `import { createRequire } from 'node:module';`,\n        `globalThis['require'] ??= createRequire(import.meta.url);`,\n      ].join('\\n'),\n    },\n    entryPoints,\n    supported: getFeatureSupport(target),\n    plugins: [\n      createSourcemapIgnorelistPlugin(),\n      createCompilerPlugin(\n        // JS/TS options\n        { ...pluginOptions, noopTypeScriptCompilation: true },\n        // Component stylesheet options\n        styleOptions,\n      ),\n    ],\n  };\n\n  buildOptions.plugins ??= [];\n  if (options.externalPackages) {\n    buildOptions.packages = 'external';\n  } else {\n    buildOptions.plugins.push(createRxjsEsmResolutionPlugin());\n  }\n\n  const polyfills: string[] = [];\n  if (options.polyfills?.includes('zone.js')) {\n    polyfills.push(`import 'zone.js/node';`);\n  }\n\n  if (jit) {\n    polyfills.push(`import '@angular/compiler';`);\n  }\n\n  polyfills.push(`import '@angular/platform-server/init';`);\n\n  buildOptions.plugins.push(\n    createVirtualModulePlugin({\n      namespace: mainServerNamespace,\n      loadContent: async () => {\n        const mainServerEntryPoint = relative(workspaceRoot, serverEntryPoint).replace(/\\\\/g, '/');\n\n        const contents = [\n          ...polyfills,\n          `import moduleOrBootstrapFn from './${mainServerEntryPoint}';`,\n          `export default moduleOrBootstrapFn;`,\n          `export * from './${mainServerEntryPoint}';`,\n          `export { renderApplication, renderModule, ɵSERVER_CONTEXT } from '@angular/platform-server';`,\n        ];\n\n        if (options.prerenderOptions?.discoverRoutes) {\n          // We do not import it directly so that node.js modules are resolved using the correct context.\n          const routesExtractorCode = await readFile(\n            join(__dirname, '../../utils/routes-extractor/extractor.js'),\n            'utf-8',\n          );\n\n          // Remove source map URL comments from the code if a sourcemap is present as this will not match the file.\n          contents.push(routesExtractorCode.replace(/^\\/\\/# sourceMappingURL=[^\\r\\n]*/gm, ''));\n        }\n\n        return {\n          contents: contents.join('\\n'),\n          loader: 'js',\n          resolveDir: workspaceRoot,\n        };\n      },\n    }),\n  );\n\n  if (ssrEntryPoint) {\n    buildOptions.plugins.push(\n      createVirtualModulePlugin({\n        namespace: ssrEntryNamespace,\n        loadContent: () => {\n          const serverEntryPoint = relative(workspaceRoot, ssrEntryPoint).replace(/\\\\/g, '/');\n\n          return {\n            contents: [\n              ...polyfills,\n              `import './${serverEntryPoint}';`,\n              `export * from './${serverEntryPoint}';`,\n            ].join('\\n'),\n            loader: 'js',\n            resolveDir: workspaceRoot,\n          };\n        },\n      }),\n    );\n  }\n\n  return buildOptions;\n}\n\nfunction getEsBuildCommonOptions(options: NormalizedApplicationBuildOptions): BuildOptions {\n  const {\n    workspaceRoot,\n    outExtension,\n    optimizationOptions,\n    sourcemapOptions,\n    tsconfig,\n    externalDependencies,\n    outputNames,\n    preserveSymlinks,\n    jit,\n  } = options;\n\n  return {\n    absWorkingDir: workspaceRoot,\n    bundle: true,\n    format: 'esm',\n    assetNames: outputNames.media,\n    conditions: ['es2020', 'es2015', 'module'],\n    resolveExtensions: ['.ts', '.tsx', '.mjs', '.js'],\n    metafile: true,\n    legalComments: options.extractLicenses ? 'none' : 'eof',\n    logLevel: options.verbose ? 'debug' : 'silent',\n    minifyIdentifiers: optimizationOptions.scripts && allowMangle,\n    minifySyntax: optimizationOptions.scripts,\n    minifyWhitespace: optimizationOptions.scripts,\n    pure: ['forwardRef'],\n    outdir: workspaceRoot,\n    outExtension: outExtension ? { '.js': `.${outExtension}` } : undefined,\n    sourcemap: sourcemapOptions.scripts && (sourcemapOptions.hidden ? 'external' : true),\n    splitting: true,\n    chunkNames: 'chunk-[hash]',\n    tsconfig,\n    external: externalDependencies,\n    write: false,\n    preserveSymlinks,\n    define: {\n      // Only set to false when script optimizations are enabled. It should not be set to true because\n      // Angular turns `ngDevMode` into an object for development debugging purposes when not defined\n      // which a constant true value would break.\n      ...(optimizationOptions.scripts ? { 'ngDevMode': 'false' } : undefined),\n      'ngJitMode': jit ? 'true' : 'false',\n    },\n  };\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application-code-bundle.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/application-code-bundle.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 { BuildOptions } from 'esbuild';\nimport assert from 'node:assert';\nimport { createHash } from 'node:crypto';\nimport { readFile } from 'node:fs/promises';\nimport { join, relative } from 'node:path';\nimport type { NormalizedApplicationBuildOptions } from '../../builders/application/options';\nimport { allowMangle } from '../../utils/environment-options';\nimport { SourceFileCache, createCompilerPlugin } from './angular/compiler-plugin';\nimport { createCompilerPluginOptions } from './compiler-plugin-options';\nimport { createAngularLocaleDataPlugin } from './i18n-locale-plugin';\nimport { createRxjsEsmResolutionPlugin } from './rxjs-esm-resolution-plugin';\nimport { createSourcemapIgnorelistPlugin } from './sourcemap-ignorelist-plugin';\nimport { getFeatureSupport } from './utils';\nimport { createVirtualModulePlugin } from './virtual-module-plugin';\n\nexport function createBrowserCodeBundleOptions(\n  options: NormalizedApplicationBuildOptions,\n  target: string[],\n  sourceFileCache?: SourceFileCache,\n): BuildOptions {\n  const { workspaceRoot, entryPoints, outputNames, jit } = options;\n\n  const { pluginOptions, styleOptions } = createCompilerPluginOptions(\n    options,\n    target,\n    sourceFileCache,\n  );\n\n  const buildOptions: BuildOptions = {\n    ...getEsBuildCommonOptions(options),\n    platform: 'browser',\n    // Note: `es2015` is needed for RxJS v6. If not specified, `module` would\n    // match and the ES5 distribution would be bundled and ends up breaking at\n    // runtime with the RxJS testing library.\n    // More details: https://github.com/angular/angular-cli/issues/25405.\n    mainFields: ['es2020', 'es2015', 'browser', 'module', 'main'],\n    entryNames: outputNames.bundles,\n    entryPoints,\n    target,\n    supported: getFeatureSupport(target),\n    plugins: [\n      createSourcemapIgnorelistPlugin(),\n      createCompilerPlugin(\n        // JS/TS options\n        pluginOptions,\n        // Component stylesheet options\n        styleOptions,\n      ),\n    ],\n  };\n\n  if (options.externalPackages) {\n    buildOptions.packages = 'external';\n  }\n\n  const polyfills = options.polyfills ? [...options.polyfills] : [];\n\n  // Angular JIT mode requires the runtime compiler\n  if (jit) {\n    polyfills.push('@angular/compiler');\n  }\n\n  // Add Angular's global locale data if i18n options are present.\n  // Locale data should go first so that project provided polyfill code can augment if needed.\n  let needLocaleDataPlugin = false;\n  if (options.i18nOptions.shouldInline) {\n    // When inlining, a placeholder is used to allow the post-processing step to inject the $localize locale identifier\n    polyfills.unshift('angular:locale/placeholder');\n    buildOptions.plugins?.unshift(\n      createVirtualModulePlugin({\n        namespace: 'angular:locale/placeholder',\n        entryPointOnly: false,\n        loadContent: () => ({\n          contents: `(globalThis.$localize ??= {}).locale = \"___NG_LOCALE_INSERT___\";\\n`,\n          loader: 'js',\n          resolveDir: workspaceRoot,\n        }),\n      }),\n    );\n\n    // Add locale data for all active locales\n    // TODO: Inject each individually within the inlining process itself\n    for (const locale of options.i18nOptions.inlineLocales) {\n      polyfills.unshift(`angular:locale/data:${locale}`);\n    }\n    needLocaleDataPlugin = true;\n  } else if (options.i18nOptions.hasDefinedSourceLocale) {\n    // When not inlining and a source local is present, use the source locale data directly\n    polyfills.unshift(`angular:locale/data:${options.i18nOptions.sourceLocale}`);\n    needLocaleDataPlugin = true;\n  }\n  if (needLocaleDataPlugin) {\n    buildOptions.plugins?.push(createAngularLocaleDataPlugin());\n  }\n\n  // Add polyfill entry point if polyfills are present\n  if (polyfills.length) {\n    const namespace = 'angular:polyfills';\n    buildOptions.entryPoints = {\n      ...buildOptions.entryPoints,\n      'polyfills': namespace,\n    };\n\n    buildOptions.plugins?.unshift(\n      createVirtualModulePlugin({\n        namespace,\n        loadContent: () => ({\n          contents: polyfills.map((file) => `import '${file.replace(/\\\\/g, '/')}';`).join('\\n'),\n          loader: 'js',\n          resolveDir: workspaceRoot,\n        }),\n      }),\n    );\n  }\n\n  return buildOptions;\n}\n\n/**\n * Create an esbuild 'build' options object for the server bundle.\n * @param options The builder's user-provider normalized options.\n * @returns An esbuild BuildOptions object.\n */\nexport function createServerCodeBundleOptions(\n  options: NormalizedApplicationBuildOptions,\n  target: string[],\n  sourceFileCache: SourceFileCache,\n): BuildOptions {\n  const { jit, serverEntryPoint, workspaceRoot, ssrOptions } = options;\n\n  assert(\n    serverEntryPoint,\n    'createServerCodeBundleOptions should not be called without a defined serverEntryPoint.',\n  );\n\n  const { pluginOptions, styleOptions } = createCompilerPluginOptions(\n    options,\n    target,\n    sourceFileCache,\n  );\n\n  const mainServerNamespace = 'angular:main-server';\n  const ssrEntryNamespace = 'angular:ssr-entry';\n\n  const entryPoints: Record<string, string> = {\n    'main.server': mainServerNamespace,\n  };\n\n  const ssrEntryPoint = ssrOptions?.entry;\n  if (ssrEntryPoint) {\n    entryPoints['server'] = ssrEntryNamespace;\n  }\n\n  const buildOptions: BuildOptions = {\n    ...getEsBuildCommonOptions(options),\n    platform: 'node',\n    // TODO: Invesigate why enabling `splitting` in JIT mode causes an \"'@angular/compiler' is not available\" error.\n    splitting: !jit,\n    outExtension: { '.js': '.mjs' },\n    // Note: `es2015` is needed for RxJS v6. If not specified, `module` would\n    // match and the ES5 distribution would be bundled and ends up breaking at\n    // runtime with the RxJS testing library.\n    // More details: https://github.com/angular/angular-cli/issues/25405.\n    mainFields: ['es2020', 'es2015', 'module', 'main'],\n    entryNames: '[name]',\n    target,\n    banner: {\n      // Note: Needed as esbuild does not provide require shims / proxy from ESModules.\n      // See: https://github.com/evanw/esbuild/issues/1921.\n      js: [\n        `import { createRequire } from 'node:module';`,\n        `globalThis['require'] ??= createRequire(import.meta.url);`,\n      ].join('\\n'),\n    },\n    entryPoints,\n    supported: getFeatureSupport(target),\n    plugins: [\n      createSourcemapIgnorelistPlugin(),\n      createCompilerPlugin(\n        // JS/TS options\n        { ...pluginOptions, noopTypeScriptCompilation: true },\n        // Component stylesheet options\n        styleOptions,\n      ),\n    ],\n  };\n\n  buildOptions.plugins ??= [];\n  if (options.externalPackages) {\n    buildOptions.packages = 'external';\n  } else {\n    buildOptions.plugins.push(createRxjsEsmResolutionPlugin());\n  }\n\n  const polyfills: string[] = [];\n  if (options.polyfills?.includes('zone.js')) {\n    polyfills.push(`import 'zone.js/node';`);\n  }\n\n  if (jit) {\n    polyfills.push(`import '@angular/compiler';`);\n  }\n\n  polyfills.push(`import '@angular/platform-server/init';`);\n\n  buildOptions.plugins.push(\n    createVirtualModulePlugin({\n      namespace: mainServerNamespace,\n      loadContent: async () => {\n        const mainServerEntryPoint = relative(workspaceRoot, serverEntryPoint).replace(/\\\\/g, '/');\n\n        const contents = [\n          ...polyfills,\n          `import moduleOrBootstrapFn from './${mainServerEntryPoint}';`,\n          `export default moduleOrBootstrapFn;`,\n          `export * from './${mainServerEntryPoint}';`,\n          `export { renderApplication, renderModule, ɵSERVER_CONTEXT } from '@angular/platform-server';`,\n        ];\n\n        if (options.prerenderOptions?.discoverRoutes) {\n          // We do not import it directly so that node.js modules are resolved using the correct context.\n          const routesExtractorCode = await readFile(\n            join(__dirname, '../../utils/routes-extractor/extractor.js'),\n            'utf-8',\n          );\n\n          // Remove source map URL comments from the code if a sourcemap is present as this will not match the file.\n          contents.push(routesExtractorCode.replace(/^\\/\\/# sourceMappingURL=[^\\r\\n]*/gm, ''));\n        }\n\n        return {\n          contents: contents.join('\\n'),\n          loader: 'js',\n          resolveDir: workspaceRoot,\n        };\n      },\n    }),\n  );\n\n  if (ssrEntryPoint) {\n    buildOptions.plugins.push(\n      createVirtualModulePlugin({\n        namespace: ssrEntryNamespace,\n        loadContent: () => {\n          const serverEntryPoint = relative(workspaceRoot, ssrEntryPoint).replace(/\\\\/g, '/');\n\n          return {\n            contents: [\n              ...polyfills,\n              `import './${serverEntryPoint}';`,\n              `export * from './${serverEntryPoint}';`,\n            ].join('\\n'),\n            loader: 'js',\n            resolveDir: workspaceRoot,\n          };\n        },\n      }),\n    );\n  }\n\n  return buildOptions;\n}\n\nfunction getEsBuildCommonOptions(options: NormalizedApplicationBuildOptions): BuildOptions {\n  const {\n    workspaceRoot,\n    outExtension,\n    optimizationOptions,\n    sourcemapOptions,\n    tsconfig,\n    externalDependencies,\n    outputNames,\n    preserveSymlinks,\n    jit,\n  } = options;\n\n  // Ensure unique hashes for i18n translation changes when using post-process inlining.\n  // This hash value is added as a footer to each file and ensures that the output file names (with hashes)\n  // change when translation files have changed. If this is not done the post processed files may have\n  // different content but would retain identical production file names which would lead to browser caching problems.\n  let footer;\n  if (options.i18nOptions.shouldInline) {\n    // Update file hashes to include translation file content\n    const i18nHash = Object.values(options.i18nOptions.locales).reduce(\n      (data, locale) => data + locale.files.map((file) => file.integrity || '').join('|'),\n      '',\n    );\n\n    footer = { js: `/**i18n:${createHash('sha256').update(i18nHash).digest('hex')}*/` };\n  }\n\n  return {\n    absWorkingDir: workspaceRoot,\n    bundle: true,\n    format: 'esm',\n    assetNames: outputNames.media,\n    conditions: ['es2020', 'es2015', 'module'],\n    resolveExtensions: ['.ts', '.tsx', '.mjs', '.js'],\n    metafile: true,\n    legalComments: options.extractLicenses ? 'none' : 'eof',\n    logLevel: options.verbose ? 'debug' : 'silent',\n    minifyIdentifiers: optimizationOptions.scripts && allowMangle,\n    minifySyntax: optimizationOptions.scripts,\n    minifyWhitespace: optimizationOptions.scripts,\n    pure: ['forwardRef'],\n    outdir: workspaceRoot,\n    outExtension: outExtension ? { '.js': `.${outExtension}` } : undefined,\n    sourcemap: sourcemapOptions.scripts && (sourcemapOptions.hidden ? 'external' : true),\n    splitting: true,\n    chunkNames: 'chunk-[hash]',\n    tsconfig,\n    external: externalDependencies,\n    write: false,\n    preserveSymlinks,\n    define: {\n      // Only set to false when script optimizations are enabled. It should not be set to true because\n      // Angular turns `ngDevMode` into an object for development debugging purposes when not defined\n      // which a constant true value would break.\n      ...(optimizationOptions.scripts ? { 'ngDevMode': 'false' } : undefined),\n      'ngJitMode': jit ? 'true' : 'false',\n    },\n    footer,\n  };\n}\n"]} \ No newline at end of file diff --git a/src/tools/esbuild/bundler-execution-result.d.ts b/src/tools/esbuild/bundler-execution-result.d.ts index 137e11fb..78d20a08 100644 --- a/src/tools/esbuild/bundler-execution-result.d.ts +++ b/src/tools/esbuild/bundler-execution-result.d.ts @@ -20,8 +20,8 @@ export interface RebuildState { export declare class ExecutionResult { private rebuildContexts; private codeBundleCache?; - readonly outputFiles: OutputFile[]; - readonly assetFiles: { + outputFiles: OutputFile[]; + assetFiles: { source: string; destination: string; }[]; diff --git a/src/tools/esbuild/bundler-execution-result.js b/src/tools/esbuild/bundler-execution-result.js index 3d4fcc30..739fe158 100644 --- a/src/tools/esbuild/bundler-execution-result.js +++ b/src/tools/esbuild/bundler-execution-result.js @@ -56,4 +56,4 @@ class ExecutionResult { } } exports.ExecutionResult = ExecutionResult; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVuZGxlci1leGVjdXRpb24tcmVzdWx0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC9idW5kbGVyLWV4ZWN1dGlvbi1yZXN1bHQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBTUgsbUNBQW1EO0FBUW5EOztHQUVHO0FBQ0gsTUFBYSxlQUFlO0lBS2hCO0lBQ0E7SUFMRCxXQUFXLEdBQWlCLEVBQUUsQ0FBQztJQUMvQixVQUFVLEdBQThDLEVBQUUsQ0FBQztJQUVwRSxZQUNVLGVBQWlDLEVBQ2pDLGVBQWlDO1FBRGpDLG9CQUFlLEdBQWYsZUFBZSxDQUFrQjtRQUNqQyxvQkFBZSxHQUFmLGVBQWUsQ0FBa0I7SUFDeEMsQ0FBQztJQUVKLGFBQWEsQ0FBQyxJQUFZLEVBQUUsT0FBZTtRQUN6QyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxJQUFBLGdDQUF3QixFQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUFDO0lBQ2pFLENBQUM7SUFFRCxJQUFJLE1BQU07UUFDUixPQUFPO1lBQ0wsT0FBTyxFQUFFLElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxHQUFHLENBQUM7U0FDckMsQ0FBQztJQUNKLENBQUM7SUFFRCxJQUFJLGVBQWU7UUFDakIsT0FBTztZQUNMLE9BQU8sRUFBRSxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sR0FBRyxDQUFDO1lBQ3BDLFdBQVcsRUFBRSxJQUFJLENBQUMsV0FBVztZQUM3QixVQUFVLEVBQUUsSUFBSSxDQUFDLFVBQVU7U0FDNUIsQ0FBQztJQUNKLENBQUM7SUFFRCxJQUFJLFVBQVU7UUFDWixNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsZUFBZSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQyxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDO1FBQ2pGLElBQUksSUFBSSxDQUFDLGVBQWUsRUFBRSxlQUFlLEVBQUU7WUFDekMsS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsZUFBZSxDQUFDLENBQUM7U0FDckQ7UUFFRCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxXQUF5QjtRQUMxQyxJQUFJLENBQUMsZUFBZSxFQUFFLFVBQVUsQ0FBQyxDQUFDLEdBQUcsV0FBVyxDQUFDLFFBQVEsRUFBRSxHQUFHLFdBQVcsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1FBRXBGLE9BQU87WUFDTCxlQUFlLEVBQUUsSUFBSSxDQUFDLGVBQWU7WUFDckMsZUFBZSxFQUFFLElBQUksQ0FBQyxlQUFlO1lBQ3JDLFdBQVc7U0FDWixDQUFDO0lBQ0osQ0FBQztJQUVELEtBQUssQ0FBQyxPQUFPO1FBQ1gsTUFBTSxPQUFPLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQ3JGLENBQUM7Q0FDRjtBQWpERCwwQ0FpREMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHsgT3V0cHV0RmlsZSB9IGZyb20gJ2VzYnVpbGQnO1xuaW1wb3J0IHR5cGUgeyBDaGFuZ2VkRmlsZXMgfSBmcm9tICcuLi8uLi90b29scy9lc2J1aWxkL3dhdGNoZXInO1xuaW1wb3J0IHR5cGUgeyBTb3VyY2VGaWxlQ2FjaGUgfSBmcm9tICcuL2FuZ3VsYXIvY29tcGlsZXItcGx1Z2luJztcbmltcG9ydCB0eXBlIHsgQnVuZGxlckNvbnRleHQgfSBmcm9tICcuL2J1bmRsZXItY29udGV4dCc7XG5pbXBvcnQgeyBjcmVhdGVPdXRwdXRGaWxlRnJvbVRleHQgfSBmcm9tICcuL3V0aWxzJztcblxuZXhwb3J0IGludGVyZmFjZSBSZWJ1aWxkU3RhdGUge1xuICByZWJ1aWxkQ29udGV4dHM6IEJ1bmRsZXJDb250ZXh0W107XG4gIGNvZGVCdW5kbGVDYWNoZT86IFNvdXJjZUZpbGVDYWNoZTtcbiAgZmlsZUNoYW5nZXM6IENoYW5nZWRGaWxlcztcbn1cblxuLyoqXG4gKiBSZXByZXNlbnRzIHRoZSByZXN1bHQgb2YgYSBzaW5nbGUgYnVpbGRlciBleGVjdXRlIGNhbGwuXG4gKi9cbmV4cG9ydCBjbGFzcyBFeGVjdXRpb25SZXN1bHQge1xuICByZWFkb25seSBvdXRwdXRGaWxlczogT3V0cHV0RmlsZVtdID0gW107XG4gIHJlYWRvbmx5IGFzc2V0RmlsZXM6IHsgc291cmNlOiBzdHJpbmc7IGRlc3RpbmF0aW9uOiBzdHJpbmcgfVtdID0gW107XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWJ1aWxkQ29udGV4dHM6IEJ1bmRsZXJDb250ZXh0W10sXG4gICAgcHJpdmF0ZSBjb2RlQnVuZGxlQ2FjaGU/OiBTb3VyY2VGaWxlQ2FjaGUsXG4gICkge31cblxuICBhZGRPdXRwdXRGaWxlKHBhdGg6IHN0cmluZywgY29udGVudDogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5vdXRwdXRGaWxlcy5wdXNoKGNyZWF0ZU91dHB1dEZpbGVGcm9tVGV4dChwYXRoLCBjb250ZW50KSk7XG4gIH1cblxuICBnZXQgb3V0cHV0KCkge1xuICAgIHJldHVybiB7XG4gICAgICBzdWNjZXNzOiB0aGlzLm91dHB1dEZpbGVzLmxlbmd0aCA+IDAsXG4gICAgfTtcbiAgfVxuXG4gIGdldCBvdXRwdXRXaXRoRmlsZXMoKSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHN1Y2Nlc3M6IHRoaXMub3V0cHV0RmlsZXMubGVuZ3RoID4gMCxcbiAgICAgIG91dHB1dEZpbGVzOiB0aGlzLm91dHB1dEZpbGVzLFxuICAgICAgYXNzZXRGaWxlczogdGhpcy5hc3NldEZpbGVzLFxuICAgIH07XG4gIH1cblxuICBnZXQgd2F0Y2hGaWxlcygpIHtcbiAgICBjb25zdCBmaWxlcyA9IHRoaXMucmVidWlsZENvbnRleHRzLmZsYXRNYXAoKGNvbnRleHQpID0+IFsuLi5jb250ZXh0LndhdGNoRmlsZXNdKTtcbiAgICBpZiAodGhpcy5jb2RlQnVuZGxlQ2FjaGU/LnJlZmVyZW5jZWRGaWxlcykge1xuICAgICAgZmlsZXMucHVzaCguLi50aGlzLmNvZGVCdW5kbGVDYWNoZS5yZWZlcmVuY2VkRmlsZXMpO1xuICAgIH1cblxuICAgIHJldHVybiBmaWxlcztcbiAgfVxuXG4gIGNyZWF0ZVJlYnVpbGRTdGF0ZShmaWxlQ2hhbmdlczogQ2hhbmdlZEZpbGVzKTogUmVidWlsZFN0YXRlIHtcbiAgICB0aGlzLmNvZGVCdW5kbGVDYWNoZT8uaW52YWxpZGF0ZShbLi4uZmlsZUNoYW5nZXMubW9kaWZpZWQsIC4uLmZpbGVDaGFuZ2VzLnJlbW92ZWRdKTtcblxuICAgIHJldHVybiB7XG4gICAgICByZWJ1aWxkQ29udGV4dHM6IHRoaXMucmVidWlsZENvbnRleHRzLFxuICAgICAgY29kZUJ1bmRsZUNhY2hlOiB0aGlzLmNvZGVCdW5kbGVDYWNoZSxcbiAgICAgIGZpbGVDaGFuZ2VzLFxuICAgIH07XG4gIH1cblxuICBhc3luYyBkaXNwb3NlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZCh0aGlzLnJlYnVpbGRDb250ZXh0cy5tYXAoKGNvbnRleHQpID0+IGNvbnRleHQuZGlzcG9zZSgpKSk7XG4gIH1cbn1cbiJdfQ== \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVuZGxlci1leGVjdXRpb24tcmVzdWx0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC9idW5kbGVyLWV4ZWN1dGlvbi1yZXN1bHQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBTUgsbUNBQW1EO0FBUW5EOztHQUVHO0FBQ0gsTUFBYSxlQUFlO0lBS2hCO0lBQ0E7SUFMVixXQUFXLEdBQWlCLEVBQUUsQ0FBQztJQUMvQixVQUFVLEdBQThDLEVBQUUsQ0FBQztJQUUzRCxZQUNVLGVBQWlDLEVBQ2pDLGVBQWlDO1FBRGpDLG9CQUFlLEdBQWYsZUFBZSxDQUFrQjtRQUNqQyxvQkFBZSxHQUFmLGVBQWUsQ0FBa0I7SUFDeEMsQ0FBQztJQUVKLGFBQWEsQ0FBQyxJQUFZLEVBQUUsT0FBZTtRQUN6QyxJQUFJLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxJQUFBLGdDQUF3QixFQUFDLElBQUksRUFBRSxPQUFPLENBQUMsQ0FBQyxDQUFDO0lBQ2pFLENBQUM7SUFFRCxJQUFJLE1BQU07UUFDUixPQUFPO1lBQ0wsT0FBTyxFQUFFLElBQUksQ0FBQyxXQUFXLENBQUMsTUFBTSxHQUFHLENBQUM7U0FDckMsQ0FBQztJQUNKLENBQUM7SUFFRCxJQUFJLGVBQWU7UUFDakIsT0FBTztZQUNMLE9BQU8sRUFBRSxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sR0FBRyxDQUFDO1lBQ3BDLFdBQVcsRUFBRSxJQUFJLENBQUMsV0FBVztZQUM3QixVQUFVLEVBQUUsSUFBSSxDQUFDLFVBQVU7U0FDNUIsQ0FBQztJQUNKLENBQUM7SUFFRCxJQUFJLFVBQVU7UUFDWixNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsZUFBZSxDQUFDLE9BQU8sQ0FBQyxDQUFDLE9BQU8sRUFBRSxFQUFFLENBQUMsQ0FBQyxHQUFHLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDO1FBQ2pGLElBQUksSUFBSSxDQUFDLGVBQWUsRUFBRSxlQUFlLEVBQUU7WUFDekMsS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLElBQUksQ0FBQyxlQUFlLENBQUMsZUFBZSxDQUFDLENBQUM7U0FDckQ7UUFFRCxPQUFPLEtBQUssQ0FBQztJQUNmLENBQUM7SUFFRCxrQkFBa0IsQ0FBQyxXQUF5QjtRQUMxQyxJQUFJLENBQUMsZUFBZSxFQUFFLFVBQVUsQ0FBQyxDQUFDLEdBQUcsV0FBVyxDQUFDLFFBQVEsRUFBRSxHQUFHLFdBQVcsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO1FBRXBGLE9BQU87WUFDTCxlQUFlLEVBQUUsSUFBSSxDQUFDLGVBQWU7WUFDckMsZUFBZSxFQUFFLElBQUksQ0FBQyxlQUFlO1lBQ3JDLFdBQVc7U0FDWixDQUFDO0lBQ0osQ0FBQztJQUVELEtBQUssQ0FBQyxPQUFPO1FBQ1gsTUFBTSxPQUFPLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQ3JGLENBQUM7Q0FDRjtBQWpERCwwQ0FpREMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHsgT3V0cHV0RmlsZSB9IGZyb20gJ2VzYnVpbGQnO1xuaW1wb3J0IHR5cGUgeyBDaGFuZ2VkRmlsZXMgfSBmcm9tICcuLi8uLi90b29scy9lc2J1aWxkL3dhdGNoZXInO1xuaW1wb3J0IHR5cGUgeyBTb3VyY2VGaWxlQ2FjaGUgfSBmcm9tICcuL2FuZ3VsYXIvY29tcGlsZXItcGx1Z2luJztcbmltcG9ydCB0eXBlIHsgQnVuZGxlckNvbnRleHQgfSBmcm9tICcuL2J1bmRsZXItY29udGV4dCc7XG5pbXBvcnQgeyBjcmVhdGVPdXRwdXRGaWxlRnJvbVRleHQgfSBmcm9tICcuL3V0aWxzJztcblxuZXhwb3J0IGludGVyZmFjZSBSZWJ1aWxkU3RhdGUge1xuICByZWJ1aWxkQ29udGV4dHM6IEJ1bmRsZXJDb250ZXh0W107XG4gIGNvZGVCdW5kbGVDYWNoZT86IFNvdXJjZUZpbGVDYWNoZTtcbiAgZmlsZUNoYW5nZXM6IENoYW5nZWRGaWxlcztcbn1cblxuLyoqXG4gKiBSZXByZXNlbnRzIHRoZSByZXN1bHQgb2YgYSBzaW5nbGUgYnVpbGRlciBleGVjdXRlIGNhbGwuXG4gKi9cbmV4cG9ydCBjbGFzcyBFeGVjdXRpb25SZXN1bHQge1xuICBvdXRwdXRGaWxlczogT3V0cHV0RmlsZVtdID0gW107XG4gIGFzc2V0RmlsZXM6IHsgc291cmNlOiBzdHJpbmc7IGRlc3RpbmF0aW9uOiBzdHJpbmcgfVtdID0gW107XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWJ1aWxkQ29udGV4dHM6IEJ1bmRsZXJDb250ZXh0W10sXG4gICAgcHJpdmF0ZSBjb2RlQnVuZGxlQ2FjaGU/OiBTb3VyY2VGaWxlQ2FjaGUsXG4gICkge31cblxuICBhZGRPdXRwdXRGaWxlKHBhdGg6IHN0cmluZywgY29udGVudDogc3RyaW5nKTogdm9pZCB7XG4gICAgdGhpcy5vdXRwdXRGaWxlcy5wdXNoKGNyZWF0ZU91dHB1dEZpbGVGcm9tVGV4dChwYXRoLCBjb250ZW50KSk7XG4gIH1cblxuICBnZXQgb3V0cHV0KCkge1xuICAgIHJldHVybiB7XG4gICAgICBzdWNjZXNzOiB0aGlzLm91dHB1dEZpbGVzLmxlbmd0aCA+IDAsXG4gICAgfTtcbiAgfVxuXG4gIGdldCBvdXRwdXRXaXRoRmlsZXMoKSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHN1Y2Nlc3M6IHRoaXMub3V0cHV0RmlsZXMubGVuZ3RoID4gMCxcbiAgICAgIG91dHB1dEZpbGVzOiB0aGlzLm91dHB1dEZpbGVzLFxuICAgICAgYXNzZXRGaWxlczogdGhpcy5hc3NldEZpbGVzLFxuICAgIH07XG4gIH1cblxuICBnZXQgd2F0Y2hGaWxlcygpIHtcbiAgICBjb25zdCBmaWxlcyA9IHRoaXMucmVidWlsZENvbnRleHRzLmZsYXRNYXAoKGNvbnRleHQpID0+IFsuLi5jb250ZXh0LndhdGNoRmlsZXNdKTtcbiAgICBpZiAodGhpcy5jb2RlQnVuZGxlQ2FjaGU/LnJlZmVyZW5jZWRGaWxlcykge1xuICAgICAgZmlsZXMucHVzaCguLi50aGlzLmNvZGVCdW5kbGVDYWNoZS5yZWZlcmVuY2VkRmlsZXMpO1xuICAgIH1cblxuICAgIHJldHVybiBmaWxlcztcbiAgfVxuXG4gIGNyZWF0ZVJlYnVpbGRTdGF0ZShmaWxlQ2hhbmdlczogQ2hhbmdlZEZpbGVzKTogUmVidWlsZFN0YXRlIHtcbiAgICB0aGlzLmNvZGVCdW5kbGVDYWNoZT8uaW52YWxpZGF0ZShbLi4uZmlsZUNoYW5nZXMubW9kaWZpZWQsIC4uLmZpbGVDaGFuZ2VzLnJlbW92ZWRdKTtcblxuICAgIHJldHVybiB7XG4gICAgICByZWJ1aWxkQ29udGV4dHM6IHRoaXMucmVidWlsZENvbnRleHRzLFxuICAgICAgY29kZUJ1bmRsZUNhY2hlOiB0aGlzLmNvZGVCdW5kbGVDYWNoZSxcbiAgICAgIGZpbGVDaGFuZ2VzLFxuICAgIH07XG4gIH1cblxuICBhc3luYyBkaXNwb3NlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGF3YWl0IFByb21pc2UuYWxsU2V0dGxlZCh0aGlzLnJlYnVpbGRDb250ZXh0cy5tYXAoKGNvbnRleHQpID0+IGNvbnRleHQuZGlzcG9zZSgpKSk7XG4gIH1cbn1cbiJdfQ== \ No newline at end of file diff --git a/src/tools/esbuild/i18n-inliner-worker.d.ts b/src/tools/esbuild/i18n-inliner-worker.d.ts new file mode 100644 index 00000000..4a63d427 --- /dev/null +++ b/src/tools/esbuild/i18n-inliner-worker.d.ts @@ -0,0 +1,37 @@ +/** + * @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 + */ +/** + * The options passed to the inliner for each file request + */ +interface InlineRequest { + /** + * The filename that should be processed. The data for the file is provided to the Worker + * during Worker initialization. + */ + filename: string; + /** + * The locale specifier that should be used during the inlining process of the file. + */ + locale: string; + /** + * The translation messages for the locale that should be used during the inlining process of the file. + */ + translation?: Record; +} +/** + * Inlines the provided locale and translation into a JavaScript file that contains `$localize` usage. + * This function is the main entry for the Worker's action that is called by the worker pool. + * + * @param request An InlineRequest object representing the options for inlining + * @returns An array containing the inlined file and optional map content. + */ +export default function inlineLocale(request: InlineRequest): Promise<{ + file: string; + contents: string; +}[]>; +export {}; diff --git a/src/tools/esbuild/i18n-inliner-worker.js b/src/tools/esbuild/i18n-inliner-worker.js new file mode 100644 index 00000000..589fbbd9 --- /dev/null +++ b/src/tools/esbuild/i18n-inliner-worker.js @@ -0,0 +1,138 @@ +"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 __importDefault = (this && this.__importDefault) || function (mod) { + return (mod && mod.__esModule) ? mod : { "default": mod }; +}; +Object.defineProperty(exports, "__esModule", { value: true }); +const remapping_1 = __importDefault(require("@ampproject/remapping")); +const core_1 = require("@babel/core"); +const node_assert_1 = __importDefault(require("node:assert")); +const node_worker_threads_1 = require("node:worker_threads"); +const error_1 = require("../../utils/error"); +const load_esm_1 = require("../../utils/load-esm"); +// Extract the application files and common options used for inline requests from the Worker context +// TODO: Evaluate overall performance difference of passing translations here as well +const { files, missingTranslation, shouldOptimize } = (node_worker_threads_1.workerData || {}); +/** + * Inlines the provided locale and translation into a JavaScript file that contains `$localize` usage. + * This function is the main entry for the Worker's action that is called by the worker pool. + * + * @param request An InlineRequest object representing the options for inlining + * @returns An array containing the inlined file and optional map content. + */ +async function inlineLocale(request) { + const data = files.get(request.filename); + (0, node_assert_1.default)(data !== undefined, `Invalid inline request for file '${request.filename}'.`); + const code = await data.text(); + const map = await files.get(request.filename + '.map')?.text(); + const result = await transformWithBabel(code, map && JSON.parse(map), request); + // TODO: Return diagnostics + // TODO: Consider buffer transfer instead of string copying + const response = [{ file: request.filename, contents: result.code }]; + if (result.map) { + response.push({ file: request.filename + '.map', contents: result.map }); + } + return response; +} +exports.default = inlineLocale; +/** + * Cached instance of the `@angular/localize/tools` module. + * This is used to remove the need to repeatedly import the module per file translation. + */ +let localizeToolsModule; +/** + * Attempts to load the `@angular/localize/tools` module containing the functionality to + * perform the file translations. + * This module must be dynamically loaded as it is an ESM module and this file is CommonJS. + */ +async function loadLocalizeTools() { + // Load ESM `@angular/localize/tools` using the TypeScript dynamic import workaround. + // Once TypeScript provides support for keeping the dynamic import this workaround can be + // changed to a direct dynamic import. + localizeToolsModule ??= await (0, load_esm_1.loadEsmModule)('@angular/localize/tools'); + return localizeToolsModule; +} +/** + * Creates the needed Babel plugins to inline a given locale and translation for a JavaScript file. + * @param locale A string containing the locale specifier to use. + * @param translation A object record containing locale specific messages to use. + * @returns An array of Babel plugins. + */ +async function createI18nPlugins(locale, translation) { + const { Diagnostics, makeEs2015TranslatePlugin } = await loadLocalizeTools(); + const plugins = []; + const diagnostics = new Diagnostics(); + plugins.push( + // eslint-disable-next-line @typescript-eslint/no-explicit-any + makeEs2015TranslatePlugin(diagnostics, (translation || {}), { + missingTranslation: translation === undefined ? 'ignore' : missingTranslation, + })); + // Create a plugin to replace the locale specifier constant inject by the build system with the actual specifier + plugins.push({ + visitor: { + StringLiteral(path) { + if (path.node.value === '___NG_LOCALE_INSERT___') { + path.replaceWith(core_1.types.stringLiteral(locale)); + } + }, + }, + }); + return { diagnostics, plugins }; +} +/** + * Transforms a JavaScript file using Babel to inline the request locale and translation. + * @param code A string containing the JavaScript code to transform. + * @param map A sourcemap object for the provided JavaScript code. + * @param options The inline request options to use. + * @returns An object containing the code, map, and diagnostics from the transformation. + */ +async function transformWithBabel(code, map, options) { + let ast; + try { + ast = (0, core_1.parseSync)(code, { + babelrc: false, + configFile: false, + sourceType: 'unambiguous', + filename: options.filename, + }); + } + catch (error) { + (0, error_1.assertIsError)(error); + // Make the error more readable. + // Same errors will contain the full content of the file as the error message + // Which makes it hard to find the actual error message. + const index = error.message.indexOf(')\n'); + const msg = index !== -1 ? error.message.slice(0, index + 1) : error.message; + throw new Error(`${msg}\nAn error occurred inlining file "${options.filename}"`); + } + if (!ast) { + throw new Error(`Unknown error occurred inlining file "${options.filename}"`); + } + const { diagnostics, plugins } = await createI18nPlugins(options.locale, options.translation); + const transformResult = await (0, core_1.transformFromAstAsync)(ast, code, { + filename: options.filename, + // false is a valid value but not included in the type definition + inputSourceMap: false, + sourceMaps: !!map, + compact: shouldOptimize, + configFile: false, + babelrc: false, + browserslistConfigFile: false, + plugins, + }); + if (!transformResult || !transformResult.code) { + throw new Error(`Unknown error occurred processing bundle for "${options.filename}".`); + } + let outputMap; + if (map && transformResult.map) { + outputMap = (0, remapping_1.default)([transformResult.map, map], () => null); + } + return { code: transformResult.code, map: outputMap && JSON.stringify(outputMap), diagnostics }; +} +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n-inliner-worker.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/i18n-inliner-worker.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 remapping, { SourceMapInput } from '@ampproject/remapping';\nimport { PluginObj, parseSync, transformFromAstAsync, types } from '@babel/core';\nimport assert from 'node:assert';\nimport { workerData } from 'node:worker_threads';\nimport { assertIsError } from '../../utils/error';\nimport { loadEsmModule } from '../../utils/load-esm';\n\n/**\n * The options passed to the inliner for each file request\n */\ninterface InlineRequest {\n  /**\n   * The filename that should be processed. The data for the file is provided to the Worker\n   * during Worker initialization.\n   */\n  filename: string;\n  /**\n   * The locale specifier that should be used during the inlining process of the file.\n   */\n  locale: string;\n  /**\n   * The translation messages for the locale that should be used during the inlining process of the file.\n   */\n  translation?: Record<string, unknown>;\n}\n\n// Extract the application files and common options used for inline requests from the Worker context\n// TODO: Evaluate overall performance difference of passing translations here as well\nconst { files, missingTranslation, shouldOptimize } = (workerData || {}) as {\n  files: ReadonlyMap<string, Blob>;\n  missingTranslation: 'error' | 'warning' | 'ignore';\n  shouldOptimize: boolean;\n};\n\n/**\n * Inlines the provided locale and translation into a JavaScript file that contains `$localize` usage.\n * This function is the main entry for the Worker's action that is called by the worker pool.\n *\n * @param request An InlineRequest object representing the options for inlining\n * @returns An array containing the inlined file and optional map content.\n */\nexport default async function inlineLocale(request: InlineRequest) {\n  const data = files.get(request.filename);\n\n  assert(data !== undefined, `Invalid inline request for file '${request.filename}'.`);\n\n  const code = await data.text();\n  const map = await files.get(request.filename + '.map')?.text();\n  const result = await transformWithBabel(\n    code,\n    map && (JSON.parse(map) as SourceMapInput),\n    request,\n  );\n\n  // TODO: Return diagnostics\n  // TODO: Consider buffer transfer instead of string copying\n  const response = [{ file: request.filename, contents: result.code }];\n  if (result.map) {\n    response.push({ file: request.filename + '.map', contents: result.map });\n  }\n\n  return response;\n}\n\n/**\n * A Type representing the localize tools module.\n */\ntype LocalizeUtilityModule = typeof import('@angular/localize/tools');\n\n/**\n * Cached instance of the `@angular/localize/tools` module.\n * This is used to remove the need to repeatedly import the module per file translation.\n */\nlet localizeToolsModule: LocalizeUtilityModule | undefined;\n\n/**\n * Attempts to load the `@angular/localize/tools` module containing the functionality to\n * perform the file translations.\n * This module must be dynamically loaded as it is an ESM module and this file is CommonJS.\n */\nasync function loadLocalizeTools(): Promise<LocalizeUtilityModule> {\n  // Load ESM `@angular/localize/tools` using the TypeScript dynamic import workaround.\n  // Once TypeScript provides support for keeping the dynamic import this workaround can be\n  // changed to a direct dynamic import.\n  localizeToolsModule ??= await loadEsmModule<LocalizeUtilityModule>('@angular/localize/tools');\n\n  return localizeToolsModule;\n}\n\n/**\n * Creates the needed Babel plugins to inline a given locale and translation for a JavaScript file.\n * @param locale A string containing the locale specifier to use.\n * @param translation A object record containing locale specific messages to use.\n * @returns An array of Babel plugins.\n */\nasync function createI18nPlugins(locale: string, translation: Record<string, unknown> | undefined) {\n  const { Diagnostics, makeEs2015TranslatePlugin } = await loadLocalizeTools();\n\n  const plugins: PluginObj[] = [];\n  const diagnostics = new Diagnostics();\n\n  plugins.push(\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    makeEs2015TranslatePlugin(diagnostics, (translation || {}) as any, {\n      missingTranslation: translation === undefined ? 'ignore' : missingTranslation,\n    }),\n  );\n\n  // Create a plugin to replace the locale specifier constant inject by the build system with the actual specifier\n  plugins.push({\n    visitor: {\n      StringLiteral(path) {\n        if (path.node.value === '___NG_LOCALE_INSERT___') {\n          path.replaceWith(types.stringLiteral(locale));\n        }\n      },\n    },\n  });\n\n  return { diagnostics, plugins };\n}\n\n/**\n * Transforms a JavaScript file using Babel to inline the request locale and translation.\n * @param code A string containing the JavaScript code to transform.\n * @param map A sourcemap object for the provided JavaScript code.\n * @param options The inline request options to use.\n * @returns An object containing the code, map, and diagnostics from the transformation.\n */\nasync function transformWithBabel(\n  code: string,\n  map: SourceMapInput | undefined,\n  options: InlineRequest,\n) {\n  let ast;\n  try {\n    ast = parseSync(code, {\n      babelrc: false,\n      configFile: false,\n      sourceType: 'unambiguous',\n      filename: options.filename,\n    });\n  } catch (error) {\n    assertIsError(error);\n\n    // Make the error more readable.\n    // Same errors will contain the full content of the file as the error message\n    // Which makes it hard to find the actual error message.\n    const index = error.message.indexOf(')\\n');\n    const msg = index !== -1 ? error.message.slice(0, index + 1) : error.message;\n    throw new Error(`${msg}\\nAn error occurred inlining file \"${options.filename}\"`);\n  }\n\n  if (!ast) {\n    throw new Error(`Unknown error occurred inlining file \"${options.filename}\"`);\n  }\n\n  const { diagnostics, plugins } = await createI18nPlugins(options.locale, options.translation);\n  const transformResult = await transformFromAstAsync(ast, code, {\n    filename: options.filename,\n    // false is a valid value but not included in the type definition\n    inputSourceMap: false as unknown as undefined,\n    sourceMaps: !!map,\n    compact: shouldOptimize,\n    configFile: false,\n    babelrc: false,\n    browserslistConfigFile: false,\n    plugins,\n  });\n\n  if (!transformResult || !transformResult.code) {\n    throw new Error(`Unknown error occurred processing bundle for \"${options.filename}\".`);\n  }\n\n  let outputMap;\n  if (map && transformResult.map) {\n    outputMap = remapping([transformResult.map as SourceMapInput, map], () => null);\n  }\n\n  return { code: transformResult.code, map: outputMap && JSON.stringify(outputMap), diagnostics };\n}\n"]} \ No newline at end of file diff --git a/src/tools/esbuild/i18n-inliner.d.ts b/src/tools/esbuild/i18n-inliner.d.ts new file mode 100644 index 00000000..7c88245a --- /dev/null +++ b/src/tools/esbuild/i18n-inliner.d.ts @@ -0,0 +1,40 @@ +/** + * @license + * Copyright Google LLC All Rights Reserved. + * + * Use of this source code is governed by an MIT-style license that can be + * found in the LICENSE file at https://angular.io/license + */ +import type { OutputFile } from 'esbuild'; +/** + * Inlining options that should apply to all transformed code. + */ +export interface I18nInlinerOptions { + missingTranslation: 'error' | 'warning' | 'ignore'; + outputFiles: OutputFile[]; + shouldOptimize?: boolean; +} +/** + * A class that performs i18n translation inlining of JavaScript code. + * A worker pool is used to distribute the transformation actions and allow + * parallel processing. Inlining is only performed on code that contains the + * localize function (`$localize`). + */ +export declare class I18nInliner { + #private; + constructor(options: I18nInlinerOptions, maxThreads?: number); + /** + * Performs inlining of translations for the provided locale and translations. The files that + * are processed originate from the files passed to the class constructor and filter by presence + * of the localize function keyword. + * @param locale The string representing the locale to inline. + * @param translation The translation messages to use when inlining. + * @returns A promise that resolves to an array of OutputFiles representing a translated result. + */ + inlineForLocale(locale: string, translation: Record | undefined): Promise; + /** + * Stops all active transformation tasks and shuts down all workers. + * @returns A void promise that resolves when closing is complete. + */ + close(): Promise; +} diff --git a/src/tools/esbuild/i18n-inliner.js b/src/tools/esbuild/i18n-inliner.js new file mode 100644 index 00000000..cca9c4b5 --- /dev/null +++ b/src/tools/esbuild/i18n-inliner.js @@ -0,0 +1,119 @@ +"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 __importDefault = (this && this.__importDefault) || function (mod) { + return (mod && mod.__esModule) ? mod : { "default": mod }; +}; +Object.defineProperty(exports, "__esModule", { value: true }); +exports.I18nInliner = void 0; +const piscina_1 = __importDefault(require("piscina")); +const utils_1 = require("./utils"); +/** + * A keyword used to indicate if a JavaScript file may require inlining of translations. + * This keyword is used to avoid processing files that would not otherwise need i18n processing. + */ +const LOCALIZE_KEYWORD = '$localize'; +/** + * A class that performs i18n translation inlining of JavaScript code. + * A worker pool is used to distribute the transformation actions and allow + * parallel processing. Inlining is only performed on code that contains the + * localize function (`$localize`). + */ +class I18nInliner { + #workerPool; + #localizeFiles; + #unmodifiedFiles; + constructor(options, maxThreads) { + this.#unmodifiedFiles = []; + const files = new Map(); + const pendingMaps = []; + for (const file of options.outputFiles) { + if (file.path.endsWith('.js')) { + // Check if localizations are present + const contentBuffer = Buffer.isBuffer(file.contents) + ? file.contents + : Buffer.from(file.contents.buffer, file.contents.byteOffset, file.contents.byteLength); + const hasLocalize = contentBuffer.includes(LOCALIZE_KEYWORD); + if (hasLocalize) { + // A Blob is an immutable data structure that allows sharing the data between workers + // without copying until the data is actually used within a Worker. This is useful here + // since each file may not actually be processed in each Worker and the Blob avoids + // unneeded repeat copying of potentially large JavaScript files. + files.set(file.path, new Blob([file.contents])); + continue; + } + } + else if (file.path.endsWith('.js.map')) { + // The related JS file may not have been checked yet. To ensure that map files are not + // missed, store any pending map files and check them after all output files. + pendingMaps.push(file); + continue; + } + this.#unmodifiedFiles.push(file); + } + // Check if any pending map files should be processed by checking if the parent JS file is present + for (const file of pendingMaps) { + if (files.has(file.path.slice(0, -4))) { + files.set(file.path, new Blob([file.contents])); + } + else { + this.#unmodifiedFiles.push(file); + } + } + this.#localizeFiles = files; + this.#workerPool = new piscina_1.default({ + filename: require.resolve('./i18n-inliner-worker'), + maxThreads, + // Extract options to ensure only the named options are serialized and sent to the worker + workerData: { + missingTranslation: options.missingTranslation, + shouldOptimize: options.shouldOptimize, + files, + }, + }); + } + /** + * Performs inlining of translations for the provided locale and translations. The files that + * are processed originate from the files passed to the class constructor and filter by presence + * of the localize function keyword. + * @param locale The string representing the locale to inline. + * @param translation The translation messages to use when inlining. + * @returns A promise that resolves to an array of OutputFiles representing a translated result. + */ + async inlineForLocale(locale, translation) { + // Request inlining for each file that contains localize calls + const requests = []; + for (const filename of this.#localizeFiles.keys()) { + if (filename.endsWith('.map')) { + continue; + } + const fileRequest = this.#workerPool.run({ + filename, + locale, + translation, + }); + requests.push(fileRequest); + } + // Wait for all file requests to complete + const rawResults = await Promise.all(requests); + // Convert raw results to output file objects and include all unmodified files + return [ + ...rawResults.flat().map(({ file, contents }) => (0, utils_1.createOutputFileFromData)(file, contents)), + ...this.#unmodifiedFiles.map((file) => (0, utils_1.cloneOutputFile)(file)), + ]; + } + /** + * Stops all active transformation tasks and shuts down all workers. + * @returns A void promise that resolves when closing is complete. + */ + close() { + return this.#workerPool.destroy(); + } +} +exports.I18nInliner = I18nInliner; +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"i18n-inliner.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/i18n-inliner.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 { OutputFile } from 'esbuild';\nimport Piscina from 'piscina';\nimport { cloneOutputFile, createOutputFileFromData } from './utils';\n\n/**\n * A keyword used to indicate if a JavaScript file may require inlining of translations.\n * This keyword is used to avoid processing files that would not otherwise need i18n processing.\n */\nconst LOCALIZE_KEYWORD = '$localize';\n\n/**\n * Inlining options that should apply to all transformed code.\n */\nexport interface I18nInlinerOptions {\n  missingTranslation: 'error' | 'warning' | 'ignore';\n  outputFiles: OutputFile[];\n  shouldOptimize?: boolean;\n}\n\n/**\n * A class that performs i18n translation inlining of JavaScript code.\n * A worker pool is used to distribute the transformation actions and allow\n * parallel processing. Inlining is only performed on code that contains the\n * localize function (`$localize`).\n */\nexport class I18nInliner {\n  #workerPool: Piscina;\n  readonly #localizeFiles: ReadonlyMap<string, Blob>;\n  readonly #unmodifiedFiles: Array<OutputFile>;\n\n  constructor(options: I18nInlinerOptions, maxThreads?: number) {\n    this.#unmodifiedFiles = [];\n\n    const files = new Map<string, Blob>();\n    const pendingMaps = [];\n    for (const file of options.outputFiles) {\n      if (file.path.endsWith('.js')) {\n        // Check if localizations are present\n        const contentBuffer = Buffer.isBuffer(file.contents)\n          ? file.contents\n          : Buffer.from(file.contents.buffer, file.contents.byteOffset, file.contents.byteLength);\n        const hasLocalize = contentBuffer.includes(LOCALIZE_KEYWORD);\n\n        if (hasLocalize) {\n          // A Blob is an immutable data structure that allows sharing the data between workers\n          // without copying until the data is actually used within a Worker. This is useful here\n          // since each file may not actually be processed in each Worker and the Blob avoids\n          // unneeded repeat copying of potentially large JavaScript files.\n          files.set(file.path, new Blob([file.contents]));\n\n          continue;\n        }\n      } else if (file.path.endsWith('.js.map')) {\n        // The related JS file may not have been checked yet. To ensure that map files are not\n        // missed, store any pending map files and check them after all output files.\n        pendingMaps.push(file);\n        continue;\n      }\n\n      this.#unmodifiedFiles.push(file);\n    }\n\n    // Check if any pending map files should be processed by checking if the parent JS file is present\n    for (const file of pendingMaps) {\n      if (files.has(file.path.slice(0, -4))) {\n        files.set(file.path, new Blob([file.contents]));\n      } else {\n        this.#unmodifiedFiles.push(file);\n      }\n    }\n\n    this.#localizeFiles = files;\n\n    this.#workerPool = new Piscina({\n      filename: require.resolve('./i18n-inliner-worker'),\n      maxThreads,\n      // Extract options to ensure only the named options are serialized and sent to the worker\n      workerData: {\n        missingTranslation: options.missingTranslation,\n        shouldOptimize: options.shouldOptimize,\n        files,\n      },\n    });\n  }\n\n  /**\n   * Performs inlining of translations for the provided locale and translations. The files that\n   * are processed originate from the files passed to the class constructor and filter by presence\n   * of the localize function keyword.\n   * @param locale The string representing the locale to inline.\n   * @param translation The translation messages to use when inlining.\n   * @returns A promise that resolves to an array of OutputFiles representing a translated result.\n   */\n  async inlineForLocale(\n    locale: string,\n    translation: Record<string, unknown> | undefined,\n  ): Promise<OutputFile[]> {\n    // Request inlining for each file that contains localize calls\n    const requests = [];\n    for (const filename of this.#localizeFiles.keys()) {\n      if (filename.endsWith('.map')) {\n        continue;\n      }\n\n      const fileRequest = this.#workerPool.run({\n        filename,\n        locale,\n        translation,\n      });\n      requests.push(fileRequest);\n    }\n\n    // Wait for all file requests to complete\n    const rawResults = await Promise.all(requests);\n\n    // Convert raw results to output file objects and include all unmodified files\n    return [\n      ...rawResults.flat().map(({ file, contents }) => createOutputFileFromData(file, contents)),\n      ...this.#unmodifiedFiles.map((file) => cloneOutputFile(file)),\n    ];\n  }\n\n  /**\n   * Stops all active transformation tasks and shuts down all workers.\n   * @returns A void promise that resolves when closing is complete.\n   */\n  close(): Promise<void> {\n    return this.#workerPool.destroy();\n  }\n}\n"]} \ No newline at end of file diff --git a/src/tools/esbuild/i18n-locale-plugin.d.ts b/src/tools/esbuild/i18n-locale-plugin.d.ts new file mode 100644 index 00000000..6a5110dd --- /dev/null +++ b/src/tools/esbuild/i18n-locale-plugin.d.ts @@ -0,0 +1,18 @@ +/** + * @license + * Copyright Google LLC All Rights Reserved. + * + * Use of this source code is governed by an MIT-style license that can be + * found in the LICENSE file at https://angular.io/license + */ +import type { Plugin } from 'esbuild'; +/** + * The base module location used to search for locale specific data. + */ +export declare const LOCALE_DATA_BASE_MODULE = "@angular/common/locales/global"; +/** + * Creates an esbuild plugin that resolves Angular locale data files from `@angular/common`. + * + * @returns An esbuild plugin. + */ +export declare function createAngularLocaleDataPlugin(): Plugin; diff --git a/src/tools/esbuild/i18n-locale-plugin.js b/src/tools/esbuild/i18n-locale-plugin.js new file mode 100644 index 00000000..6d9ac66d --- /dev/null +++ b/src/tools/esbuild/i18n-locale-plugin.js @@ -0,0 +1,91 @@ +"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 + */ +Object.defineProperty(exports, "__esModule", { value: true }); +exports.createAngularLocaleDataPlugin = exports.LOCALE_DATA_BASE_MODULE = void 0; +/** + * The base module location used to search for locale specific data. + */ +exports.LOCALE_DATA_BASE_MODULE = '@angular/common/locales/global'; +/** + * Creates an esbuild plugin that resolves Angular locale data files from `@angular/common`. + * + * @returns An esbuild plugin. + */ +function createAngularLocaleDataPlugin() { + return { + name: 'angular-locale-data', + setup(build) { + // If packages are configured to be external then leave the original angular locale import path. + // This happens when using the development server with caching enabled to allow Vite prebundling to work. + // There currently is no option on the esbuild resolve function to resolve while disabling the option. To + // workaround the inability to resolve the full locale location here, the Vite dev server prebundling also + // contains a plugin to allow the locales to be correctly resolved when prebundling. + // NOTE: If esbuild eventually allows controlling the external package options in a build.resolve call, this + // workaround can be removed. + if (build.initialOptions.packages === 'external') { + return; + } + build.onResolve({ filter: /^angular:locale\/data:/ }, async ({ path }) => { + // Extract the locale from the path + const originalLocale = path.split(':', 3)[2]; + // Remove any private subtags since these will never match + let partialLocale = originalLocale.replace(/-x(-[a-zA-Z0-9]{1,8})+$/, ''); + let exact = true; + while (partialLocale) { + const potentialPath = `${exports.LOCALE_DATA_BASE_MODULE}/${partialLocale}`; + const result = await build.resolve(potentialPath, { + kind: 'import-statement', + resolveDir: build.initialOptions.absWorkingDir, + }); + if (result.path) { + if (exact) { + return result; + } + else { + return { + ...result, + warnings: [ + ...result.warnings, + { + location: null, + text: `Locale data for '${originalLocale}' cannot be found. Using locale data for '${partialLocale}'.`, + }, + ], + }; + } + } + // Remove the last subtag and try again with a less specific locale + const parts = partialLocale.split('-'); + partialLocale = parts.slice(0, -1).join('-'); + exact = false; + // The locales "en" and "en-US" are considered exact to retain existing behavior + if (originalLocale === 'en-US' && partialLocale === 'en') { + exact = true; + } + } + // Not found so issue a warning and use an empty loader. Framework built-in `en-US` data will be used. + // This retains existing behavior as in the Webpack-based builder. + return { + path: originalLocale, + namespace: 'angular:locale/data', + warnings: [ + { + location: null, + text: `Locale data for '${originalLocale}' cannot be found. No locale data will be included for this locale.`, + }, + ], + }; + }); + // Locales that cannot be found will be loaded as empty content with a warning from the resolve step + build.onLoad({ filter: /./, namespace: 'angular:locale/data' }, () => ({ loader: 'empty' })); + }, + }; +} +exports.createAngularLocaleDataPlugin = createAngularLocaleDataPlugin; +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaTE4bi1sb2NhbGUtcGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC9pMThuLWxvY2FsZS1wbHVnaW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBSUg7O0dBRUc7QUFDVSxRQUFBLHVCQUF1QixHQUFHLGdDQUFnQyxDQUFDO0FBRXhFOzs7O0dBSUc7QUFDSCxTQUFnQiw2QkFBNkI7SUFDM0MsT0FBTztRQUNMLElBQUksRUFBRSxxQkFBcUI7UUFDM0IsS0FBSyxDQUFDLEtBQUs7WUFDVCxnR0FBZ0c7WUFDaEcseUdBQXlHO1lBQ3pHLHlHQUF5RztZQUN6RywwR0FBMEc7WUFDMUcsb0ZBQW9GO1lBQ3BGLDRHQUE0RztZQUM1RyxtQ0FBbUM7WUFDbkMsSUFBSSxLQUFLLENBQUMsY0FBYyxDQUFDLFFBQVEsS0FBSyxVQUFVLEVBQUU7Z0JBQ2hELE9BQU87YUFDUjtZQUVELEtBQUssQ0FBQyxTQUFTLENBQUMsRUFBRSxNQUFNLEVBQUUsd0JBQXdCLEVBQUUsRUFBRSxLQUFLLEVBQUUsRUFBRSxJQUFJLEVBQUUsRUFBRSxFQUFFO2dCQUN2RSxtQ0FBbUM7Z0JBQ25DLE1BQU0sY0FBYyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO2dCQUU3QywwREFBMEQ7Z0JBQzFELElBQUksYUFBYSxHQUFHLGNBQWMsQ0FBQyxPQUFPLENBQUMseUJBQXlCLEVBQUUsRUFBRSxDQUFDLENBQUM7Z0JBRTFFLElBQUksS0FBSyxHQUFHLElBQUksQ0FBQztnQkFDakIsT0FBTyxhQUFhLEVBQUU7b0JBQ3BCLE1BQU0sYUFBYSxHQUFHLEdBQUcsK0JBQXVCLElBQUksYUFBYSxFQUFFLENBQUM7b0JBRXBFLE1BQU0sTUFBTSxHQUFHLE1BQU0sS0FBSyxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUU7d0JBQ2hELElBQUksRUFBRSxrQkFBa0I7d0JBQ3hCLFVBQVUsRUFBRSxLQUFLLENBQUMsY0FBYyxDQUFDLGFBQWE7cUJBQy9DLENBQUMsQ0FBQztvQkFDSCxJQUFJLE1BQU0sQ0FBQyxJQUFJLEVBQUU7d0JBQ2YsSUFBSSxLQUFLLEVBQUU7NEJBQ1QsT0FBTyxNQUFNLENBQUM7eUJBQ2Y7NkJBQU07NEJBQ0wsT0FBTztnQ0FDTCxHQUFHLE1BQU07Z0NBQ1QsUUFBUSxFQUFFO29DQUNSLEdBQUcsTUFBTSxDQUFDLFFBQVE7b0NBQ2xCO3dDQUNFLFFBQVEsRUFBRSxJQUFJO3dDQUNkLElBQUksRUFBRSxvQkFBb0IsY0FBYyw2Q0FBNkMsYUFBYSxJQUFJO3FDQUN2RztpQ0FDRjs2QkFDRixDQUFDO3lCQUNIO3FCQUNGO29CQUVELG1FQUFtRTtvQkFDbkUsTUFBTSxLQUFLLEdBQUcsYUFBYSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztvQkFDdkMsYUFBYSxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO29CQUM3QyxLQUFLLEdBQUcsS0FBSyxDQUFDO29CQUNkLGdGQUFnRjtvQkFDaEYsSUFBSSxjQUFjLEtBQUssT0FBTyxJQUFJLGFBQWEsS0FBSyxJQUFJLEVBQUU7d0JBQ3hELEtBQUssR0FBRyxJQUFJLENBQUM7cUJBQ2Q7aUJBQ0Y7Z0JBRUQsc0dBQXNHO2dCQUN0RyxrRUFBa0U7Z0JBQ2xFLE9BQU87b0JBQ0wsSUFBSSxFQUFFLGNBQWM7b0JBQ3BCLFNBQVMsRUFBRSxxQkFBcUI7b0JBQ2hDLFFBQVEsRUFBRTt3QkFDUjs0QkFDRSxRQUFRLEVBQUUsSUFBSTs0QkFDZCxJQUFJLEVBQUUsb0JBQW9CLGNBQWMscUVBQXFFO3lCQUM5RztxQkFDRjtpQkFDRixDQUFDO1lBQ0osQ0FBQyxDQUFDLENBQUM7WUFFSCxvR0FBb0c7WUFDcEcsS0FBSyxDQUFDLE1BQU0sQ0FBQyxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLHFCQUFxQixFQUFFLEVBQUUsR0FBRyxFQUFFLENBQUMsQ0FBQyxFQUFFLE1BQU0sRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFDLENBQUM7UUFDL0YsQ0FBQztLQUNGLENBQUM7QUFDSixDQUFDO0FBM0VELHNFQTJFQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFBsdWdpbiB9IGZyb20gJ2VzYnVpbGQnO1xuXG4vKipcbiAqIFRoZSBiYXNlIG1vZHVsZSBsb2NhdGlvbiB1c2VkIHRvIHNlYXJjaCBmb3IgbG9jYWxlIHNwZWNpZmljIGRhdGEuXG4gKi9cbmV4cG9ydCBjb25zdCBMT0NBTEVfREFUQV9CQVNFX01PRFVMRSA9ICdAYW5ndWxhci9jb21tb24vbG9jYWxlcy9nbG9iYWwnO1xuXG4vKipcbiAqIENyZWF0ZXMgYW4gZXNidWlsZCBwbHVnaW4gdGhhdCByZXNvbHZlcyBBbmd1bGFyIGxvY2FsZSBkYXRhIGZpbGVzIGZyb20gYEBhbmd1bGFyL2NvbW1vbmAuXG4gKlxuICogQHJldHVybnMgQW4gZXNidWlsZCBwbHVnaW4uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVBbmd1bGFyTG9jYWxlRGF0YVBsdWdpbigpOiBQbHVnaW4ge1xuICByZXR1cm4ge1xuICAgIG5hbWU6ICdhbmd1bGFyLWxvY2FsZS1kYXRhJyxcbiAgICBzZXR1cChidWlsZCk6IHZvaWQge1xuICAgICAgLy8gSWYgcGFja2FnZXMgYXJlIGNvbmZpZ3VyZWQgdG8gYmUgZXh0ZXJuYWwgdGhlbiBsZWF2ZSB0aGUgb3JpZ2luYWwgYW5ndWxhciBsb2NhbGUgaW1wb3J0IHBhdGguXG4gICAgICAvLyBUaGlzIGhhcHBlbnMgd2hlbiB1c2luZyB0aGUgZGV2ZWxvcG1lbnQgc2VydmVyIHdpdGggY2FjaGluZyBlbmFibGVkIHRvIGFsbG93IFZpdGUgcHJlYnVuZGxpbmcgdG8gd29yay5cbiAgICAgIC8vIFRoZXJlIGN1cnJlbnRseSBpcyBubyBvcHRpb24gb24gdGhlIGVzYnVpbGQgcmVzb2x2ZSBmdW5jdGlvbiB0byByZXNvbHZlIHdoaWxlIGRpc2FibGluZyB0aGUgb3B0aW9uLiBUb1xuICAgICAgLy8gd29ya2Fyb3VuZCB0aGUgaW5hYmlsaXR5IHRvIHJlc29sdmUgdGhlIGZ1bGwgbG9jYWxlIGxvY2F0aW9uIGhlcmUsIHRoZSBWaXRlIGRldiBzZXJ2ZXIgcHJlYnVuZGxpbmcgYWxzb1xuICAgICAgLy8gY29udGFpbnMgYSBwbHVnaW4gdG8gYWxsb3cgdGhlIGxvY2FsZXMgdG8gYmUgY29ycmVjdGx5IHJlc29sdmVkIHdoZW4gcHJlYnVuZGxpbmcuXG4gICAgICAvLyBOT1RFOiBJZiBlc2J1aWxkIGV2ZW50dWFsbHkgYWxsb3dzIGNvbnRyb2xsaW5nIHRoZSBleHRlcm5hbCBwYWNrYWdlIG9wdGlvbnMgaW4gYSBidWlsZC5yZXNvbHZlIGNhbGwsIHRoaXNcbiAgICAgIC8vICAgICAgIHdvcmthcm91bmQgY2FuIGJlIHJlbW92ZWQuXG4gICAgICBpZiAoYnVpbGQuaW5pdGlhbE9wdGlvbnMucGFja2FnZXMgPT09ICdleHRlcm5hbCcpIHtcbiAgICAgICAgcmV0dXJuO1xuICAgICAgfVxuXG4gICAgICBidWlsZC5vblJlc29sdmUoeyBmaWx0ZXI6IC9eYW5ndWxhcjpsb2NhbGVcXC9kYXRhOi8gfSwgYXN5bmMgKHsgcGF0aCB9KSA9PiB7XG4gICAgICAgIC8vIEV4dHJhY3QgdGhlIGxvY2FsZSBmcm9tIHRoZSBwYXRoXG4gICAgICAgIGNvbnN0IG9yaWdpbmFsTG9jYWxlID0gcGF0aC5zcGxpdCgnOicsIDMpWzJdO1xuXG4gICAgICAgIC8vIFJlbW92ZSBhbnkgcHJpdmF0ZSBzdWJ0YWdzIHNpbmNlIHRoZXNlIHdpbGwgbmV2ZXIgbWF0Y2hcbiAgICAgICAgbGV0IHBhcnRpYWxMb2NhbGUgPSBvcmlnaW5hbExvY2FsZS5yZXBsYWNlKC8teCgtW2EtekEtWjAtOV17MSw4fSkrJC8sICcnKTtcblxuICAgICAgICBsZXQgZXhhY3QgPSB0cnVlO1xuICAgICAgICB3aGlsZSAocGFydGlhbExvY2FsZSkge1xuICAgICAgICAgIGNvbnN0IHBvdGVudGlhbFBhdGggPSBgJHtMT0NBTEVfREFUQV9CQVNFX01PRFVMRX0vJHtwYXJ0aWFsTG9jYWxlfWA7XG5cbiAgICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBidWlsZC5yZXNvbHZlKHBvdGVudGlhbFBhdGgsIHtcbiAgICAgICAgICAgIGtpbmQ6ICdpbXBvcnQtc3RhdGVtZW50JyxcbiAgICAgICAgICAgIHJlc29sdmVEaXI6IGJ1aWxkLmluaXRpYWxPcHRpb25zLmFic1dvcmtpbmdEaXIsXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgaWYgKHJlc3VsdC5wYXRoKSB7XG4gICAgICAgICAgICBpZiAoZXhhY3QpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgLi4ucmVzdWx0LFxuICAgICAgICAgICAgICAgIHdhcm5pbmdzOiBbXG4gICAgICAgICAgICAgICAgICAuLi5yZXN1bHQud2FybmluZ3MsXG4gICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgIGxvY2F0aW9uOiBudWxsLFxuICAgICAgICAgICAgICAgICAgICB0ZXh0OiBgTG9jYWxlIGRhdGEgZm9yICcke29yaWdpbmFsTG9jYWxlfScgY2Fubm90IGJlIGZvdW5kLiBVc2luZyBsb2NhbGUgZGF0YSBmb3IgJyR7cGFydGlhbExvY2FsZX0nLmAsXG4gICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICAgIH07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gUmVtb3ZlIHRoZSBsYXN0IHN1YnRhZyBhbmQgdHJ5IGFnYWluIHdpdGggYSBsZXNzIHNwZWNpZmljIGxvY2FsZVxuICAgICAgICAgIGNvbnN0IHBhcnRzID0gcGFydGlhbExvY2FsZS5zcGxpdCgnLScpO1xuICAgICAgICAgIHBhcnRpYWxMb2NhbGUgPSBwYXJ0cy5zbGljZSgwLCAtMSkuam9pbignLScpO1xuICAgICAgICAgIGV4YWN0ID0gZmFsc2U7XG4gICAgICAgICAgLy8gVGhlIGxvY2FsZXMgXCJlblwiIGFuZCBcImVuLVVTXCIgYXJlIGNvbnNpZGVyZWQgZXhhY3QgdG8gcmV0YWluIGV4aXN0aW5nIGJlaGF2aW9yXG4gICAgICAgICAgaWYgKG9yaWdpbmFsTG9jYWxlID09PSAnZW4tVVMnICYmIHBhcnRpYWxMb2NhbGUgPT09ICdlbicpIHtcbiAgICAgICAgICAgIGV4YWN0ID0gdHJ1ZTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBOb3QgZm91bmQgc28gaXNzdWUgYSB3YXJuaW5nIGFuZCB1c2UgYW4gZW1wdHkgbG9hZGVyLiBGcmFtZXdvcmsgYnVpbHQtaW4gYGVuLVVTYCBkYXRhIHdpbGwgYmUgdXNlZC5cbiAgICAgICAgLy8gVGhpcyByZXRhaW5zIGV4aXN0aW5nIGJlaGF2aW9yIGFzIGluIHRoZSBXZWJwYWNrLWJhc2VkIGJ1aWxkZXIuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgcGF0aDogb3JpZ2luYWxMb2NhbGUsXG4gICAgICAgICAgbmFtZXNwYWNlOiAnYW5ndWxhcjpsb2NhbGUvZGF0YScsXG4gICAgICAgICAgd2FybmluZ3M6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgbG9jYXRpb246IG51bGwsXG4gICAgICAgICAgICAgIHRleHQ6IGBMb2NhbGUgZGF0YSBmb3IgJyR7b3JpZ2luYWxMb2NhbGV9JyBjYW5ub3QgYmUgZm91bmQuIE5vIGxvY2FsZSBkYXRhIHdpbGwgYmUgaW5jbHVkZWQgZm9yIHRoaXMgbG9jYWxlLmAsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIF0sXG4gICAgICAgIH07XG4gICAgICB9KTtcblxuICAgICAgLy8gTG9jYWxlcyB0aGF0IGNhbm5vdCBiZSBmb3VuZCB3aWxsIGJlIGxvYWRlZCBhcyBlbXB0eSBjb250ZW50IHdpdGggYSB3YXJuaW5nIGZyb20gdGhlIHJlc29sdmUgc3RlcFxuICAgICAgYnVpbGQub25Mb2FkKHsgZmlsdGVyOiAvLi8sIG5hbWVzcGFjZTogJ2FuZ3VsYXI6bG9jYWxlL2RhdGEnIH0sICgpID0+ICh7IGxvYWRlcjogJ2VtcHR5JyB9KSk7XG4gICAgfSxcbiAgfTtcbn1cbiJdfQ== \ No newline at end of file diff --git a/src/tools/esbuild/index-html-generator.d.ts b/src/tools/esbuild/index-html-generator.d.ts index 426033de..273b2be2 100644 --- a/src/tools/esbuild/index-html-generator.d.ts +++ b/src/tools/esbuild/index-html-generator.d.ts @@ -5,10 +5,10 @@ * 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 type { OutputFile } from 'esbuild'; import { NormalizedApplicationBuildOptions } from '../../builders/application/options'; import { InitialFileRecord } from './bundler-context'; -import type { ExecutionResult } from './bundler-execution-result'; -export declare function generateIndexHtml(initialFiles: Map, executionResult: ExecutionResult, buildOptions: NormalizedApplicationBuildOptions): Promise<{ +export declare function generateIndexHtml(initialFiles: Map, outputFiles: OutputFile[], buildOptions: NormalizedApplicationBuildOptions, lang?: string): Promise<{ content: string; contentWithoutCriticalCssInlined: string; warnings: string[]; diff --git a/src/tools/esbuild/index-html-generator.js b/src/tools/esbuild/index-html-generator.js index d81f96fb..0fbbaf30 100644 --- a/src/tools/esbuild/index-html-generator.js +++ b/src/tools/esbuild/index-html-generator.js @@ -15,7 +15,7 @@ const node_assert_1 = __importDefault(require("node:assert")); const node_path_1 = __importDefault(require("node:path")); const index_html_generator_1 = require("../../utils/index-file/index-html-generator"); const inline_critical_css_1 = require("../../utils/index-file/inline-critical-css"); -async function generateIndexHtml(initialFiles, executionResult, buildOptions) { +async function generateIndexHtml(initialFiles, outputFiles, buildOptions, lang) { // Analyze metafile for initial link-based hints. // Skip if the internal externalPackages option is enabled since this option requires // dev server cooperation to properly resolve and fetch imports. @@ -43,7 +43,7 @@ async function generateIndexHtml(initialFiles, executionResult, buildOptions) { const readAsset = async function (filePath) { // Remove leading directory separator const relativefilePath = node_path_1.default.relative(virtualOutputPath, filePath); - const file = executionResult.outputFiles.find((file) => file.path === relativefilePath); + const file = outputFiles.find((file) => file.path === relativefilePath); if (file) { return file.text; } @@ -66,7 +66,7 @@ async function generateIndexHtml(initialFiles, executionResult, buildOptions) { indexHtmlGenerator.readAsset = readAsset; const transformResult = await indexHtmlGenerator.process({ baseHref, - lang: undefined, + lang, outputPath: virtualOutputPath, files: [...initialFiles].map(([file, record]) => ({ name: record.name ?? '', @@ -97,4 +97,4 @@ async function generateIndexHtml(initialFiles, executionResult, buildOptions) { }; } exports.generateIndexHtml = generateIndexHtml; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index-html-generator.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/index-html-generator.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 assert from 'node:assert';\nimport path from 'node:path';\nimport { NormalizedApplicationBuildOptions } from '../../builders/application/options';\nimport { IndexHtmlGenerator } from '../../utils/index-file/index-html-generator';\nimport { InlineCriticalCssProcessor } from '../../utils/index-file/inline-critical-css';\nimport { InitialFileRecord } from './bundler-context';\nimport type { ExecutionResult } from './bundler-execution-result';\n\nexport async function generateIndexHtml(\n  initialFiles: Map<string, InitialFileRecord>,\n  executionResult: ExecutionResult,\n  buildOptions: NormalizedApplicationBuildOptions,\n): Promise<{\n  content: string;\n  contentWithoutCriticalCssInlined: string;\n  warnings: string[];\n  errors: string[];\n}> {\n  // Analyze metafile for initial link-based hints.\n  // Skip if the internal externalPackages option is enabled since this option requires\n  // dev server cooperation to properly resolve and fetch imports.\n  const hints = [];\n  const {\n    indexHtmlOptions,\n    externalPackages,\n    optimizationOptions,\n    crossOrigin,\n    subresourceIntegrity,\n    baseHref,\n  } = buildOptions;\n\n  assert(indexHtmlOptions, 'indexHtmlOptions cannot be undefined.');\n\n  if (!externalPackages) {\n    for (const [key, value] of initialFiles) {\n      if (value.entrypoint) {\n        // Entry points are already referenced in the HTML\n        continue;\n      }\n      if (value.type === 'script') {\n        hints.push({ url: key, mode: 'modulepreload' as const });\n      } else if (value.type === 'style') {\n        // Provide an \"as\" value of \"style\" to ensure external URLs which may not have a\n        // file extension are treated as stylesheets.\n        hints.push({ url: key, mode: 'preload' as const, as: 'style' });\n      }\n    }\n  }\n\n  /** Virtual output path to support reading in-memory files. */\n  const virtualOutputPath = '/';\n  const readAsset = async function (filePath: string): Promise<string> {\n    // Remove leading directory separator\n    const relativefilePath = path.relative(virtualOutputPath, filePath);\n    const file = executionResult.outputFiles.find((file) => file.path === relativefilePath);\n    if (file) {\n      return file.text;\n    }\n\n    throw new Error(`Output file does not exist: ${relativefilePath}`);\n  };\n\n  // Create an index HTML generator that reads from the in-memory output files\n  const indexHtmlGenerator = new IndexHtmlGenerator({\n    indexPath: indexHtmlOptions.input,\n    entrypoints: indexHtmlOptions.insertionOrder,\n    sri: subresourceIntegrity,\n    optimization: {\n      ...optimizationOptions,\n      styles: {\n        ...optimizationOptions.styles,\n        inlineCritical: false, // Disable critical css inline as for SSR and SSG this will be done during rendering.\n      },\n    },\n    crossOrigin: crossOrigin,\n  });\n\n  indexHtmlGenerator.readAsset = readAsset;\n\n  const transformResult = await indexHtmlGenerator.process({\n    baseHref,\n    lang: undefined,\n    outputPath: virtualOutputPath,\n    files: [...initialFiles].map(([file, record]) => ({\n      name: record.name ?? '',\n      file,\n      extension: path.extname(file),\n    })),\n    hints,\n  });\n\n  const contentWithoutCriticalCssInlined = transformResult.content;\n  if (!optimizationOptions.styles.inlineCritical) {\n    return {\n      ...transformResult,\n      contentWithoutCriticalCssInlined,\n    };\n  }\n\n  const inlineCriticalCssProcessor = new InlineCriticalCssProcessor({\n    minify: false, // CSS has already been minified during the build.\n    readAsset,\n  });\n\n  const { content, errors, warnings } = await inlineCriticalCssProcessor.process(\n    contentWithoutCriticalCssInlined,\n    {\n      outputPath: virtualOutputPath,\n    },\n  );\n\n  return {\n    errors: [...transformResult.errors, ...errors],\n    warnings: [...transformResult.warnings, ...warnings],\n    content,\n    contentWithoutCriticalCssInlined,\n  };\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index-html-generator.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/index-html-generator.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 { OutputFile } from 'esbuild';\nimport assert from 'node:assert';\nimport path from 'node:path';\nimport { NormalizedApplicationBuildOptions } from '../../builders/application/options';\nimport { IndexHtmlGenerator } from '../../utils/index-file/index-html-generator';\nimport { InlineCriticalCssProcessor } from '../../utils/index-file/inline-critical-css';\nimport { InitialFileRecord } from './bundler-context';\n\nexport async function generateIndexHtml(\n  initialFiles: Map<string, InitialFileRecord>,\n  outputFiles: OutputFile[],\n  buildOptions: NormalizedApplicationBuildOptions,\n  lang?: string,\n): Promise<{\n  content: string;\n  contentWithoutCriticalCssInlined: string;\n  warnings: string[];\n  errors: string[];\n}> {\n  // Analyze metafile for initial link-based hints.\n  // Skip if the internal externalPackages option is enabled since this option requires\n  // dev server cooperation to properly resolve and fetch imports.\n  const hints = [];\n  const {\n    indexHtmlOptions,\n    externalPackages,\n    optimizationOptions,\n    crossOrigin,\n    subresourceIntegrity,\n    baseHref,\n  } = buildOptions;\n\n  assert(indexHtmlOptions, 'indexHtmlOptions cannot be undefined.');\n\n  if (!externalPackages) {\n    for (const [key, value] of initialFiles) {\n      if (value.entrypoint) {\n        // Entry points are already referenced in the HTML\n        continue;\n      }\n      if (value.type === 'script') {\n        hints.push({ url: key, mode: 'modulepreload' as const });\n      } else if (value.type === 'style') {\n        // Provide an \"as\" value of \"style\" to ensure external URLs which may not have a\n        // file extension are treated as stylesheets.\n        hints.push({ url: key, mode: 'preload' as const, as: 'style' });\n      }\n    }\n  }\n\n  /** Virtual output path to support reading in-memory files. */\n  const virtualOutputPath = '/';\n  const readAsset = async function (filePath: string): Promise<string> {\n    // Remove leading directory separator\n    const relativefilePath = path.relative(virtualOutputPath, filePath);\n    const file = outputFiles.find((file) => file.path === relativefilePath);\n    if (file) {\n      return file.text;\n    }\n\n    throw new Error(`Output file does not exist: ${relativefilePath}`);\n  };\n\n  // Create an index HTML generator that reads from the in-memory output files\n  const indexHtmlGenerator = new IndexHtmlGenerator({\n    indexPath: indexHtmlOptions.input,\n    entrypoints: indexHtmlOptions.insertionOrder,\n    sri: subresourceIntegrity,\n    optimization: {\n      ...optimizationOptions,\n      styles: {\n        ...optimizationOptions.styles,\n        inlineCritical: false, // Disable critical css inline as for SSR and SSG this will be done during rendering.\n      },\n    },\n    crossOrigin: crossOrigin,\n  });\n\n  indexHtmlGenerator.readAsset = readAsset;\n\n  const transformResult = await indexHtmlGenerator.process({\n    baseHref,\n    lang,\n    outputPath: virtualOutputPath,\n    files: [...initialFiles].map(([file, record]) => ({\n      name: record.name ?? '',\n      file,\n      extension: path.extname(file),\n    })),\n    hints,\n  });\n\n  const contentWithoutCriticalCssInlined = transformResult.content;\n  if (!optimizationOptions.styles.inlineCritical) {\n    return {\n      ...transformResult,\n      contentWithoutCriticalCssInlined,\n    };\n  }\n\n  const inlineCriticalCssProcessor = new InlineCriticalCssProcessor({\n    minify: false, // CSS has already been minified during the build.\n    readAsset,\n  });\n\n  const { content, errors, warnings } = await inlineCriticalCssProcessor.process(\n    contentWithoutCriticalCssInlined,\n    {\n      outputPath: virtualOutputPath,\n    },\n  );\n\n  return {\n    errors: [...transformResult.errors, ...errors],\n    warnings: [...transformResult.warnings, ...warnings],\n    content,\n    contentWithoutCriticalCssInlined,\n  };\n}\n"]} \ No newline at end of file diff --git a/src/tools/esbuild/utils.d.ts b/src/tools/esbuild/utils.d.ts index 3da9613c..a2ee54bf 100644 --- a/src/tools/esbuild/utils.d.ts +++ b/src/tools/esbuild/utils.d.ts @@ -28,6 +28,8 @@ export declare function writeResultFiles(outputFiles: OutputFile[], assetFiles: destination: string; }[] | undefined, outputPath: string): Promise; export declare function createOutputFileFromText(path: string, text: string): OutputFile; +export declare function createOutputFileFromData(path: string, data: Uint8Array): OutputFile; +export declare function cloneOutputFile(file: OutputFile): OutputFile; /** * Transform browserlists result to esbuild target. * @see https://esbuild.github.io/api/#target diff --git a/src/tools/esbuild/utils.js b/src/tools/esbuild/utils.js index 8aefcc66..b91c2709 100644 --- a/src/tools/esbuild/utils.js +++ b/src/tools/esbuild/utils.js @@ -10,7 +10,7 @@ var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); -exports.transformSupportedBrowsersToTargets = exports.createOutputFileFromText = exports.writeResultFiles = exports.getFeatureSupport = exports.logMessages = exports.withNoProgress = exports.withSpinner = exports.calculateEstimatedTransferSizes = exports.logBuildStats = void 0; +exports.transformSupportedBrowsersToTargets = exports.cloneOutputFile = exports.createOutputFileFromData = exports.createOutputFileFromText = exports.writeResultFiles = exports.getFeatureSupport = exports.logMessages = exports.withNoProgress = exports.withSpinner = exports.calculateEstimatedTransferSizes = exports.logBuildStats = void 0; const esbuild_1 = require("esbuild"); const node_crypto_1 = require("node:crypto"); const node_fs_1 = require("node:fs"); @@ -180,6 +180,38 @@ function createOutputFileFromText(path, text) { }; } exports.createOutputFileFromText = createOutputFileFromText; +function createOutputFileFromData(path, data) { + return { + path, + get text() { + return Buffer.from(data.buffer, data.byteOffset, data.byteLength).toString('utf-8'); + }, + get hash() { + return (0, node_crypto_1.createHash)('sha256').update(data).digest('hex'); + }, + get contents() { + return data; + }, + }; +} +exports.createOutputFileFromData = createOutputFileFromData; +function cloneOutputFile(file) { + const path = file.path; + const data = file.contents; + return { + path, + get text() { + return Buffer.from(data.buffer, data.byteOffset, data.byteLength).toString('utf-8'); + }, + get hash() { + return (0, node_crypto_1.createHash)('sha256').update(data).digest('hex'); + }, + get contents() { + return data; + }, + }; +} +exports.cloneOutputFile = cloneOutputFile; /** * Transform browserlists result to esbuild target. * @see https://esbuild.github.io/api/#target @@ -224,4 +256,4 @@ function transformSupportedBrowsersToTargets(supportedBrowsers) { return transformed; } exports.transformSupportedBrowsersToTargets = transformSupportedBrowsersToTargets; -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/utils.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 { BuildOptions, Metafile, OutputFile, PartialMessage, formatMessages } from 'esbuild';\nimport { createHash } from 'node:crypto';\nimport { constants as fsConstants } from 'node:fs';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\nimport { promisify } from 'node:util';\nimport { brotliCompress } from 'node:zlib';\nimport { Spinner } from '../../utils/spinner';\nimport { BundleStats, generateBuildStatsTable } from '../webpack/utils/stats';\nimport { InitialFileRecord } from './bundler-context';\n\nconst compressAsync = promisify(brotliCompress);\n\nexport function logBuildStats(\n  context: BuilderContext,\n  metafile: Metafile,\n  initial: Map<string, InitialFileRecord>,\n  estimatedTransferSizes?: Map<string, number>,\n): void {\n  const stats: BundleStats[] = [];\n  for (const [file, output] of Object.entries(metafile.outputs)) {\n    // Only display JavaScript and CSS files\n    if (!file.endsWith('.js') && !file.endsWith('.css')) {\n      continue;\n    }\n    // Skip internal component resources\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    if ((output as any)['ng-component']) {\n      continue;\n    }\n\n    stats.push({\n      initial: initial.has(file),\n      stats: [\n        file,\n        initial.get(file)?.name ?? '-',\n        output.bytes,\n        estimatedTransferSizes?.get(file) ?? '-',\n      ],\n    });\n  }\n\n  const tableText = generateBuildStatsTable(stats, true, true, !!estimatedTransferSizes, undefined);\n\n  context.logger.info('\\n' + tableText + '\\n');\n}\n\nexport async function calculateEstimatedTransferSizes(\n  outputFiles: OutputFile[],\n): Promise<Map<string, number>> {\n  const sizes = new Map<string, number>();\n  const pendingCompression = [];\n\n  for (const outputFile of outputFiles) {\n    // Only calculate JavaScript and CSS files\n    if (!outputFile.path.endsWith('.js') && !outputFile.path.endsWith('.css')) {\n      continue;\n    }\n\n    // Skip compressing small files which may end being larger once compressed and will most likely not be\n    // compressed in actual transit.\n    if (outputFile.contents.byteLength < 1024) {\n      sizes.set(outputFile.path, outputFile.contents.byteLength);\n      continue;\n    }\n\n    pendingCompression.push(\n      compressAsync(outputFile.contents).then((result) =>\n        sizes.set(outputFile.path, result.byteLength),\n      ),\n    );\n  }\n\n  await Promise.all(pendingCompression);\n\n  return sizes;\n}\n\nexport async function withSpinner<T>(text: string, action: () => T | Promise<T>): Promise<T> {\n  const spinner = new Spinner(text);\n  spinner.start();\n\n  try {\n    return await action();\n  } finally {\n    spinner.stop();\n  }\n}\n\nexport async function withNoProgress<T>(test: string, action: () => T | Promise<T>): Promise<T> {\n  return action();\n}\n\nexport async function logMessages(\n  context: BuilderContext,\n  { errors, warnings }: { errors?: PartialMessage[]; warnings?: PartialMessage[] },\n): Promise<void> {\n  if (warnings?.length) {\n    const warningMessages = await formatMessages(warnings, { kind: 'warning', color: true });\n    context.logger.warn(warningMessages.join('\\n'));\n  }\n\n  if (errors?.length) {\n    const errorMessages = await formatMessages(errors, { kind: 'error', color: true });\n    context.logger.error(errorMessages.join('\\n'));\n  }\n}\n\n/**\n * Generates a syntax feature object map for Angular applications based on a list of targets.\n * A full set of feature names can be found here: https://esbuild.github.io/api/#supported\n * @param target An array of browser/engine targets in the format accepted by the esbuild `target` option.\n * @returns An object that can be used with the esbuild build `supported` option.\n */\nexport function getFeatureSupport(target: string[]): BuildOptions['supported'] {\n  const supported: Record<string, boolean> = {\n    // Native async/await is not supported with Zone.js. Disabling support here will cause\n    // esbuild to downlevel async/await, async generators, and for await...of to a Zone.js supported form.\n    'async-await': false,\n    // V8 currently has a performance defect involving object spread operations that can cause signficant\n    // degradation in runtime performance. By not supporting the language feature here, a downlevel form\n    // will be used instead which provides a workaround for the performance issue.\n    // For more details: https://bugs.chromium.org/p/v8/issues/detail?id=11536\n    'object-rest-spread': false,\n  };\n\n  // Detect Safari browser versions that have a class field behavior bug\n  // See: https://github.com/angular/angular-cli/issues/24355#issuecomment-1333477033\n  // See: https://github.com/WebKit/WebKit/commit/e8788a34b3d5f5b4edd7ff6450b80936bff396f2\n  let safariClassFieldScopeBug = false;\n  for (const browser of target) {\n    let majorVersion;\n    if (browser.startsWith('ios')) {\n      majorVersion = Number(browser.slice(3, 5));\n    } else if (browser.startsWith('safari')) {\n      majorVersion = Number(browser.slice(6, 8));\n    } else {\n      continue;\n    }\n    // Technically, 14.0 is not broken but rather does not have support. However, the behavior\n    // is identical since it would be set to false by esbuild if present as a target.\n    if (majorVersion === 14 || majorVersion === 15) {\n      safariClassFieldScopeBug = true;\n      break;\n    }\n  }\n  // If class field support cannot be used set to false; otherwise leave undefined to allow\n  // esbuild to use `target` to determine support.\n  if (safariClassFieldScopeBug) {\n    supported['class-field'] = false;\n    supported['class-static-field'] = false;\n  }\n\n  return supported;\n}\n\nexport async function writeResultFiles(\n  outputFiles: OutputFile[],\n  assetFiles: { source: string; destination: string }[] | undefined,\n  outputPath: string,\n) {\n  const directoryExists = new Set<string>();\n  await Promise.all(\n    outputFiles.map(async (file) => {\n      // Ensure output subdirectories exist\n      const basePath = path.dirname(file.path);\n      if (basePath && !directoryExists.has(basePath)) {\n        await fs.mkdir(path.join(outputPath, basePath), { recursive: true });\n        directoryExists.add(basePath);\n      }\n      // Write file contents\n      await fs.writeFile(path.join(outputPath, file.path), file.contents);\n    }),\n  );\n\n  if (assetFiles?.length) {\n    await Promise.all(\n      assetFiles.map(async ({ source, destination }) => {\n        // Ensure output subdirectories exist\n        const basePath = path.dirname(destination);\n        if (basePath && !directoryExists.has(basePath)) {\n          await fs.mkdir(path.join(outputPath, basePath), { recursive: true });\n          directoryExists.add(basePath);\n        }\n        // Copy file contents\n        await fs.copyFile(source, path.join(outputPath, destination), fsConstants.COPYFILE_FICLONE);\n      }),\n    );\n  }\n}\n\nexport function createOutputFileFromText(path: string, text: string): OutputFile {\n  return {\n    path,\n    text,\n    get hash() {\n      return createHash('sha256').update(this.text).digest('hex');\n    },\n    get contents() {\n      return Buffer.from(this.text, 'utf-8');\n    },\n  };\n}\n\n/**\n * Transform browserlists result to esbuild target.\n * @see https://esbuild.github.io/api/#target\n */\nexport function transformSupportedBrowsersToTargets(supportedBrowsers: string[]): string[] {\n  const transformed: string[] = [];\n\n  // https://esbuild.github.io/api/#target\n  const esBuildSupportedBrowsers = new Set([\n    'chrome',\n    'edge',\n    'firefox',\n    'ie',\n    'ios',\n    'node',\n    'opera',\n    'safari',\n  ]);\n\n  for (const browser of supportedBrowsers) {\n    let [browserName, version] = browser.toLowerCase().split(' ');\n\n    // browserslist uses the name `ios_saf` for iOS Safari whereas esbuild uses `ios`\n    if (browserName === 'ios_saf') {\n      browserName = 'ios';\n    }\n\n    // browserslist uses ranges `15.2-15.3` versions but only the lowest is required\n    // to perform minimum supported feature checks. esbuild also expects a single version.\n    [version] = version.split('-');\n\n    if (esBuildSupportedBrowsers.has(browserName)) {\n      if (browserName === 'safari' && version === 'tp') {\n        // esbuild only supports numeric versions so `TP` is converted to a high number (999) since\n        // a Technology Preview (TP) of Safari is assumed to support all currently known features.\n        version = '999';\n      } else if (!version.includes('.')) {\n        // A lone major version is considered by esbuild to include all minor versions. However,\n        // browserslist does not and is also inconsistent in its `.0` version naming. For example,\n        // Safari 15.0 is named `safari 15` but Safari 16.0 is named `safari 16.0`.\n        version += '.0';\n      }\n\n      transformed.push(browserName + version);\n    }\n  }\n\n  return transformed;\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/utils.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 { BuildOptions, Metafile, OutputFile, PartialMessage, formatMessages } from 'esbuild';\nimport { createHash } from 'node:crypto';\nimport { constants as fsConstants } from 'node:fs';\nimport fs from 'node:fs/promises';\nimport path from 'node:path';\nimport { promisify } from 'node:util';\nimport { brotliCompress } from 'node:zlib';\nimport { Spinner } from '../../utils/spinner';\nimport { BundleStats, generateBuildStatsTable } from '../webpack/utils/stats';\nimport { InitialFileRecord } from './bundler-context';\n\nconst compressAsync = promisify(brotliCompress);\n\nexport function logBuildStats(\n  context: BuilderContext,\n  metafile: Metafile,\n  initial: Map<string, InitialFileRecord>,\n  estimatedTransferSizes?: Map<string, number>,\n): void {\n  const stats: BundleStats[] = [];\n  for (const [file, output] of Object.entries(metafile.outputs)) {\n    // Only display JavaScript and CSS files\n    if (!file.endsWith('.js') && !file.endsWith('.css')) {\n      continue;\n    }\n    // Skip internal component resources\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\n    if ((output as any)['ng-component']) {\n      continue;\n    }\n\n    stats.push({\n      initial: initial.has(file),\n      stats: [\n        file,\n        initial.get(file)?.name ?? '-',\n        output.bytes,\n        estimatedTransferSizes?.get(file) ?? '-',\n      ],\n    });\n  }\n\n  const tableText = generateBuildStatsTable(stats, true, true, !!estimatedTransferSizes, undefined);\n\n  context.logger.info('\\n' + tableText + '\\n');\n}\n\nexport async function calculateEstimatedTransferSizes(\n  outputFiles: OutputFile[],\n): Promise<Map<string, number>> {\n  const sizes = new Map<string, number>();\n  const pendingCompression = [];\n\n  for (const outputFile of outputFiles) {\n    // Only calculate JavaScript and CSS files\n    if (!outputFile.path.endsWith('.js') && !outputFile.path.endsWith('.css')) {\n      continue;\n    }\n\n    // Skip compressing small files which may end being larger once compressed and will most likely not be\n    // compressed in actual transit.\n    if (outputFile.contents.byteLength < 1024) {\n      sizes.set(outputFile.path, outputFile.contents.byteLength);\n      continue;\n    }\n\n    pendingCompression.push(\n      compressAsync(outputFile.contents).then((result) =>\n        sizes.set(outputFile.path, result.byteLength),\n      ),\n    );\n  }\n\n  await Promise.all(pendingCompression);\n\n  return sizes;\n}\n\nexport async function withSpinner<T>(text: string, action: () => T | Promise<T>): Promise<T> {\n  const spinner = new Spinner(text);\n  spinner.start();\n\n  try {\n    return await action();\n  } finally {\n    spinner.stop();\n  }\n}\n\nexport async function withNoProgress<T>(test: string, action: () => T | Promise<T>): Promise<T> {\n  return action();\n}\n\nexport async function logMessages(\n  context: BuilderContext,\n  { errors, warnings }: { errors?: PartialMessage[]; warnings?: PartialMessage[] },\n): Promise<void> {\n  if (warnings?.length) {\n    const warningMessages = await formatMessages(warnings, { kind: 'warning', color: true });\n    context.logger.warn(warningMessages.join('\\n'));\n  }\n\n  if (errors?.length) {\n    const errorMessages = await formatMessages(errors, { kind: 'error', color: true });\n    context.logger.error(errorMessages.join('\\n'));\n  }\n}\n\n/**\n * Generates a syntax feature object map for Angular applications based on a list of targets.\n * A full set of feature names can be found here: https://esbuild.github.io/api/#supported\n * @param target An array of browser/engine targets in the format accepted by the esbuild `target` option.\n * @returns An object that can be used with the esbuild build `supported` option.\n */\nexport function getFeatureSupport(target: string[]): BuildOptions['supported'] {\n  const supported: Record<string, boolean> = {\n    // Native async/await is not supported with Zone.js. Disabling support here will cause\n    // esbuild to downlevel async/await, async generators, and for await...of to a Zone.js supported form.\n    'async-await': false,\n    // V8 currently has a performance defect involving object spread operations that can cause signficant\n    // degradation in runtime performance. By not supporting the language feature here, a downlevel form\n    // will be used instead which provides a workaround for the performance issue.\n    // For more details: https://bugs.chromium.org/p/v8/issues/detail?id=11536\n    'object-rest-spread': false,\n  };\n\n  // Detect Safari browser versions that have a class field behavior bug\n  // See: https://github.com/angular/angular-cli/issues/24355#issuecomment-1333477033\n  // See: https://github.com/WebKit/WebKit/commit/e8788a34b3d5f5b4edd7ff6450b80936bff396f2\n  let safariClassFieldScopeBug = false;\n  for (const browser of target) {\n    let majorVersion;\n    if (browser.startsWith('ios')) {\n      majorVersion = Number(browser.slice(3, 5));\n    } else if (browser.startsWith('safari')) {\n      majorVersion = Number(browser.slice(6, 8));\n    } else {\n      continue;\n    }\n    // Technically, 14.0 is not broken but rather does not have support. However, the behavior\n    // is identical since it would be set to false by esbuild if present as a target.\n    if (majorVersion === 14 || majorVersion === 15) {\n      safariClassFieldScopeBug = true;\n      break;\n    }\n  }\n  // If class field support cannot be used set to false; otherwise leave undefined to allow\n  // esbuild to use `target` to determine support.\n  if (safariClassFieldScopeBug) {\n    supported['class-field'] = false;\n    supported['class-static-field'] = false;\n  }\n\n  return supported;\n}\n\nexport async function writeResultFiles(\n  outputFiles: OutputFile[],\n  assetFiles: { source: string; destination: string }[] | undefined,\n  outputPath: string,\n) {\n  const directoryExists = new Set<string>();\n  await Promise.all(\n    outputFiles.map(async (file) => {\n      // Ensure output subdirectories exist\n      const basePath = path.dirname(file.path);\n      if (basePath && !directoryExists.has(basePath)) {\n        await fs.mkdir(path.join(outputPath, basePath), { recursive: true });\n        directoryExists.add(basePath);\n      }\n      // Write file contents\n      await fs.writeFile(path.join(outputPath, file.path), file.contents);\n    }),\n  );\n\n  if (assetFiles?.length) {\n    await Promise.all(\n      assetFiles.map(async ({ source, destination }) => {\n        // Ensure output subdirectories exist\n        const basePath = path.dirname(destination);\n        if (basePath && !directoryExists.has(basePath)) {\n          await fs.mkdir(path.join(outputPath, basePath), { recursive: true });\n          directoryExists.add(basePath);\n        }\n        // Copy file contents\n        await fs.copyFile(source, path.join(outputPath, destination), fsConstants.COPYFILE_FICLONE);\n      }),\n    );\n  }\n}\n\nexport function createOutputFileFromText(path: string, text: string): OutputFile {\n  return {\n    path,\n    text,\n    get hash() {\n      return createHash('sha256').update(this.text).digest('hex');\n    },\n    get contents() {\n      return Buffer.from(this.text, 'utf-8');\n    },\n  };\n}\n\nexport function createOutputFileFromData(path: string, data: Uint8Array): OutputFile {\n  return {\n    path,\n    get text() {\n      return Buffer.from(data.buffer, data.byteOffset, data.byteLength).toString('utf-8');\n    },\n    get hash() {\n      return createHash('sha256').update(data).digest('hex');\n    },\n    get contents() {\n      return data;\n    },\n  };\n}\n\nexport function cloneOutputFile(file: OutputFile): OutputFile {\n  const path = file.path;\n  const data = file.contents;\n\n  return {\n    path,\n    get text() {\n      return Buffer.from(data.buffer, data.byteOffset, data.byteLength).toString('utf-8');\n    },\n    get hash() {\n      return createHash('sha256').update(data).digest('hex');\n    },\n    get contents() {\n      return data;\n    },\n  };\n}\n\n/**\n * Transform browserlists result to esbuild target.\n * @see https://esbuild.github.io/api/#target\n */\nexport function transformSupportedBrowsersToTargets(supportedBrowsers: string[]): string[] {\n  const transformed: string[] = [];\n\n  // https://esbuild.github.io/api/#target\n  const esBuildSupportedBrowsers = new Set([\n    'chrome',\n    'edge',\n    'firefox',\n    'ie',\n    'ios',\n    'node',\n    'opera',\n    'safari',\n  ]);\n\n  for (const browser of supportedBrowsers) {\n    let [browserName, version] = browser.toLowerCase().split(' ');\n\n    // browserslist uses the name `ios_saf` for iOS Safari whereas esbuild uses `ios`\n    if (browserName === 'ios_saf') {\n      browserName = 'ios';\n    }\n\n    // browserslist uses ranges `15.2-15.3` versions but only the lowest is required\n    // to perform minimum supported feature checks. esbuild also expects a single version.\n    [version] = version.split('-');\n\n    if (esBuildSupportedBrowsers.has(browserName)) {\n      if (browserName === 'safari' && version === 'tp') {\n        // esbuild only supports numeric versions so `TP` is converted to a high number (999) since\n        // a Technology Preview (TP) of Safari is assumed to support all currently known features.\n        version = '999';\n      } else if (!version.includes('.')) {\n        // A lone major version is considered by esbuild to include all minor versions. However,\n        // browserslist does not and is also inconsistent in its `.0` version naming. For example,\n        // Safari 15.0 is named `safari 15` but Safari 16.0 is named `safari 16.0`.\n        version += '.0';\n      }\n\n      transformed.push(browserName + version);\n    }\n  }\n\n  return transformed;\n}\n"]} \ No newline at end of file diff --git a/src/tools/esbuild/virtual-module-plugin.d.ts b/src/tools/esbuild/virtual-module-plugin.d.ts index ee6d5b03..80840580 100644 --- a/src/tools/esbuild/virtual-module-plugin.d.ts +++ b/src/tools/esbuild/virtual-module-plugin.d.ts @@ -19,6 +19,8 @@ export interface VirtualModulePluginOptions { transformPath?: (path: string) => string; /** Method to provide the module content. */ loadContent: (args: OnLoadArgs, build: PluginBuild) => ReturnType[1]>; + /** Restrict to only entry points. Defaults to `true`. */ + entryPointOnly?: boolean; } /** * Creates an esbuild plugin that generated virtual modules. diff --git a/src/tools/esbuild/virtual-module-plugin.js b/src/tools/esbuild/virtual-module-plugin.js index f8a9b433..9fa023df 100644 --- a/src/tools/esbuild/virtual-module-plugin.js +++ b/src/tools/esbuild/virtual-module-plugin.js @@ -14,12 +14,12 @@ exports.createVirtualModulePlugin = void 0; * @returns An esbuild plugin. */ function createVirtualModulePlugin(options) { - const { namespace, external, transformPath: pathTransformer, loadContent } = options; + const { namespace, external, transformPath: pathTransformer, loadContent, entryPointOnly = true, } = options; return { name: namespace.replace(/[/:]/g, '-'), setup(build) { build.onResolve({ filter: new RegExp('^' + namespace) }, ({ kind, path }) => { - if (kind !== 'entry-point') { + if (entryPointOnly && kind !== 'entry-point') { return null; } return { @@ -40,4 +40,4 @@ function createVirtualModulePlugin(options) { }; } exports.createVirtualModulePlugin = createVirtualModulePlugin; -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmlydHVhbC1tb2R1bGUtcGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC92aXJ0dWFsLW1vZHVsZS1wbHVnaW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBc0JIOzs7O0dBSUc7QUFDSCxTQUFnQix5QkFBeUIsQ0FBQyxPQUFtQztJQUMzRSxNQUFNLEVBQUUsU0FBUyxFQUFFLFFBQVEsRUFBRSxhQUFhLEVBQUUsZUFBZSxFQUFFLFdBQVcsRUFBRSxHQUFHLE9BQU8sQ0FBQztJQUVyRixPQUFPO1FBQ0wsSUFBSSxFQUFFLFNBQVMsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLEdBQUcsQ0FBQztRQUNyQyxLQUFLLENBQUMsS0FBSztZQUNULEtBQUssQ0FBQyxTQUFTLENBQUMsRUFBRSxNQUFNLEVBQUUsSUFBSSxNQUFNLENBQUMsR0FBRyxHQUFHLFNBQVMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsRUFBRSxFQUFFO2dCQUMxRSxJQUFJLElBQUksS0FBSyxhQUFhLEVBQUU7b0JBQzFCLE9BQU8sSUFBSSxDQUFDO2lCQUNiO2dCQUVELE9BQU87b0JBQ0wsSUFBSSxFQUFFLGVBQWUsRUFBRSxDQUFDLElBQUksQ0FBQyxJQUFJLElBQUk7b0JBQ3JDLFNBQVM7aUJBQ1YsQ0FBQztZQUNKLENBQUMsQ0FBQyxDQUFDO1lBRUgsSUFBSSxRQUFRLEVBQUU7Z0JBQ1osS0FBSyxDQUFDLFNBQVMsQ0FBQyxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxFQUFFLEVBQUU7b0JBQ3ZELE9BQU87d0JBQ0wsSUFBSTt3QkFDSixRQUFRLEVBQUUsSUFBSTtxQkFDZixDQUFDO2dCQUNKLENBQUMsQ0FBQyxDQUFDO2FBQ0o7WUFFRCxLQUFLLENBQUMsTUFBTSxDQUFDLEVBQUUsTUFBTSxFQUFFLEdBQUcsRUFBRSxTQUFTLEVBQUUsRUFBRSxDQUFDLElBQUksRUFBRSxFQUFFLENBQUMsV0FBVyxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQy9FLENBQUM7S0FDRixDQUFDO0FBQ0osQ0FBQztBQTdCRCw4REE2QkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBPbkxvYWRBcmdzLCBQbHVnaW4sIFBsdWdpbkJ1aWxkIH0gZnJvbSAnZXNidWlsZCc7XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgdGhlIGNyZWF0ZVZpcnR1YWxNb2R1bGVQbHVnaW5cbiAqIEBzZWUgY3JlYXRlVmlydHVhbE1vZHVsZVBsdWdpblxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZpcnR1YWxNb2R1bGVQbHVnaW5PcHRpb25zIHtcbiAgLyoqIE5hbWVzcGFjZS4gRXhhbXBsZTogYGFuZ3VsYXI6cG9seWZpbGxzYC4gKi9cbiAgbmFtZXNwYWNlOiBzdHJpbmc7XG4gIC8qKiBJZiB0aGUgZ2VuZXJhdGVkIG1vZHVsZSBzaG91bGQgYmUgbWFya2VkIGFzIGV4dGVybmFsLiAqL1xuICBleHRlcm5hbD86IGJvb2xlYW47XG4gIC8qKiBNZXRob2QgdG8gdHJhbnNmb3JtIHRoZSBvblJlc29sdmUgcGF0aC4gKi9cbiAgdHJhbnNmb3JtUGF0aD86IChwYXRoOiBzdHJpbmcpID0+IHN0cmluZztcbiAgLyoqIE1ldGhvZCB0byBwcm92aWRlIHRoZSBtb2R1bGUgY29udGVudC4gKi9cbiAgbG9hZENvbnRlbnQ6IChcbiAgICBhcmdzOiBPbkxvYWRBcmdzLFxuICAgIGJ1aWxkOiBQbHVnaW5CdWlsZCxcbiAgKSA9PiBSZXR1cm5UeXBlPFBhcmFtZXRlcnM8UGx1Z2luQnVpbGRbJ29uTG9hZCddPlsxXT47XG59XG5cbi8qKlxuICogQ3JlYXRlcyBhbiBlc2J1aWxkIHBsdWdpbiB0aGF0IGdlbmVyYXRlZCB2aXJ0dWFsIG1vZHVsZXMuXG4gKlxuICogQHJldHVybnMgQW4gZXNidWlsZCBwbHVnaW4uXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVWaXJ0dWFsTW9kdWxlUGx1Z2luKG9wdGlvbnM6IFZpcnR1YWxNb2R1bGVQbHVnaW5PcHRpb25zKTogUGx1Z2luIHtcbiAgY29uc3QgeyBuYW1lc3BhY2UsIGV4dGVybmFsLCB0cmFuc2Zvcm1QYXRoOiBwYXRoVHJhbnNmb3JtZXIsIGxvYWRDb250ZW50IH0gPSBvcHRpb25zO1xuXG4gIHJldHVybiB7XG4gICAgbmFtZTogbmFtZXNwYWNlLnJlcGxhY2UoL1svOl0vZywgJy0nKSxcbiAgICBzZXR1cChidWlsZCk6IHZvaWQge1xuICAgICAgYnVpbGQub25SZXNvbHZlKHsgZmlsdGVyOiBuZXcgUmVnRXhwKCdeJyArIG5hbWVzcGFjZSkgfSwgKHsga2luZCwgcGF0aCB9KSA9PiB7XG4gICAgICAgIGlmIChraW5kICE9PSAnZW50cnktcG9pbnQnKSB7XG4gICAgICAgICAgcmV0dXJuIG51bGw7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHBhdGg6IHBhdGhUcmFuc2Zvcm1lcj8uKHBhdGgpID8/IHBhdGgsXG4gICAgICAgICAgbmFtZXNwYWNlLFxuICAgICAgICB9O1xuICAgICAgfSk7XG5cbiAgICAgIGlmIChleHRlcm5hbCkge1xuICAgICAgICBidWlsZC5vblJlc29sdmUoeyBmaWx0ZXI6IC8uLywgbmFtZXNwYWNlIH0sICh7IHBhdGggfSkgPT4ge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBwYXRoLFxuICAgICAgICAgICAgZXh0ZXJuYWw6IHRydWUsXG4gICAgICAgICAgfTtcbiAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIGJ1aWxkLm9uTG9hZCh7IGZpbHRlcjogLy4vLCBuYW1lc3BhY2UgfSwgKGFyZ3MpID0+IGxvYWRDb250ZW50KGFyZ3MsIGJ1aWxkKSk7XG4gICAgfSxcbiAgfTtcbn1cbiJdfQ== \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmlydHVhbC1tb2R1bGUtcGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvZXNidWlsZC92aXJ0dWFsLW1vZHVsZS1wbHVnaW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBd0JIOzs7O0dBSUc7QUFDSCxTQUFnQix5QkFBeUIsQ0FBQyxPQUFtQztJQUMzRSxNQUFNLEVBQ0osU0FBUyxFQUNULFFBQVEsRUFDUixhQUFhLEVBQUUsZUFBZSxFQUM5QixXQUFXLEVBQ1gsY0FBYyxHQUFHLElBQUksR0FDdEIsR0FBRyxPQUFPLENBQUM7SUFFWixPQUFPO1FBQ0wsSUFBSSxFQUFFLFNBQVMsQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLEdBQUcsQ0FBQztRQUNyQyxLQUFLLENBQUMsS0FBSztZQUNULEtBQUssQ0FBQyxTQUFTLENBQUMsRUFBRSxNQUFNLEVBQUUsSUFBSSxNQUFNLENBQUMsR0FBRyxHQUFHLFNBQVMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsRUFBRSxFQUFFO2dCQUMxRSxJQUFJLGNBQWMsSUFBSSxJQUFJLEtBQUssYUFBYSxFQUFFO29CQUM1QyxPQUFPLElBQUksQ0FBQztpQkFDYjtnQkFFRCxPQUFPO29CQUNMLElBQUksRUFBRSxlQUFlLEVBQUUsQ0FBQyxJQUFJLENBQUMsSUFBSSxJQUFJO29CQUNyQyxTQUFTO2lCQUNWLENBQUM7WUFDSixDQUFDLENBQUMsQ0FBQztZQUVILElBQUksUUFBUSxFQUFFO2dCQUNaLEtBQUssQ0FBQyxTQUFTLENBQUMsRUFBRSxNQUFNLEVBQUUsR0FBRyxFQUFFLFNBQVMsRUFBRSxFQUFFLENBQUMsRUFBRSxJQUFJLEVBQUUsRUFBRSxFQUFFO29CQUN2RCxPQUFPO3dCQUNMLElBQUk7d0JBQ0osUUFBUSxFQUFFLElBQUk7cUJBQ2YsQ0FBQztnQkFDSixDQUFDLENBQUMsQ0FBQzthQUNKO1lBRUQsS0FBSyxDQUFDLE1BQU0sQ0FBQyxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsU0FBUyxFQUFFLEVBQUUsQ0FBQyxJQUFJLEVBQUUsRUFBRSxDQUFDLFdBQVcsQ0FBQyxJQUFJLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUMvRSxDQUFDO0tBQ0YsQ0FBQztBQUNKLENBQUM7QUFuQ0QsOERBbUNDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbmltcG9ydCB0eXBlIHsgT25Mb2FkQXJncywgUGx1Z2luLCBQbHVnaW5CdWlsZCB9IGZyb20gJ2VzYnVpbGQnO1xuXG4vKipcbiAqIE9wdGlvbnMgZm9yIHRoZSBjcmVhdGVWaXJ0dWFsTW9kdWxlUGx1Z2luXG4gKiBAc2VlIGNyZWF0ZVZpcnR1YWxNb2R1bGVQbHVnaW5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBWaXJ0dWFsTW9kdWxlUGx1Z2luT3B0aW9ucyB7XG4gIC8qKiBOYW1lc3BhY2UuIEV4YW1wbGU6IGBhbmd1bGFyOnBvbHlmaWxsc2AuICovXG4gIG5hbWVzcGFjZTogc3RyaW5nO1xuICAvKiogSWYgdGhlIGdlbmVyYXRlZCBtb2R1bGUgc2hvdWxkIGJlIG1hcmtlZCBhcyBleHRlcm5hbC4gKi9cbiAgZXh0ZXJuYWw/OiBib29sZWFuO1xuICAvKiogTWV0aG9kIHRvIHRyYW5zZm9ybSB0aGUgb25SZXNvbHZlIHBhdGguICovXG4gIHRyYW5zZm9ybVBhdGg/OiAocGF0aDogc3RyaW5nKSA9PiBzdHJpbmc7XG4gIC8qKiBNZXRob2QgdG8gcHJvdmlkZSB0aGUgbW9kdWxlIGNvbnRlbnQuICovXG4gIGxvYWRDb250ZW50OiAoXG4gICAgYXJnczogT25Mb2FkQXJncyxcbiAgICBidWlsZDogUGx1Z2luQnVpbGQsXG4gICkgPT4gUmV0dXJuVHlwZTxQYXJhbWV0ZXJzPFBsdWdpbkJ1aWxkWydvbkxvYWQnXT5bMV0+O1xuICAvKiogUmVzdHJpY3QgdG8gb25seSBlbnRyeSBwb2ludHMuIERlZmF1bHRzIHRvIGB0cnVlYC4gKi9cbiAgZW50cnlQb2ludE9ubHk/OiBib29sZWFuO1xufVxuXG4vKipcbiAqIENyZWF0ZXMgYW4gZXNidWlsZCBwbHVnaW4gdGhhdCBnZW5lcmF0ZWQgdmlydHVhbCBtb2R1bGVzLlxuICpcbiAqIEByZXR1cm5zIEFuIGVzYnVpbGQgcGx1Z2luLlxuICovXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlVmlydHVhbE1vZHVsZVBsdWdpbihvcHRpb25zOiBWaXJ0dWFsTW9kdWxlUGx1Z2luT3B0aW9ucyk6IFBsdWdpbiB7XG4gIGNvbnN0IHtcbiAgICBuYW1lc3BhY2UsXG4gICAgZXh0ZXJuYWwsXG4gICAgdHJhbnNmb3JtUGF0aDogcGF0aFRyYW5zZm9ybWVyLFxuICAgIGxvYWRDb250ZW50LFxuICAgIGVudHJ5UG9pbnRPbmx5ID0gdHJ1ZSxcbiAgfSA9IG9wdGlvbnM7XG5cbiAgcmV0dXJuIHtcbiAgICBuYW1lOiBuYW1lc3BhY2UucmVwbGFjZSgvWy86XS9nLCAnLScpLFxuICAgIHNldHVwKGJ1aWxkKTogdm9pZCB7XG4gICAgICBidWlsZC5vblJlc29sdmUoeyBmaWx0ZXI6IG5ldyBSZWdFeHAoJ14nICsgbmFtZXNwYWNlKSB9LCAoeyBraW5kLCBwYXRoIH0pID0+IHtcbiAgICAgICAgaWYgKGVudHJ5UG9pbnRPbmx5ICYmIGtpbmQgIT09ICdlbnRyeS1wb2ludCcpIHtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgcGF0aDogcGF0aFRyYW5zZm9ybWVyPy4ocGF0aCkgPz8gcGF0aCxcbiAgICAgICAgICBuYW1lc3BhY2UsXG4gICAgICAgIH07XG4gICAgICB9KTtcblxuICAgICAgaWYgKGV4dGVybmFsKSB7XG4gICAgICAgIGJ1aWxkLm9uUmVzb2x2ZSh7IGZpbHRlcjogLy4vLCBuYW1lc3BhY2UgfSwgKHsgcGF0aCB9KSA9PiB7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIHBhdGgsXG4gICAgICAgICAgICBleHRlcm5hbDogdHJ1ZSxcbiAgICAgICAgICB9O1xuICAgICAgICB9KTtcbiAgICAgIH1cblxuICAgICAgYnVpbGQub25Mb2FkKHsgZmlsdGVyOiAvLi8sIG5hbWVzcGFjZSB9LCAoYXJncykgPT4gbG9hZENvbnRlbnQoYXJncywgYnVpbGQpKTtcbiAgICB9LFxuICB9O1xufVxuIl19 \ No newline at end of file diff --git a/src/tools/vite/i18n-locale-plugin.d.ts b/src/tools/vite/i18n-locale-plugin.d.ts new file mode 100644 index 00000000..09df59cc --- /dev/null +++ b/src/tools/vite/i18n-locale-plugin.d.ts @@ -0,0 +1,18 @@ +/** + * @license + * Copyright Google LLC All Rights Reserved. + * + * Use of this source code is governed by an MIT-style license that can be + * found in the LICENSE file at https://angular.io/license + */ +import type { Plugin } from 'vite'; +/** + * The base module location used to search for locale specific data. + */ +export declare const LOCALE_DATA_BASE_MODULE = "@angular/common/locales/global"; +/** + * Creates a Vite plugin that resolves Angular locale data files from `@angular/common`. + * + * @returns A Vite plugin. + */ +export declare function createAngularLocaleDataPlugin(): Plugin; diff --git a/src/tools/vite/i18n-locale-plugin.js b/src/tools/vite/i18n-locale-plugin.js new file mode 100644 index 00000000..ae7e8534 --- /dev/null +++ b/src/tools/vite/i18n-locale-plugin.js @@ -0,0 +1,56 @@ +"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 + */ +Object.defineProperty(exports, "__esModule", { value: true }); +exports.createAngularLocaleDataPlugin = exports.LOCALE_DATA_BASE_MODULE = void 0; +/** + * The base module location used to search for locale specific data. + */ +exports.LOCALE_DATA_BASE_MODULE = '@angular/common/locales/global'; +/** + * Creates a Vite plugin that resolves Angular locale data files from `@angular/common`. + * + * @returns A Vite plugin. + */ +function createAngularLocaleDataPlugin() { + return { + name: 'angular-locale-data', + enforce: 'pre', + async resolveId(source) { + if (!source.startsWith('angular:locale/data:')) { + return; + } + // Extract the locale from the path + const originalLocale = source.split(':', 3)[2]; + // Remove any private subtags since these will never match + let partialLocale = originalLocale.replace(/-x(-[a-zA-Z0-9]{1,8})+$/, ''); + let exact = true; + while (partialLocale) { + const potentialPath = `${exports.LOCALE_DATA_BASE_MODULE}/${partialLocale}`; + const result = await this.resolve(potentialPath); + if (result) { + if (!exact) { + this.warn(`Locale data for '${originalLocale}' cannot be found. Using locale data for '${partialLocale}'.`); + } + return result; + } + // Remove the last subtag and try again with a less specific locale + const parts = partialLocale.split('-'); + partialLocale = parts.slice(0, -1).join('-'); + exact = false; + // The locales "en" and "en-US" are considered exact to retain existing behavior + if (originalLocale === 'en-US' && partialLocale === 'en') { + exact = true; + } + } + return null; + }, + }; +} +exports.createAngularLocaleDataPlugin = createAngularLocaleDataPlugin; +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaTE4bi1sb2NhbGUtcGx1Z2luLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvYW5ndWxhcl9kZXZraXQvYnVpbGRfYW5ndWxhci9zcmMvdG9vbHMvdml0ZS9pMThuLWxvY2FsZS1wbHVnaW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7O0FBSUg7O0dBRUc7QUFDVSxRQUFBLHVCQUF1QixHQUFHLGdDQUFnQyxDQUFDO0FBRXhFOzs7O0dBSUc7QUFDSCxTQUFnQiw2QkFBNkI7SUFDM0MsT0FBTztRQUNMLElBQUksRUFBRSxxQkFBcUI7UUFDM0IsT0FBTyxFQUFFLEtBQUs7UUFDZCxLQUFLLENBQUMsU0FBUyxDQUFDLE1BQU07WUFDcEIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsc0JBQXNCLENBQUMsRUFBRTtnQkFDOUMsT0FBTzthQUNSO1lBRUQsbUNBQW1DO1lBQ25DLE1BQU0sY0FBYyxHQUFHLE1BQU0sQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBRS9DLDBEQUEwRDtZQUMxRCxJQUFJLGFBQWEsR0FBRyxjQUFjLENBQUMsT0FBTyxDQUFDLHlCQUF5QixFQUFFLEVBQUUsQ0FBQyxDQUFDO1lBRTFFLElBQUksS0FBSyxHQUFHLElBQUksQ0FBQztZQUNqQixPQUFPLGFBQWEsRUFBRTtnQkFDcEIsTUFBTSxhQUFhLEdBQUcsR0FBRywrQkFBdUIsSUFBSSxhQUFhLEVBQUUsQ0FBQztnQkFFcEUsTUFBTSxNQUFNLEdBQUcsTUFBTSxJQUFJLENBQUMsT0FBTyxDQUFDLGFBQWEsQ0FBQyxDQUFDO2dCQUNqRCxJQUFJLE1BQU0sRUFBRTtvQkFDVixJQUFJLENBQUMsS0FBSyxFQUFFO3dCQUNWLElBQUksQ0FBQyxJQUFJLENBQ1Asb0JBQW9CLGNBQWMsNkNBQTZDLGFBQWEsSUFBSSxDQUNqRyxDQUFDO3FCQUNIO29CQUVELE9BQU8sTUFBTSxDQUFDO2lCQUNmO2dCQUVELG1FQUFtRTtnQkFDbkUsTUFBTSxLQUFLLEdBQUcsYUFBYSxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQztnQkFDdkMsYUFBYSxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO2dCQUM3QyxLQUFLLEdBQUcsS0FBSyxDQUFDO2dCQUNkLGdGQUFnRjtnQkFDaEYsSUFBSSxjQUFjLEtBQUssT0FBTyxJQUFJLGFBQWEsS0FBSyxJQUFJLEVBQUU7b0JBQ3hELEtBQUssR0FBRyxJQUFJLENBQUM7aUJBQ2Q7YUFDRjtZQUVELE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQztLQUNGLENBQUM7QUFDSixDQUFDO0FBM0NELHNFQTJDQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQGxpY2Vuc2VcbiAqIENvcHlyaWdodCBHb29nbGUgTExDIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4gKlxuICogVXNlIG9mIHRoaXMgc291cmNlIGNvZGUgaXMgZ292ZXJuZWQgYnkgYW4gTUlULXN0eWxlIGxpY2Vuc2UgdGhhdCBjYW4gYmVcbiAqIGZvdW5kIGluIHRoZSBMSUNFTlNFIGZpbGUgYXQgaHR0cHM6Ly9hbmd1bGFyLmlvL2xpY2Vuc2VcbiAqL1xuXG5pbXBvcnQgdHlwZSB7IFBsdWdpbiB9IGZyb20gJ3ZpdGUnO1xuXG4vKipcbiAqIFRoZSBiYXNlIG1vZHVsZSBsb2NhdGlvbiB1c2VkIHRvIHNlYXJjaCBmb3IgbG9jYWxlIHNwZWNpZmljIGRhdGEuXG4gKi9cbmV4cG9ydCBjb25zdCBMT0NBTEVfREFUQV9CQVNFX01PRFVMRSA9ICdAYW5ndWxhci9jb21tb24vbG9jYWxlcy9nbG9iYWwnO1xuXG4vKipcbiAqIENyZWF0ZXMgYSBWaXRlIHBsdWdpbiB0aGF0IHJlc29sdmVzIEFuZ3VsYXIgbG9jYWxlIGRhdGEgZmlsZXMgZnJvbSBgQGFuZ3VsYXIvY29tbW9uYC5cbiAqXG4gKiBAcmV0dXJucyBBIFZpdGUgcGx1Z2luLlxuICovXG5leHBvcnQgZnVuY3Rpb24gY3JlYXRlQW5ndWxhckxvY2FsZURhdGFQbHVnaW4oKTogUGx1Z2luIHtcbiAgcmV0dXJuIHtcbiAgICBuYW1lOiAnYW5ndWxhci1sb2NhbGUtZGF0YScsXG4gICAgZW5mb3JjZTogJ3ByZScsXG4gICAgYXN5bmMgcmVzb2x2ZUlkKHNvdXJjZSkge1xuICAgICAgaWYgKCFzb3VyY2Uuc3RhcnRzV2l0aCgnYW5ndWxhcjpsb2NhbGUvZGF0YTonKSkge1xuICAgICAgICByZXR1cm47XG4gICAgICB9XG5cbiAgICAgIC8vIEV4dHJhY3QgdGhlIGxvY2FsZSBmcm9tIHRoZSBwYXRoXG4gICAgICBjb25zdCBvcmlnaW5hbExvY2FsZSA9IHNvdXJjZS5zcGxpdCgnOicsIDMpWzJdO1xuXG4gICAgICAvLyBSZW1vdmUgYW55IHByaXZhdGUgc3VidGFncyBzaW5jZSB0aGVzZSB3aWxsIG5ldmVyIG1hdGNoXG4gICAgICBsZXQgcGFydGlhbExvY2FsZSA9IG9yaWdpbmFsTG9jYWxlLnJlcGxhY2UoLy14KC1bYS16QS1aMC05XXsxLDh9KSskLywgJycpO1xuXG4gICAgICBsZXQgZXhhY3QgPSB0cnVlO1xuICAgICAgd2hpbGUgKHBhcnRpYWxMb2NhbGUpIHtcbiAgICAgICAgY29uc3QgcG90ZW50aWFsUGF0aCA9IGAke0xPQ0FMRV9EQVRBX0JBU0VfTU9EVUxFfS8ke3BhcnRpYWxMb2NhbGV9YDtcblxuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLnJlc29sdmUocG90ZW50aWFsUGF0aCk7XG4gICAgICAgIGlmIChyZXN1bHQpIHtcbiAgICAgICAgICBpZiAoIWV4YWN0KSB7XG4gICAgICAgICAgICB0aGlzLndhcm4oXG4gICAgICAgICAgICAgIGBMb2NhbGUgZGF0YSBmb3IgJyR7b3JpZ2luYWxMb2NhbGV9JyBjYW5ub3QgYmUgZm91bmQuIFVzaW5nIGxvY2FsZSBkYXRhIGZvciAnJHtwYXJ0aWFsTG9jYWxlfScuYCxcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFJlbW92ZSB0aGUgbGFzdCBzdWJ0YWcgYW5kIHRyeSBhZ2FpbiB3aXRoIGEgbGVzcyBzcGVjaWZpYyBsb2NhbGVcbiAgICAgICAgY29uc3QgcGFydHMgPSBwYXJ0aWFsTG9jYWxlLnNwbGl0KCctJyk7XG4gICAgICAgIHBhcnRpYWxMb2NhbGUgPSBwYXJ0cy5zbGljZSgwLCAtMSkuam9pbignLScpO1xuICAgICAgICBleGFjdCA9IGZhbHNlO1xuICAgICAgICAvLyBUaGUgbG9jYWxlcyBcImVuXCIgYW5kIFwiZW4tVVNcIiBhcmUgY29uc2lkZXJlZCBleGFjdCB0byByZXRhaW4gZXhpc3RpbmcgYmVoYXZpb3JcbiAgICAgICAgaWYgKG9yaWdpbmFsTG9jYWxlID09PSAnZW4tVVMnICYmIHBhcnRpYWxMb2NhbGUgPT09ICdlbicpIHtcbiAgICAgICAgICBleGFjdCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfSxcbiAgfTtcbn1cbiJdfQ== \ No newline at end of file diff --git a/uniqueId b/uniqueId index 8069f5d6..f831540a 100644 --- a/uniqueId +++ b/uniqueId @@ -1 +1 @@ -Thu Sep 21 2023 14:20:08 GMT+0000 (Coordinated Universal Time) \ No newline at end of file +Fri Sep 22 2023 08:56:27 GMT+0000 (Coordinated Universal Time) \ No newline at end of file