@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
328 lines • 51.4 kB
JavaScript
"use strict";
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
__setModuleDefault(result, mod);
return result;
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.createCompilerPlugin = exports.SourceFileCache = void 0;
const promises_1 = require("node:fs/promises");
const node_os_1 = require("node:os");
const path = __importStar(require("node:path"));
const node_url_1 = require("node:url");
const typescript_1 = __importDefault(require("typescript"));
const environment_options_1 = require("../../../utils/environment-options");
const javascript_transformer_1 = require("../javascript-transformer");
const load_result_cache_1 = require("../load-result-cache");
const profiling_1 = require("../profiling");
const bundle_options_1 = require("../stylesheets/bundle-options");
const compilation_1 = require("./compilation");
const diagnostics_1 = require("./diagnostics");
const jit_plugin_callbacks_1 = require("./jit-plugin-callbacks");
const USING_WINDOWS = (0, node_os_1.platform)() === 'win32';
const WINDOWS_SEP_REGEXP = new RegExp(`\\${path.win32.sep}`, 'g');
class SourceFileCache extends Map {
constructor(persistentCachePath) {
super();
this.persistentCachePath = persistentCachePath;
this.modifiedFiles = new Set();
this.babelFileCache = new Map();
this.typeScriptFileCache = new Map();
this.loadResultCache = new load_result_cache_1.MemoryLoadResultCache();
}
invalidate(files) {
this.modifiedFiles.clear();
for (let file of files) {
this.babelFileCache.delete(file);
this.typeScriptFileCache.delete((0, node_url_1.pathToFileURL)(file).href);
this.loadResultCache.invalidate(file);
// Normalize separators to allow matching TypeScript Host paths
if (USING_WINDOWS) {
file = file.replace(WINDOWS_SEP_REGEXP, path.posix.sep);
}
this.delete(file);
this.modifiedFiles.add(file);
}
}
}
exports.SourceFileCache = SourceFileCache;
// TODO: find a better way to unblock TS compilation of server bundles.
let TS_COMPILATION_READY;
// eslint-disable-next-line max-lines-per-function
function createCompilerPlugin(pluginOptions, styleOptions) {
let resolveCompilationReady;
if (!pluginOptions.noopTypeScriptCompilation) {
TS_COMPILATION_READY = new Promise((resolve) => {
resolveCompilationReady = resolve;
});
}
return {
name: 'angular-compiler',
// eslint-disable-next-line max-lines-per-function
async setup(build) {
var _a;
let setupWarnings = [];
const preserveSymlinks = build.initialOptions.preserveSymlinks;
let tsconfigPath = pluginOptions.tsconfig;
if (!preserveSymlinks) {
// Use the real path of the tsconfig if not preserving symlinks.
// This ensures the TS source file paths are based on the real path of the configuration.
try {
tsconfigPath = await (0, promises_1.realpath)(tsconfigPath);
}
catch { }
}
// Initialize a worker pool for JavaScript transformations
const javascriptTransformer = new javascript_transformer_1.JavaScriptTransformer(pluginOptions, environment_options_1.maxWorkers);
// Setup defines based on the values provided by the Angular compiler-cli
const { GLOBAL_DEFS_FOR_TERSER_WITH_AOT } = await compilation_1.AngularCompilation.loadCompilerCli();
(_a = build.initialOptions).define ?? (_a.define = {});
for (const [key, value] of Object.entries(GLOBAL_DEFS_FOR_TERSER_WITH_AOT)) {
if (key in build.initialOptions.define) {
// Skip keys that have been manually provided
continue;
}
if (key === 'ngDevMode') {
// ngDevMode is already set based on the builder's script optimization option
continue;
}
// esbuild requires values to be a string (actual strings need to be quoted).
// In this case, all provided values are booleans.
build.initialOptions.define[key] = value.toString();
}
// The in-memory cache of TypeScript file outputs will be used during the build in `onLoad` callbacks for TS files.
// A string value indicates direct TS/NG output and a Uint8Array indicates fully transformed code.
const typeScriptFileCache = pluginOptions.sourceFileCache?.typeScriptFileCache ??
new Map();
// The stylesheet resources from component stylesheets that will be added to the build results output files
let stylesheetResourceFiles = [];
let stylesheetMetafiles;
// Create new reusable compilation for the appropriate mode based on the `jit` plugin option
const compilation = pluginOptions.noopTypeScriptCompilation
? new compilation_1.NoopCompilation()
: pluginOptions.jit
? new compilation_1.JitCompilation()
: new compilation_1.AotCompilation();
// Determines if TypeScript should process JavaScript files based on tsconfig `allowJs` option
let shouldTsIgnoreJs = true;
build.onStart(async () => {
const result = {
warnings: setupWarnings,
};
// Reset debug performance tracking
(0, profiling_1.resetCumulativeDurations)();
// Reset stylesheet resource output files
stylesheetResourceFiles = [];
stylesheetMetafiles = [];
// Create Angular compiler host options
const hostOptions = {
fileReplacements: pluginOptions.fileReplacements,
modifiedFiles: pluginOptions.sourceFileCache?.modifiedFiles,
sourceFileCache: pluginOptions.sourceFileCache,
async transformStylesheet(data, containingFile, stylesheetFile) {
// Stylesheet file only exists for external stylesheets
const filename = stylesheetFile ?? containingFile;
const stylesheetResult = await (0, bundle_options_1.bundleComponentStylesheet)(styleOptions.inlineStyleLanguage, data, filename, !stylesheetFile, styleOptions, pluginOptions.loadResultCache);
const { contents, resourceFiles, errors, warnings } = stylesheetResult;
if (errors) {
(result.errors ?? (result.errors = [])).push(...errors);
}
(result.warnings ?? (result.warnings = [])).push(...warnings);
stylesheetResourceFiles.push(...resourceFiles);
if (stylesheetResult.metafile) {
stylesheetMetafiles.push(stylesheetResult.metafile);
}
return contents;
},
};
// Initialize the Angular compilation for the current build.
// In watch mode, previous build state will be reused.
const { compilerOptions: { allowJs }, referencedFiles, } = await compilation.initialize(tsconfigPath, hostOptions, (compilerOptions) => {
if (compilerOptions.target === undefined ||
compilerOptions.target < typescript_1.default.ScriptTarget.ES2022) {
// If 'useDefineForClassFields' is already defined in the users project leave the value as is.
// Otherwise fallback to false due to https://github.com/microsoft/TypeScript/issues/45995
// which breaks the deprecated `@Effects` NGRX decorator and potentially other existing code as well.
compilerOptions.target = typescript_1.default.ScriptTarget.ES2022;
compilerOptions.useDefineForClassFields ?? (compilerOptions.useDefineForClassFields = false);
// Only add the warning on the initial build
setupWarnings?.push({
text: 'TypeScript compiler options "target" and "useDefineForClassFields" are set to "ES2022" and ' +
'"false" respectively by the Angular CLI.',
location: { file: pluginOptions.tsconfig },
notes: [
{
text: 'To control ECMA version and features use the Browerslist configuration. ' +
'For more information, see https://angular.io/guide/build#configuring-browser-compatibility',
},
],
});
}
// Enable incremental compilation by default if caching is enabled
if (pluginOptions.sourceFileCache?.persistentCachePath) {
compilerOptions.incremental ?? (compilerOptions.incremental = true);
// Set the build info file location to the configured cache directory
compilerOptions.tsBuildInfoFile = path.join(pluginOptions.sourceFileCache?.persistentCachePath, '.tsbuildinfo');
}
else {
compilerOptions.incremental = false;
}
return {
...compilerOptions,
noEmitOnError: false,
inlineSources: pluginOptions.sourcemap,
inlineSourceMap: pluginOptions.sourcemap,
mapRoot: undefined,
sourceRoot: undefined,
preserveSymlinks,
};
});
shouldTsIgnoreJs = !allowJs;
if (compilation instanceof compilation_1.NoopCompilation) {
await TS_COMPILATION_READY;
return result;
}
(0, profiling_1.profileSync)('NG_DIAGNOSTICS_TOTAL', () => {
for (const diagnostic of compilation.collectDiagnostics()) {
const message = (0, diagnostics_1.convertTypeScriptDiagnostic)(diagnostic);
if (diagnostic.category === typescript_1.default.DiagnosticCategory.Error) {
(result.errors ?? (result.errors = [])).push(message);
}
else {
(result.warnings ?? (result.warnings = [])).push(message);
}
}
});
// Update TypeScript file output cache for all affected files
(0, profiling_1.profileSync)('NG_EMIT_TS', () => {
for (const { filename, contents } of compilation.emitAffectedFiles()) {
typeScriptFileCache.set((0, node_url_1.pathToFileURL)(filename).href, contents);
}
});
// Store referenced files for updated file watching if enabled
if (pluginOptions.sourceFileCache) {
pluginOptions.sourceFileCache.referencedFiles = referencedFiles;
}
// Reset the setup warnings so that they are only shown during the first build.
setupWarnings = undefined;
// TODO: find a better way to unblock TS compilation of server bundles.
resolveCompilationReady?.();
return result;
});
build.onLoad({ filter: /\.[cm]?[jt]sx?$/ }, async (args) => {
const request = pluginOptions.fileReplacements?.[args.path] ?? args.path;
// Skip TS load attempt if JS TypeScript compilation not enabled and file is JS
if (shouldTsIgnoreJs && /\.[cm]?js$/.test(request)) {
return undefined;
}
// The filename is currently used as a cache key. Since the cache is memory only,
// the options cannot change and do not need to be represented in the key. If the
// cache is later stored to disk, then the options that affect transform output
// would need to be added to the key as well as a check for any change of content.
let contents = typeScriptFileCache.get((0, node_url_1.pathToFileURL)(request).href);
if (contents === undefined) {
// No TS result indicates the file is not part of the TypeScript program.
// If allowJs is enabled and the file is JS then defer to the next load hook.
if (!shouldTsIgnoreJs && /\.[cm]?js$/.test(request)) {
return undefined;
}
// Otherwise return an error
return {
errors: [
createMissingFileError(request, args.path, build.initialOptions.absWorkingDir ?? ''),
],
};
}
else if (typeof contents === 'string') {
// A string indicates untransformed output from the TS/NG compiler
contents = await javascriptTransformer.transformData(request, contents, true /* skipLinker */);
// Store as the returned Uint8Array to allow caching the fully transformed code
typeScriptFileCache.set((0, node_url_1.pathToFileURL)(request).href, contents);
}
return {
contents,
loader: 'js',
};
});
build.onLoad({ filter: /\.[cm]?js$/ }, (args) => (0, profiling_1.profileAsync)('NG_EMIT_JS*', async () => {
// The filename is currently used as a cache key. Since the cache is memory only,
// the options cannot change and do not need to be represented in the key. If the
// cache is later stored to disk, then the options that affect transform output
// would need to be added to the key as well as a check for any change of content.
let contents = pluginOptions.sourceFileCache?.babelFileCache.get(args.path);
if (contents === undefined) {
contents = await javascriptTransformer.transformFile(args.path, pluginOptions.jit);
pluginOptions.sourceFileCache?.babelFileCache.set(args.path, contents);
}
return {
contents,
loader: 'js',
};
}, true));
// Setup bundling of component templates and stylesheets when in JIT mode
if (pluginOptions.jit) {
(0, jit_plugin_callbacks_1.setupJitPluginCallbacks)(build, styleOptions, stylesheetResourceFiles, pluginOptions.loadResultCache);
}
build.onEnd((result) => {
// Add any component stylesheet resource files to the output files
if (stylesheetResourceFiles.length) {
result.outputFiles?.push(...stylesheetResourceFiles);
}
// Combine component stylesheet metafiles with main metafile
if (result.metafile && stylesheetMetafiles.length) {
for (const metafile of stylesheetMetafiles) {
result.metafile.inputs = { ...result.metafile.inputs, ...metafile.inputs };
result.metafile.outputs = { ...result.metafile.outputs, ...metafile.outputs };
}
}
(0, profiling_1.logCumulativeDurations)();
});
},
};
}
exports.createCompilerPlugin = createCompilerPlugin;
function createMissingFileError(request, original, root) {
const error = {
text: `File '${path.relative(root, request)}' is missing from the TypeScript compilation.`,
notes: [
{
text: `Ensure the file is part of the TypeScript program via the 'files' or 'include' property.`,
},
],
};
if (request !== original) {
error.notes.push({
text: `File is requested from a file replacement of '${path.relative(root, original)}'.`,
});
}
return error;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"compiler-plugin.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/angular/compiler-plugin.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 {\n  Metafile,\n  OnStartResult,\n  OutputFile,\n  PartialMessage,\n  Plugin,\n  PluginBuild,\n} from 'esbuild';\nimport { realpath } from 'node:fs/promises';\nimport { platform } from 'node:os';\nimport * as path from 'node:path';\nimport { pathToFileURL } from 'node:url';\nimport ts from 'typescript';\nimport { maxWorkers } from '../../../utils/environment-options';\nimport { JavaScriptTransformer } from '../javascript-transformer';\nimport { LoadResultCache, MemoryLoadResultCache } from '../load-result-cache';\nimport {\n  logCumulativeDurations,\n  profileAsync,\n  profileSync,\n  resetCumulativeDurations,\n} from '../profiling';\nimport { BundleStylesheetOptions, bundleComponentStylesheet } from '../stylesheets/bundle-options';\nimport { AngularHostOptions } from './angular-host';\nimport { AngularCompilation, AotCompilation, JitCompilation, NoopCompilation } from './compilation';\nimport { convertTypeScriptDiagnostic } from './diagnostics';\nimport { setupJitPluginCallbacks } from './jit-plugin-callbacks';\n\nconst USING_WINDOWS = platform() === 'win32';\nconst WINDOWS_SEP_REGEXP = new RegExp(`\\\\${path.win32.sep}`, 'g');\n\nexport class SourceFileCache extends Map<string, ts.SourceFile> {\n  readonly modifiedFiles = new Set<string>();\n  readonly babelFileCache = new Map<string, Uint8Array>();\n  readonly typeScriptFileCache = new Map<string, string | Uint8Array>();\n  readonly loadResultCache = new MemoryLoadResultCache();\n\n  referencedFiles?: readonly string[];\n\n  constructor(readonly persistentCachePath?: string) {\n    super();\n  }\n\n  invalidate(files: Iterable<string>): void {\n    this.modifiedFiles.clear();\n    for (let file of files) {\n      this.babelFileCache.delete(file);\n      this.typeScriptFileCache.delete(pathToFileURL(file).href);\n      this.loadResultCache.invalidate(file);\n\n      // Normalize separators to allow matching TypeScript Host paths\n      if (USING_WINDOWS) {\n        file = file.replace(WINDOWS_SEP_REGEXP, path.posix.sep);\n      }\n\n      this.delete(file);\n      this.modifiedFiles.add(file);\n    }\n  }\n}\n\nexport interface CompilerPluginOptions {\n  sourcemap: boolean;\n  tsconfig: string;\n  jit?: boolean;\n  /** Skip TypeScript compilation setup. This is useful to re-use the TypeScript compilation from another plugin. */\n  noopTypeScriptCompilation?: boolean;\n  advancedOptimizations?: boolean;\n  thirdPartySourcemaps?: boolean;\n  fileReplacements?: Record<string, string>;\n  sourceFileCache?: SourceFileCache;\n  loadResultCache?: LoadResultCache;\n}\n\n// TODO: find a better way to unblock TS compilation of server bundles.\nlet TS_COMPILATION_READY: Promise<void> | undefined;\n\n// eslint-disable-next-line max-lines-per-function\nexport function createCompilerPlugin(\n  pluginOptions: CompilerPluginOptions,\n  styleOptions: BundleStylesheetOptions & { inlineStyleLanguage: string },\n): Plugin {\n  let resolveCompilationReady: (() => void) | undefined;\n\n  if (!pluginOptions.noopTypeScriptCompilation) {\n    TS_COMPILATION_READY = new Promise<void>((resolve) => {\n      resolveCompilationReady = resolve;\n    });\n  }\n\n  return {\n    name: 'angular-compiler',\n    // eslint-disable-next-line max-lines-per-function\n    async setup(build: PluginBuild): Promise<void> {\n      let setupWarnings: PartialMessage[] | undefined = [];\n\n      const preserveSymlinks = build.initialOptions.preserveSymlinks;\n      let tsconfigPath = pluginOptions.tsconfig;\n      if (!preserveSymlinks) {\n        // Use the real path of the tsconfig if not preserving symlinks.\n        // This ensures the TS source file paths are based on the real path of the configuration.\n        try {\n          tsconfigPath = await realpath(tsconfigPath);\n        } catch {}\n      }\n\n      // Initialize a worker pool for JavaScript transformations\n      const javascriptTransformer = new JavaScriptTransformer(pluginOptions, maxWorkers);\n\n      // Setup defines based on the values provided by the Angular compiler-cli\n      const { GLOBAL_DEFS_FOR_TERSER_WITH_AOT } = await AngularCompilation.loadCompilerCli();\n      build.initialOptions.define ??= {};\n      for (const [key, value] of Object.entries(GLOBAL_DEFS_FOR_TERSER_WITH_AOT)) {\n        if (key in build.initialOptions.define) {\n          // Skip keys that have been manually provided\n          continue;\n        }\n        if (key === 'ngDevMode') {\n          // ngDevMode is already set based on the builder's script optimization option\n          continue;\n        }\n        // esbuild requires values to be a string (actual strings need to be quoted).\n        // In this case, all provided values are booleans.\n        build.initialOptions.define[key] = value.toString();\n      }\n\n      // The in-memory cache of TypeScript file outputs will be used during the build in `onLoad` callbacks for TS files.\n      // A string value indicates direct TS/NG output and a Uint8Array indicates fully transformed code.\n      const typeScriptFileCache =\n        pluginOptions.sourceFileCache?.typeScriptFileCache ??\n        new Map<string, string | Uint8Array>();\n\n      // The stylesheet resources from component stylesheets that will be added to the build results output files\n      let stylesheetResourceFiles: OutputFile[] = [];\n      let stylesheetMetafiles: Metafile[];\n\n      // Create new reusable compilation for the appropriate mode based on the `jit` plugin option\n      const compilation: AngularCompilation = pluginOptions.noopTypeScriptCompilation\n        ? new NoopCompilation()\n        : pluginOptions.jit\n        ? new JitCompilation()\n        : new AotCompilation();\n\n      // Determines if TypeScript should process JavaScript files based on tsconfig `allowJs` option\n      let shouldTsIgnoreJs = true;\n\n      build.onStart(async () => {\n        const result: OnStartResult = {\n          warnings: setupWarnings,\n        };\n\n        // Reset debug performance tracking\n        resetCumulativeDurations();\n\n        // Reset stylesheet resource output files\n        stylesheetResourceFiles = [];\n        stylesheetMetafiles = [];\n\n        // Create Angular compiler host options\n        const hostOptions: AngularHostOptions = {\n          fileReplacements: pluginOptions.fileReplacements,\n          modifiedFiles: pluginOptions.sourceFileCache?.modifiedFiles,\n          sourceFileCache: pluginOptions.sourceFileCache,\n          async transformStylesheet(data, containingFile, stylesheetFile) {\n            // Stylesheet file only exists for external stylesheets\n            const filename = stylesheetFile ?? containingFile;\n\n            const stylesheetResult = await bundleComponentStylesheet(\n              styleOptions.inlineStyleLanguage,\n              data,\n              filename,\n              !stylesheetFile,\n              styleOptions,\n              pluginOptions.loadResultCache,\n            );\n\n            const { contents, resourceFiles, errors, warnings } = stylesheetResult;\n            if (errors) {\n              (result.errors ??= []).push(...errors);\n            }\n            (result.warnings ??= []).push(...warnings);\n            stylesheetResourceFiles.push(...resourceFiles);\n            if (stylesheetResult.metafile) {\n              stylesheetMetafiles.push(stylesheetResult.metafile);\n            }\n\n            return contents;\n          },\n        };\n\n        // Initialize the Angular compilation for the current build.\n        // In watch mode, previous build state will be reused.\n        const {\n          compilerOptions: { allowJs },\n          referencedFiles,\n        } = await compilation.initialize(tsconfigPath, hostOptions, (compilerOptions) => {\n          if (\n            compilerOptions.target === undefined ||\n            compilerOptions.target < ts.ScriptTarget.ES2022\n          ) {\n            // If 'useDefineForClassFields' is already defined in the users project leave the value as is.\n            // Otherwise fallback to false due to https://github.com/microsoft/TypeScript/issues/45995\n            // which breaks the deprecated `@Effects` NGRX decorator and potentially other existing code as well.\n            compilerOptions.target = ts.ScriptTarget.ES2022;\n            compilerOptions.useDefineForClassFields ??= false;\n\n            // Only add the warning on the initial build\n            setupWarnings?.push({\n              text:\n                'TypeScript compiler options \"target\" and \"useDefineForClassFields\" are set to \"ES2022\" and ' +\n                '\"false\" respectively by the Angular CLI.',\n              location: { file: pluginOptions.tsconfig },\n              notes: [\n                {\n                  text:\n                    'To control ECMA version and features use the Browerslist configuration. ' +\n                    'For more information, see https://angular.io/guide/build#configuring-browser-compatibility',\n                },\n              ],\n            });\n          }\n\n          // Enable incremental compilation by default if caching is enabled\n          if (pluginOptions.sourceFileCache?.persistentCachePath) {\n            compilerOptions.incremental ??= true;\n            // Set the build info file location to the configured cache directory\n            compilerOptions.tsBuildInfoFile = path.join(\n              pluginOptions.sourceFileCache?.persistentCachePath,\n              '.tsbuildinfo',\n            );\n          } else {\n            compilerOptions.incremental = false;\n          }\n\n          return {\n            ...compilerOptions,\n            noEmitOnError: false,\n            inlineSources: pluginOptions.sourcemap,\n            inlineSourceMap: pluginOptions.sourcemap,\n            mapRoot: undefined,\n            sourceRoot: undefined,\n            preserveSymlinks,\n          };\n        });\n        shouldTsIgnoreJs = !allowJs;\n\n        if (compilation instanceof NoopCompilation) {\n          await TS_COMPILATION_READY;\n\n          return result;\n        }\n\n        profileSync('NG_DIAGNOSTICS_TOTAL', () => {\n          for (const diagnostic of compilation.collectDiagnostics()) {\n            const message = convertTypeScriptDiagnostic(diagnostic);\n            if (diagnostic.category === ts.DiagnosticCategory.Error) {\n              (result.errors ??= []).push(message);\n            } else {\n              (result.warnings ??= []).push(message);\n            }\n          }\n        });\n\n        // Update TypeScript file output cache for all affected files\n        profileSync('NG_EMIT_TS', () => {\n          for (const { filename, contents } of compilation.emitAffectedFiles()) {\n            typeScriptFileCache.set(pathToFileURL(filename).href, contents);\n          }\n        });\n\n        // Store referenced files for updated file watching if enabled\n        if (pluginOptions.sourceFileCache) {\n          pluginOptions.sourceFileCache.referencedFiles = referencedFiles;\n        }\n\n        // Reset the setup warnings so that they are only shown during the first build.\n        setupWarnings = undefined;\n\n        // TODO: find a better way to unblock TS compilation of server bundles.\n        resolveCompilationReady?.();\n\n        return result;\n      });\n\n      build.onLoad({ filter: /\\.[cm]?[jt]sx?$/ }, async (args) => {\n        const request = pluginOptions.fileReplacements?.[args.path] ?? args.path;\n\n        // Skip TS load attempt if JS TypeScript compilation not enabled and file is JS\n        if (shouldTsIgnoreJs && /\\.[cm]?js$/.test(request)) {\n          return undefined;\n        }\n\n        // The filename is currently used as a cache key. Since the cache is memory only,\n        // the options cannot change and do not need to be represented in the key. If the\n        // cache is later stored to disk, then the options that affect transform output\n        // would need to be added to the key as well as a check for any change of content.\n        let contents = typeScriptFileCache.get(pathToFileURL(request).href);\n\n        if (contents === undefined) {\n          // No TS result indicates the file is not part of the TypeScript program.\n          // If allowJs is enabled and the file is JS then defer to the next load hook.\n          if (!shouldTsIgnoreJs && /\\.[cm]?js$/.test(request)) {\n            return undefined;\n          }\n\n          // Otherwise return an error\n          return {\n            errors: [\n              createMissingFileError(request, args.path, build.initialOptions.absWorkingDir ?? ''),\n            ],\n          };\n        } else if (typeof contents === 'string') {\n          // A string indicates untransformed output from the TS/NG compiler\n          contents = await javascriptTransformer.transformData(\n            request,\n            contents,\n            true /* skipLinker */,\n          );\n\n          // Store as the returned Uint8Array to allow caching the fully transformed code\n          typeScriptFileCache.set(pathToFileURL(request).href, contents);\n        }\n\n        return {\n          contents,\n          loader: 'js',\n        };\n      });\n\n      build.onLoad({ filter: /\\.[cm]?js$/ }, (args) =>\n        profileAsync(\n          'NG_EMIT_JS*',\n          async () => {\n            // The filename is currently used as a cache key. Since the cache is memory only,\n            // the options cannot change and do not need to be represented in the key. If the\n            // cache is later stored to disk, then the options that affect transform output\n            // would need to be added to the key as well as a check for any change of content.\n            let contents = pluginOptions.sourceFileCache?.babelFileCache.get(args.path);\n            if (contents === undefined) {\n              contents = await javascriptTransformer.transformFile(args.path, pluginOptions.jit);\n              pluginOptions.sourceFileCache?.babelFileCache.set(args.path, contents);\n            }\n\n            return {\n              contents,\n              loader: 'js',\n            };\n          },\n          true,\n        ),\n      );\n\n      // Setup bundling of component templates and stylesheets when in JIT mode\n      if (pluginOptions.jit) {\n        setupJitPluginCallbacks(\n          build,\n          styleOptions,\n          stylesheetResourceFiles,\n          pluginOptions.loadResultCache,\n        );\n      }\n\n      build.onEnd((result) => {\n        // Add any component stylesheet resource files to the output files\n        if (stylesheetResourceFiles.length) {\n          result.outputFiles?.push(...stylesheetResourceFiles);\n        }\