@angular-devkit/build-angular 16.2.0-next.0 → 16.2.0-next.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (62) hide show
  1. package/package.json +22 -21
  2. package/src/builders/app-shell/render-worker.js +6 -2
  3. package/src/builders/application/execute-build.js +50 -11
  4. package/src/builders/application/options.d.ts +17 -4
  5. package/src/builders/application/options.js +63 -28
  6. package/src/builders/application/schema.d.ts +62 -6
  7. package/src/builders/application/schema.js +1 -1
  8. package/src/builders/application/schema.json +62 -4
  9. package/src/builders/browser-esbuild/builder-status-warnings.d.ts +1 -1
  10. package/src/builders/browser-esbuild/builder-status-warnings.js +2 -1
  11. package/src/builders/browser-esbuild/schema.d.ts +17 -1
  12. package/src/builders/browser-esbuild/schema.js +1 -1
  13. package/src/builders/browser-esbuild/schema.json +1 -0
  14. package/src/builders/dev-server/builder.js +3 -2
  15. package/src/builders/dev-server/vite-server.d.ts +1 -1
  16. package/src/builders/dev-server/vite-server.js +82 -37
  17. package/src/builders/jest/index.js +2 -2
  18. package/src/tools/babel/presets/application.d.ts +1 -0
  19. package/src/tools/babel/presets/application.js +43 -41
  20. package/src/tools/babel/webpack-loader.d.ts +0 -1
  21. package/src/tools/babel/webpack-loader.js +3 -23
  22. package/src/tools/esbuild/angular/angular-host.js +5 -2
  23. package/src/tools/esbuild/angular/{angular-compilation.d.ts → compilation/angular-compilation.d.ts} +1 -1
  24. package/src/tools/esbuild/angular/compilation/angular-compilation.js +52 -0
  25. package/src/tools/esbuild/angular/{aot-compilation.d.ts → compilation/aot-compilation.d.ts} +1 -1
  26. package/src/tools/esbuild/angular/compilation/aot-compilation.js +211 -0
  27. package/src/tools/esbuild/angular/compilation/index.d.ts +11 -0
  28. package/src/tools/esbuild/angular/compilation/index.js +19 -0
  29. package/src/tools/esbuild/angular/{jit-compilation.d.ts → compilation/jit-compilation.d.ts} +1 -1
  30. package/src/tools/esbuild/angular/compilation/jit-compilation.js +106 -0
  31. package/src/tools/esbuild/angular/compilation/noop-compilation.d.ts +20 -0
  32. package/src/tools/esbuild/angular/compilation/noop-compilation.js +27 -0
  33. package/src/tools/esbuild/angular/compiler-plugin.d.ts +2 -0
  34. package/src/tools/esbuild/angular/compiler-plugin.js +22 -8
  35. package/src/tools/esbuild/application-code-bundle.d.ts +8 -2
  36. package/src/tools/esbuild/application-code-bundle.js +163 -71
  37. package/src/tools/esbuild/compiler-plugin-options.d.ts +15 -0
  38. package/src/tools/esbuild/compiler-plugin-options.js +45 -0
  39. package/src/tools/esbuild/global-scripts.js +6 -5
  40. package/src/tools/esbuild/global-styles.d.ts +1 -1
  41. package/src/tools/esbuild/global-styles.js +2 -3
  42. package/src/tools/esbuild/index-html-generator.d.ts +6 -2
  43. package/src/tools/esbuild/index-html-generator.js +41 -13
  44. package/src/tools/esbuild/javascript-transformer-worker.js +25 -6
  45. package/src/tools/esbuild/load-result-cache.js +6 -3
  46. package/src/tools/esbuild/rxjs-esm-resolution-plugin.d.ts +18 -0
  47. package/src/tools/esbuild/rxjs-esm-resolution-plugin.js +45 -0
  48. package/src/tools/esbuild/stylesheets/bundle-options.d.ts +3 -4
  49. package/src/tools/esbuild/stylesheets/bundle-options.js +3 -4
  50. package/src/tools/esbuild/stylesheets/stylesheet-plugin-factory.d.ts +8 -4
  51. package/src/tools/esbuild/stylesheets/stylesheet-plugin-factory.js +34 -16
  52. package/src/utils/server-rendering/esm-in-memory-file-loader.d.ts +11 -0
  53. package/src/utils/server-rendering/esm-in-memory-file-loader.js +53 -0
  54. package/src/utils/server-rendering/prerender.d.ts +22 -0
  55. package/src/utils/server-rendering/prerender.js +127 -0
  56. package/src/utils/server-rendering/render-page.d.ts +38 -0
  57. package/src/utils/server-rendering/render-page.js +63 -0
  58. package/src/utils/server-rendering/render-worker.d.ts +18 -0
  59. package/src/utils/server-rendering/render-worker.js +20 -0
  60. package/src/tools/esbuild/angular/angular-compilation.js +0 -52
  61. package/src/tools/esbuild/angular/aot-compilation.js +0 -211
  62. package/src/tools/esbuild/angular/jit-compilation.js +0 -106
@@ -30,14 +30,13 @@ function createStylesheetBundleOptions(options, cache, inlineComponentData) {
30
30
  sourcemap: !!options.sourcemap,
31
31
  includePaths,
32
32
  inlineComponentData,
33
- browsers: options.browsers,
34
33
  tailwindConfiguration: options.tailwindConfiguration,
35
34
  }, cache);
36
35
  return {
37
36
  absWorkingDir: options.workspaceRoot,
38
37
  bundle: true,
39
- entryNames: options.outputNames?.bundles,
40
- assetNames: options.outputNames?.media,
38
+ entryNames: options.outputNames.bundles,
39
+ assetNames: options.outputNames.media,
41
40
  logLevel: 'silent',
42
41
  minify: options.optimization,
43
42
  metafile: true,
@@ -156,4 +155,4 @@ async function bundleComponentStylesheet(language, data, filename, inline, optio
156
155
  };
157
156
  }
158
157
  exports.bundleComponentStylesheet = bundleComponentStylesheet;
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundle-options.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/bundle-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 type { BuildOptions, OutputFile } from 'esbuild';\nimport { createHash } from 'node:crypto';\nimport path from 'node:path';\nimport { BundlerContext } from '../bundler-context';\nimport { LoadResultCache } from '../load-result-cache';\nimport { CssStylesheetLanguage } from './css-language';\nimport { createCssResourcePlugin } from './css-resource-plugin';\nimport { LessStylesheetLanguage } from './less-language';\nimport { SassStylesheetLanguage } from './sass-language';\nimport { StylesheetPluginFactory } from './stylesheet-plugin-factory';\n\n/**\n * A counter for component styles used to generate unique build-time identifiers for each stylesheet.\n */\nlet componentStyleCounter = 0;\n\nexport interface BundleStylesheetOptions {\n  workspaceRoot: string;\n  optimization: boolean;\n  preserveSymlinks?: boolean;\n  sourcemap: boolean | 'external' | 'inline';\n  outputNames?: { bundles?: string; media?: string };\n  includePaths?: string[];\n  externalDependencies?: string[];\n  target: string[];\n  browsers: string[];\n  tailwindConfiguration?: { file: string; package: string };\n}\n\nexport function createStylesheetBundleOptions(\n  options: BundleStylesheetOptions,\n  cache?: LoadResultCache,\n  inlineComponentData?: Record<string, string>,\n): BuildOptions & { plugins: NonNullable<BuildOptions['plugins']> } {\n  // Ensure preprocessor include paths are absolute based on the workspace root\n  const includePaths = options.includePaths?.map((includePath) =>\n    path.resolve(options.workspaceRoot, includePath),\n  );\n\n  const pluginFactory = new StylesheetPluginFactory(\n    {\n      sourcemap: !!options.sourcemap,\n      includePaths,\n      inlineComponentData,\n      browsers: options.browsers,\n      tailwindConfiguration: options.tailwindConfiguration,\n    },\n    cache,\n  );\n\n  return {\n    absWorkingDir: options.workspaceRoot,\n    bundle: true,\n    entryNames: options.outputNames?.bundles,\n    assetNames: options.outputNames?.media,\n    logLevel: 'silent',\n    minify: options.optimization,\n    metafile: true,\n    sourcemap: options.sourcemap,\n    outdir: options.workspaceRoot,\n    write: false,\n    platform: 'browser',\n    target: options.target,\n    preserveSymlinks: options.preserveSymlinks,\n    external: options.externalDependencies,\n    conditions: ['style', 'sass'],\n    mainFields: ['style', 'sass'],\n    plugins: [\n      pluginFactory.create(SassStylesheetLanguage),\n      pluginFactory.create(LessStylesheetLanguage),\n      pluginFactory.create(CssStylesheetLanguage),\n      createCssResourcePlugin(cache),\n    ],\n  };\n}\n\n/**\n * Bundles a component stylesheet. The stylesheet can be either an inline stylesheet that\n * is contained within the Component's metadata definition or an external file referenced\n * from the Component's metadata definition.\n *\n * @param identifier A unique string identifier for the component stylesheet.\n * @param language The language of the stylesheet such as `css` or `scss`.\n * @param data The string content of the stylesheet.\n * @param filename The filename representing the source of the stylesheet content.\n * @param inline If true, the stylesheet source is within the component metadata;\n * if false, the source is a stylesheet file.\n * @param options An object containing the stylesheet bundling options.\n * @returns An object containing the output of the bundling operation.\n */\nexport async function bundleComponentStylesheet(\n  language: string,\n  data: string,\n  filename: string,\n  inline: boolean,\n  options: BundleStylesheetOptions,\n  cache?: LoadResultCache,\n) {\n  const namespace = 'angular:styles/component';\n  // Use a hash of the inline stylesheet content to ensure a consistent identifier. External stylesheets will resolve\n  // to the actual stylesheet file path.\n  // TODO: Consider xxhash instead for hashing\n  const id = inline ? createHash('sha256').update(data).digest('hex') : componentStyleCounter++;\n  const entry = [language, id, filename].join(';');\n\n  const buildOptions = createStylesheetBundleOptions(options, cache, { [entry]: data });\n  buildOptions.entryPoints = [`${namespace};${entry}`];\n  buildOptions.plugins.push({\n    name: 'angular-component-styles',\n    setup(build) {\n      build.onResolve({ filter: /^angular:styles\\/component;/ }, (args) => {\n        if (args.kind !== 'entry-point') {\n          return null;\n        }\n\n        if (inline) {\n          return {\n            path: entry,\n            namespace,\n          };\n        } else {\n          return {\n            path: filename,\n          };\n        }\n      });\n      build.onLoad({ filter: /^css;/, namespace }, async () => {\n        return {\n          contents: data,\n          loader: 'css',\n          resolveDir: path.dirname(filename),\n        };\n      });\n    },\n  });\n\n  // Execute esbuild\n  const context = new BundlerContext(options.workspaceRoot, false, buildOptions);\n  const result = await context.bundle();\n\n  // Extract the result of the bundling from the output files\n  let contents = '';\n  let map;\n  let outputPath;\n  const resourceFiles: OutputFile[] = [];\n  if (!result.errors) {\n    for (const outputFile of result.outputFiles) {\n      const filename = path.basename(outputFile.path);\n      if (filename.endsWith('.css')) {\n        outputPath = outputFile.path;\n        contents = outputFile.text;\n      } else if (filename.endsWith('.css.map')) {\n        map = outputFile.text;\n      } else {\n        // The output files could also contain resources (images/fonts/etc.) that were referenced\n        resourceFiles.push(outputFile);\n      }\n    }\n  }\n\n  let metafile;\n  if (!result.errors) {\n    metafile = result.metafile;\n    // Remove entryPoint fields from outputs to prevent the internal component styles from being\n    // treated as initial files. Also mark the entry as a component resource for stat reporting.\n    Object.values(metafile.outputs).forEach((output) => {\n      delete output.entryPoint;\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      (output as any)['ng-component'] = true;\n    });\n  }\n\n  return {\n    errors: result.errors,\n    warnings: result.warnings,\n    contents,\n    map,\n    path: outputPath,\n    resourceFiles,\n    metafile,\n  };\n}\n"]}
158
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bundle-options.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/bundle-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 type { BuildOptions, OutputFile } from 'esbuild';\nimport { createHash } from 'node:crypto';\nimport path from 'node:path';\nimport { BundlerContext } from '../bundler-context';\nimport { LoadResultCache } from '../load-result-cache';\nimport { CssStylesheetLanguage } from './css-language';\nimport { createCssResourcePlugin } from './css-resource-plugin';\nimport { LessStylesheetLanguage } from './less-language';\nimport { SassStylesheetLanguage } from './sass-language';\nimport { StylesheetPluginFactory } from './stylesheet-plugin-factory';\n\n/**\n * A counter for component styles used to generate unique build-time identifiers for each stylesheet.\n */\nlet componentStyleCounter = 0;\n\nexport interface BundleStylesheetOptions {\n  workspaceRoot: string;\n  optimization: boolean;\n  preserveSymlinks?: boolean;\n  sourcemap: boolean | 'external' | 'inline';\n  outputNames: { bundles: string; media: string };\n  includePaths?: string[];\n  externalDependencies?: string[];\n  target: string[];\n  tailwindConfiguration?: { file: string; package: string };\n}\n\nexport function createStylesheetBundleOptions(\n  options: BundleStylesheetOptions,\n  cache?: LoadResultCache,\n  inlineComponentData?: Record<string, string>,\n): BuildOptions & { plugins: NonNullable<BuildOptions['plugins']> } {\n  // Ensure preprocessor include paths are absolute based on the workspace root\n  const includePaths = options.includePaths?.map((includePath) =>\n    path.resolve(options.workspaceRoot, includePath),\n  );\n\n  const pluginFactory = new StylesheetPluginFactory(\n    {\n      sourcemap: !!options.sourcemap,\n      includePaths,\n      inlineComponentData,\n      tailwindConfiguration: options.tailwindConfiguration,\n    },\n    cache,\n  );\n\n  return {\n    absWorkingDir: options.workspaceRoot,\n    bundle: true,\n    entryNames: options.outputNames.bundles,\n    assetNames: options.outputNames.media,\n    logLevel: 'silent',\n    minify: options.optimization,\n    metafile: true,\n    sourcemap: options.sourcemap,\n    outdir: options.workspaceRoot,\n    write: false,\n    platform: 'browser',\n    target: options.target,\n    preserveSymlinks: options.preserveSymlinks,\n    external: options.externalDependencies,\n    conditions: ['style', 'sass'],\n    mainFields: ['style', 'sass'],\n    plugins: [\n      pluginFactory.create(SassStylesheetLanguage),\n      pluginFactory.create(LessStylesheetLanguage),\n      pluginFactory.create(CssStylesheetLanguage),\n      createCssResourcePlugin(cache),\n    ],\n  };\n}\n\n/**\n * Bundles a component stylesheet. The stylesheet can be either an inline stylesheet that\n * is contained within the Component's metadata definition or an external file referenced\n * from the Component's metadata definition.\n *\n * @param identifier A unique string identifier for the component stylesheet.\n * @param language The language of the stylesheet such as `css` or `scss`.\n * @param data The string content of the stylesheet.\n * @param filename The filename representing the source of the stylesheet content.\n * @param inline If true, the stylesheet source is within the component metadata;\n * if false, the source is a stylesheet file.\n * @param options An object containing the stylesheet bundling options.\n * @returns An object containing the output of the bundling operation.\n */\nexport async function bundleComponentStylesheet(\n  language: string,\n  data: string,\n  filename: string,\n  inline: boolean,\n  options: BundleStylesheetOptions,\n  cache?: LoadResultCache,\n) {\n  const namespace = 'angular:styles/component';\n  // Use a hash of the inline stylesheet content to ensure a consistent identifier. External stylesheets will resolve\n  // to the actual stylesheet file path.\n  // TODO: Consider xxhash instead for hashing\n  const id = inline ? createHash('sha256').update(data).digest('hex') : componentStyleCounter++;\n  const entry = [language, id, filename].join(';');\n\n  const buildOptions = createStylesheetBundleOptions(options, cache, { [entry]: data });\n  buildOptions.entryPoints = [`${namespace};${entry}`];\n  buildOptions.plugins.push({\n    name: 'angular-component-styles',\n    setup(build) {\n      build.onResolve({ filter: /^angular:styles\\/component;/ }, (args) => {\n        if (args.kind !== 'entry-point') {\n          return null;\n        }\n\n        if (inline) {\n          return {\n            path: entry,\n            namespace,\n          };\n        } else {\n          return {\n            path: filename,\n          };\n        }\n      });\n      build.onLoad({ filter: /^css;/, namespace }, async () => {\n        return {\n          contents: data,\n          loader: 'css',\n          resolveDir: path.dirname(filename),\n        };\n      });\n    },\n  });\n\n  // Execute esbuild\n  const context = new BundlerContext(options.workspaceRoot, false, buildOptions);\n  const result = await context.bundle();\n\n  // Extract the result of the bundling from the output files\n  let contents = '';\n  let map;\n  let outputPath;\n  const resourceFiles: OutputFile[] = [];\n  if (!result.errors) {\n    for (const outputFile of result.outputFiles) {\n      const filename = path.basename(outputFile.path);\n      if (filename.endsWith('.css')) {\n        outputPath = outputFile.path;\n        contents = outputFile.text;\n      } else if (filename.endsWith('.css.map')) {\n        map = outputFile.text;\n      } else {\n        // The output files could also contain resources (images/fonts/etc.) that were referenced\n        resourceFiles.push(outputFile);\n      }\n    }\n  }\n\n  let metafile;\n  if (!result.errors) {\n    metafile = result.metafile;\n    // Remove entryPoint fields from outputs to prevent the internal component styles from being\n    // treated as initial files. Also mark the entry as a component resource for stat reporting.\n    Object.values(metafile.outputs).forEach((output) => {\n      delete output.entryPoint;\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      (output as any)['ng-component'] = true;\n    });\n  }\n\n  return {\n    errors: result.errors,\n    warnings: result.warnings,\n    contents,\n    map,\n    path: outputPath,\n    resourceFiles,\n    metafile,\n  };\n}\n"]}
@@ -16,6 +16,10 @@ export interface StylesheetPluginOptions {
16
16
  * If true, sourcemap processing is enabled; if false, disabled.
17
17
  */
18
18
  sourcemap: boolean;
19
+ /**
20
+ * An optional array of paths that will be searched for stylesheets if the default
21
+ * resolution process for the stylesheet language does not succeed.
22
+ */
19
23
  includePaths?: string[];
20
24
  /**
21
25
  * Optional component data for any inline styles from Component decorator `styles` fields.
@@ -23,11 +27,10 @@ export interface StylesheetPluginOptions {
23
27
  */
24
28
  inlineComponentData?: Record<string, string>;
25
29
  /**
26
- * The browsers to support in browserslist format when processing stylesheets.
27
- * Some postcss plugins such as autoprefixer require the raw browserslist information instead
28
- * of the esbuild formatted target.
30
+ * Optional information used to load and configure Tailwind CSS. If present, the postcss
31
+ * will be added to the stylesheet processing with the Tailwind plugin setup as provided
32
+ * by the configuration file.
29
33
  */
30
- browsers: string[];
31
34
  tailwindConfiguration?: {
32
35
  file: string;
33
36
  package: string;
@@ -42,6 +45,7 @@ export interface StylesheetLanguage {
42
45
  export declare class StylesheetPluginFactory {
43
46
  private readonly options;
44
47
  private readonly cache?;
48
+ private postcssProcessor?;
45
49
  constructor(options: StylesheetPluginOptions, cache?: LoadResultCache | undefined);
46
50
  create(language: Readonly<StylesheetLanguage>): Plugin;
47
51
  }
@@ -44,6 +44,12 @@ const load_result_cache_1 = require("../load-result-cache");
44
44
  * It is only imported and initialized if postcss is needed.
45
45
  */
46
46
  let postcss;
47
+ /**
48
+ * An array of keywords that indicate Tailwind CSS processing is required for a stylesheet.
49
+ *
50
+ * Based on https://tailwindcss.com/docs/functions-and-directives
51
+ */
52
+ const TAILWIND_KEYWORDS = ['@tailwind', '@layer', '@apply', '@config', 'theme(', 'screen('];
47
53
  class StylesheetPluginFactory {
48
54
  constructor(options, cache) {
49
55
  this.options = options;
@@ -58,31 +64,40 @@ class StylesheetPluginFactory {
58
64
  };
59
65
  }
60
66
  const { cache, options } = this;
67
+ const setupPostcss = async () => {
68
+ // Return already created processor if present
69
+ if (this.postcssProcessor) {
70
+ return this.postcssProcessor;
71
+ }
72
+ if (options.tailwindConfiguration) {
73
+ postcss ?? (postcss = (await Promise.resolve().then(() => __importStar(require('postcss')))).default);
74
+ const tailwind = await Promise.resolve(`${options.tailwindConfiguration.package}`).then(s => __importStar(require(s)));
75
+ this.postcssProcessor = postcss().use(tailwind.default({ config: options.tailwindConfiguration.file }));
76
+ }
77
+ return this.postcssProcessor;
78
+ };
61
79
  return {
62
80
  name: 'angular-' + language.name,
63
81
  async setup(build) {
64
- // Setup postcss if needed by tailwind
65
- // TODO: Move this into the plugin factory to avoid repeat setup per created plugin
66
- let postcssProcessor;
67
- if (options.tailwindConfiguration) {
68
- postcss ?? (postcss = (await Promise.resolve().then(() => __importStar(require('postcss')))).default);
69
- postcssProcessor = postcss();
70
- if (options.tailwindConfiguration) {
71
- const tailwind = await Promise.resolve(`${options.tailwindConfiguration.package}`).then(s => __importStar(require(s)));
72
- postcssProcessor.use(tailwind.default({ config: options.tailwindConfiguration.file }));
73
- }
74
- }
82
+ // Setup postcss if needed
83
+ const postcssProcessor = await setupPostcss();
75
84
  // Add a load callback to support inline Component styles
76
- build.onLoad({ filter: language.componentFilter, namespace: 'angular:styles/component' }, (0, load_result_cache_1.createCachedLoad)(cache, async (args) => {
85
+ build.onLoad({ filter: language.componentFilter, namespace: 'angular:styles/component' }, (0, load_result_cache_1.createCachedLoad)(cache, (args) => {
77
86
  const data = options.inlineComponentData?.[args.path];
78
87
  (0, node_assert_1.default)(typeof data === 'string', `component style name should always be found [${args.path}]`);
79
88
  const [format, , filename] = args.path.split(';', 3);
80
- return processStylesheet(language, data, filename, format, options, build, postcssProcessor);
89
+ // Only use postcss if Tailwind processing is required.
90
+ // NOTE: If postcss is used for more than just Tailwind in the future this check MUST
91
+ // be updated to account for the additional use.
92
+ // TODO: use better search algorithm for keywords
93
+ const needsPostcss = !!postcssProcessor && TAILWIND_KEYWORDS.some((keyword) => data.includes(keyword));
94
+ return processStylesheet(language, data, filename, format, options, build, needsPostcss ? postcssProcessor : undefined);
81
95
  }));
82
96
  // Add a load callback to support files from disk
83
97
  build.onLoad({ filter: language.fileFilter }, (0, load_result_cache_1.createCachedLoad)(cache, async (args) => {
84
98
  const data = await (0, promises_1.readFile)(args.path, 'utf-8');
85
- return processStylesheet(language, data, args.path, (0, node_path_1.extname)(args.path).toLowerCase().slice(1), options, build, postcssProcessor);
99
+ const needsPostcss = !!postcssProcessor && TAILWIND_KEYWORDS.some((keyword) => data.includes(keyword));
100
+ return processStylesheet(language, data, args.path, (0, node_path_1.extname)(args.path).toLowerCase().slice(1), options, build, needsPostcss ? postcssProcessor : undefined);
86
101
  }));
87
102
  },
88
103
  };
@@ -183,7 +198,10 @@ async function compileString(data, filename, postcssProcessor, options) {
183
198
  typeof resultMessage['dir'] === 'string' &&
184
199
  typeof resultMessage['glob'] === 'string') {
185
200
  loadResult.watchFiles ?? (loadResult.watchFiles = []);
186
- const dependencies = await (0, fast_glob_1.default)(resultMessage['glob'], { cwd: resultMessage['dir'] });
201
+ const dependencies = await (0, fast_glob_1.default)(resultMessage['glob'], {
202
+ absolute: true,
203
+ cwd: resultMessage['dir'],
204
+ });
187
205
  loadResult.watchFiles.push(...dependencies);
188
206
  }
189
207
  }
@@ -210,4 +228,4 @@ async function compileString(data, filename, postcssProcessor, options) {
210
228
  throw error;
211
229
  }
212
230
  }
213
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stylesheet-plugin-factory.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/stylesheet-plugin-factory.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 { OnLoadResult, Plugin, PluginBuild } from 'esbuild';\nimport glob from 'fast-glob';\nimport assert from 'node:assert';\nimport { readFile } from 'node:fs/promises';\nimport { extname } from 'node:path';\nimport { LoadResultCache, createCachedLoad } from '../load-result-cache';\n\n/**\n * The lazy-loaded instance of the postcss stylesheet postprocessor.\n * It is only imported and initialized if postcss is needed.\n */\nlet postcss: typeof import('postcss')['default'] | undefined;\n\n/**\n * An object containing the plugin options to use when processing stylesheets.\n */\nexport interface StylesheetPluginOptions {\n  /**\n   * Controls the use and creation of sourcemaps when processing the stylesheets.\n   * If true, sourcemap processing is enabled; if false, disabled.\n   */\n  sourcemap: boolean;\n\n  includePaths?: string[];\n\n  /**\n   * Optional component data for any inline styles from Component decorator `styles` fields.\n   * The key is an internal angular resource URI and the value is the stylesheet content.\n   */\n  inlineComponentData?: Record<string, string>;\n\n  /**\n   * The browsers to support in browserslist format when processing stylesheets.\n   * Some postcss plugins such as autoprefixer require the raw browserslist information instead\n   * of the esbuild formatted target.\n   */\n  browsers: string[];\n\n  tailwindConfiguration?: { file: string; package: string };\n}\n\nexport interface StylesheetLanguage {\n  name: string;\n  componentFilter: RegExp;\n  fileFilter: RegExp;\n  process?(\n    data: string,\n    file: string,\n    format: string,\n    options: StylesheetPluginOptions,\n    build: PluginBuild,\n  ): OnLoadResult | Promise<OnLoadResult>;\n}\n\nexport class StylesheetPluginFactory {\n  constructor(\n    private readonly options: StylesheetPluginOptions,\n    private readonly cache?: LoadResultCache,\n  ) {}\n\n  create(language: Readonly<StylesheetLanguage>): Plugin {\n    // Return a noop plugin if no load actions are required\n    if (!language.process && !this.options.tailwindConfiguration) {\n      return {\n        name: 'angular-' + language.name,\n        setup() {},\n      };\n    }\n\n    const { cache, options } = this;\n\n    return {\n      name: 'angular-' + language.name,\n      async setup(build) {\n        // Setup postcss if needed by tailwind\n        // TODO: Move this into the plugin factory to avoid repeat setup per created plugin\n        let postcssProcessor: import('postcss').Processor | undefined;\n        if (options.tailwindConfiguration) {\n          postcss ??= (await import('postcss')).default;\n          postcssProcessor = postcss();\n          if (options.tailwindConfiguration) {\n            const tailwind = await import(options.tailwindConfiguration.package);\n            postcssProcessor.use(tailwind.default({ config: options.tailwindConfiguration.file }));\n          }\n        }\n\n        // Add a load callback to support inline Component styles\n        build.onLoad(\n          { filter: language.componentFilter, namespace: 'angular:styles/component' },\n          createCachedLoad(cache, async (args) => {\n            const data = options.inlineComponentData?.[args.path];\n            assert(\n              typeof data === 'string',\n              `component style name should always be found [${args.path}]`,\n            );\n\n            const [format, , filename] = args.path.split(';', 3);\n\n            return processStylesheet(\n              language,\n              data,\n              filename,\n              format,\n              options,\n              build,\n              postcssProcessor,\n            );\n          }),\n        );\n\n        // Add a load callback to support files from disk\n        build.onLoad(\n          { filter: language.fileFilter },\n          createCachedLoad(cache, async (args) => {\n            const data = await readFile(args.path, 'utf-8');\n\n            return processStylesheet(\n              language,\n              data,\n              args.path,\n              extname(args.path).toLowerCase().slice(1),\n              options,\n              build,\n              postcssProcessor,\n            );\n          }),\n        );\n      },\n    };\n  }\n}\n\nasync function processStylesheet(\n  language: Readonly<StylesheetLanguage>,\n  data: string,\n  filename: string,\n  format: string,\n  options: StylesheetPluginOptions,\n  build: PluginBuild,\n  postcssProcessor: import('postcss').Processor | undefined,\n) {\n  let result: OnLoadResult;\n\n  // Process the input data if the language requires preprocessing\n  if (language.process) {\n    result = await language.process(data, filename, format, options, build);\n  } else {\n    result = {\n      contents: data,\n      loader: 'css',\n      watchFiles: [filename],\n    };\n  }\n\n  // Transform with postcss if needed and there are no errors\n  if (postcssProcessor && result.contents && !result.errors?.length) {\n    const postcssResult = await compileString(\n      typeof result.contents === 'string'\n        ? result.contents\n        : Buffer.from(result.contents).toString('utf-8'),\n      filename,\n      postcssProcessor,\n      options,\n    );\n\n    // Merge results\n    if (postcssResult.errors?.length) {\n      delete result.contents;\n    }\n    if (result.warnings && postcssResult.warnings) {\n      postcssResult.warnings.unshift(...result.warnings);\n    }\n    if (result.watchFiles && postcssResult.watchFiles) {\n      postcssResult.watchFiles.unshift(...result.watchFiles);\n    }\n    if (result.watchDirs && postcssResult.watchDirs) {\n      postcssResult.watchDirs.unshift(...result.watchDirs);\n    }\n    result = {\n      ...result,\n      ...postcssResult,\n    };\n  }\n\n  return result;\n}\n\n/**\n * Compiles the provided CSS stylesheet data using a provided postcss processor and provides an\n * esbuild load result that can be used directly by an esbuild Plugin.\n * @param data The stylesheet content to process.\n * @param filename The name of the file that contains the data.\n * @param postcssProcessor A postcss processor instance to use.\n * @param options The plugin options to control the processing.\n * @returns An esbuild OnLoaderResult object with the processed content, warnings, and/or errors.\n */\nasync function compileString(\n  data: string,\n  filename: string,\n  postcssProcessor: import('postcss').Processor,\n  options: StylesheetPluginOptions,\n): Promise<OnLoadResult> {\n  try {\n    const postcssResult = await postcssProcessor.process(data, {\n      from: filename,\n      to: filename,\n      map: options.sourcemap && {\n        inline: true,\n        sourcesContent: true,\n      },\n    });\n\n    const loadResult: OnLoadResult = {\n      contents: postcssResult.css,\n      loader: 'css',\n    };\n\n    const rawWarnings = postcssResult.warnings();\n    if (rawWarnings.length > 0) {\n      const lineMappings = new Map<string, string[] | null>();\n      loadResult.warnings = rawWarnings.map((warning) => {\n        const file = warning.node.source?.input.file;\n        if (file === undefined) {\n          return { text: warning.text };\n        }\n\n        let lines = lineMappings.get(file);\n        if (lines === undefined) {\n          lines = warning.node.source?.input.css.split(/\\r?\\n/);\n          lineMappings.set(file, lines ?? null);\n        }\n\n        return {\n          text: warning.text,\n          location: {\n            file,\n            line: warning.line,\n            column: warning.column - 1,\n            lineText: lines?.[warning.line - 1],\n          },\n        };\n      });\n    }\n\n    for (const resultMessage of postcssResult.messages) {\n      if (resultMessage.type === 'dependency' && typeof resultMessage['file'] === 'string') {\n        loadResult.watchFiles ??= [];\n        loadResult.watchFiles.push(resultMessage['file']);\n      } else if (\n        resultMessage.type === 'dir-dependency' &&\n        typeof resultMessage['dir'] === 'string' &&\n        typeof resultMessage['glob'] === 'string'\n      ) {\n        loadResult.watchFiles ??= [];\n        const dependencies = await glob(resultMessage['glob'], { cwd: resultMessage['dir'] });\n        loadResult.watchFiles.push(...dependencies);\n      }\n    }\n\n    return loadResult;\n  } catch (error) {\n    postcss ??= (await import('postcss')).default;\n    if (error instanceof postcss.CssSyntaxError) {\n      const lines = error.source?.split(/\\r?\\n/);\n\n      return {\n        errors: [\n          {\n            text: error.reason,\n            location: {\n              file: error.file,\n              line: error.line,\n              column: error.column && error.column - 1,\n              lineText: error.line === undefined ? undefined : lines?.[error.line - 1],\n            },\n          },\n        ],\n      };\n    }\n\n    throw error;\n  }\n}\n"]}
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stylesheet-plugin-factory.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/stylesheet-plugin-factory.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 { OnLoadResult, Plugin, PluginBuild } from 'esbuild';\nimport glob from 'fast-glob';\nimport assert from 'node:assert';\nimport { readFile } from 'node:fs/promises';\nimport { extname } from 'node:path';\nimport { LoadResultCache, createCachedLoad } from '../load-result-cache';\n\n/**\n * The lazy-loaded instance of the postcss stylesheet postprocessor.\n * It is only imported and initialized if postcss is needed.\n */\nlet postcss: typeof import('postcss')['default'] | undefined;\n\n/**\n * An object containing the plugin options to use when processing stylesheets.\n */\nexport interface StylesheetPluginOptions {\n  /**\n   * Controls the use and creation of sourcemaps when processing the stylesheets.\n   * If true, sourcemap processing is enabled; if false, disabled.\n   */\n  sourcemap: boolean;\n\n  /**\n   * An optional array of paths that will be searched for stylesheets if the default\n   * resolution process for the stylesheet language does not succeed.\n   */\n  includePaths?: string[];\n\n  /**\n   * Optional component data for any inline styles from Component decorator `styles` fields.\n   * The key is an internal angular resource URI and the value is the stylesheet content.\n   */\n  inlineComponentData?: Record<string, string>;\n\n  /**\n   * Optional information used to load and configure Tailwind CSS. If present, the postcss\n   * will be added to the stylesheet processing with the Tailwind plugin setup as provided\n   * by the configuration file.\n   */\n  tailwindConfiguration?: { file: string; package: string };\n}\n\n/**\n * An array of keywords that indicate Tailwind CSS processing is required for a stylesheet.\n *\n * Based on https://tailwindcss.com/docs/functions-and-directives\n */\nconst TAILWIND_KEYWORDS = ['@tailwind', '@layer', '@apply', '@config', 'theme(', 'screen('];\n\nexport interface StylesheetLanguage {\n  name: string;\n  componentFilter: RegExp;\n  fileFilter: RegExp;\n  process?(\n    data: string,\n    file: string,\n    format: string,\n    options: StylesheetPluginOptions,\n    build: PluginBuild,\n  ): OnLoadResult | Promise<OnLoadResult>;\n}\n\nexport class StylesheetPluginFactory {\n  private postcssProcessor?: import('postcss').Processor;\n\n  constructor(\n    private readonly options: StylesheetPluginOptions,\n    private readonly cache?: LoadResultCache,\n  ) {}\n\n  create(language: Readonly<StylesheetLanguage>): Plugin {\n    // Return a noop plugin if no load actions are required\n    if (!language.process && !this.options.tailwindConfiguration) {\n      return {\n        name: 'angular-' + language.name,\n        setup() {},\n      };\n    }\n\n    const { cache, options } = this;\n    const setupPostcss = async () => {\n      // Return already created processor if present\n      if (this.postcssProcessor) {\n        return this.postcssProcessor;\n      }\n\n      if (options.tailwindConfiguration) {\n        postcss ??= (await import('postcss')).default;\n        const tailwind = await import(options.tailwindConfiguration.package);\n        this.postcssProcessor = postcss().use(\n          tailwind.default({ config: options.tailwindConfiguration.file }),\n        );\n      }\n\n      return this.postcssProcessor;\n    };\n\n    return {\n      name: 'angular-' + language.name,\n      async setup(build) {\n        // Setup postcss if needed\n        const postcssProcessor = await setupPostcss();\n\n        // Add a load callback to support inline Component styles\n        build.onLoad(\n          { filter: language.componentFilter, namespace: 'angular:styles/component' },\n          createCachedLoad(cache, (args) => {\n            const data = options.inlineComponentData?.[args.path];\n            assert(\n              typeof data === 'string',\n              `component style name should always be found [${args.path}]`,\n            );\n\n            const [format, , filename] = args.path.split(';', 3);\n            // Only use postcss if Tailwind processing is required.\n            // NOTE: If postcss is used for more than just Tailwind in the future this check MUST\n            // be updated to account for the additional use.\n            // TODO: use better search algorithm for keywords\n            const needsPostcss =\n              !!postcssProcessor && TAILWIND_KEYWORDS.some((keyword) => data.includes(keyword));\n\n            return processStylesheet(\n              language,\n              data,\n              filename,\n              format,\n              options,\n              build,\n              needsPostcss ? postcssProcessor : undefined,\n            );\n          }),\n        );\n\n        // Add a load callback to support files from disk\n        build.onLoad(\n          { filter: language.fileFilter },\n          createCachedLoad(cache, async (args) => {\n            const data = await readFile(args.path, 'utf-8');\n            const needsPostcss =\n              !!postcssProcessor && TAILWIND_KEYWORDS.some((keyword) => data.includes(keyword));\n\n            return processStylesheet(\n              language,\n              data,\n              args.path,\n              extname(args.path).toLowerCase().slice(1),\n              options,\n              build,\n              needsPostcss ? postcssProcessor : undefined,\n            );\n          }),\n        );\n      },\n    };\n  }\n}\n\nasync function processStylesheet(\n  language: Readonly<StylesheetLanguage>,\n  data: string,\n  filename: string,\n  format: string,\n  options: StylesheetPluginOptions,\n  build: PluginBuild,\n  postcssProcessor: import('postcss').Processor | undefined,\n) {\n  let result: OnLoadResult;\n\n  // Process the input data if the language requires preprocessing\n  if (language.process) {\n    result = await language.process(data, filename, format, options, build);\n  } else {\n    result = {\n      contents: data,\n      loader: 'css',\n      watchFiles: [filename],\n    };\n  }\n\n  // Transform with postcss if needed and there are no errors\n  if (postcssProcessor && result.contents && !result.errors?.length) {\n    const postcssResult = await compileString(\n      typeof result.contents === 'string'\n        ? result.contents\n        : Buffer.from(result.contents).toString('utf-8'),\n      filename,\n      postcssProcessor,\n      options,\n    );\n\n    // Merge results\n    if (postcssResult.errors?.length) {\n      delete result.contents;\n    }\n    if (result.warnings && postcssResult.warnings) {\n      postcssResult.warnings.unshift(...result.warnings);\n    }\n    if (result.watchFiles && postcssResult.watchFiles) {\n      postcssResult.watchFiles.unshift(...result.watchFiles);\n    }\n    if (result.watchDirs && postcssResult.watchDirs) {\n      postcssResult.watchDirs.unshift(...result.watchDirs);\n    }\n    result = {\n      ...result,\n      ...postcssResult,\n    };\n  }\n\n  return result;\n}\n\n/**\n * Compiles the provided CSS stylesheet data using a provided postcss processor and provides an\n * esbuild load result that can be used directly by an esbuild Plugin.\n * @param data The stylesheet content to process.\n * @param filename The name of the file that contains the data.\n * @param postcssProcessor A postcss processor instance to use.\n * @param options The plugin options to control the processing.\n * @returns An esbuild OnLoaderResult object with the processed content, warnings, and/or errors.\n */\nasync function compileString(\n  data: string,\n  filename: string,\n  postcssProcessor: import('postcss').Processor,\n  options: StylesheetPluginOptions,\n): Promise<OnLoadResult> {\n  try {\n    const postcssResult = await postcssProcessor.process(data, {\n      from: filename,\n      to: filename,\n      map: options.sourcemap && {\n        inline: true,\n        sourcesContent: true,\n      },\n    });\n\n    const loadResult: OnLoadResult = {\n      contents: postcssResult.css,\n      loader: 'css',\n    };\n\n    const rawWarnings = postcssResult.warnings();\n    if (rawWarnings.length > 0) {\n      const lineMappings = new Map<string, string[] | null>();\n      loadResult.warnings = rawWarnings.map((warning) => {\n        const file = warning.node.source?.input.file;\n        if (file === undefined) {\n          return { text: warning.text };\n        }\n\n        let lines = lineMappings.get(file);\n        if (lines === undefined) {\n          lines = warning.node.source?.input.css.split(/\\r?\\n/);\n          lineMappings.set(file, lines ?? null);\n        }\n\n        return {\n          text: warning.text,\n          location: {\n            file,\n            line: warning.line,\n            column: warning.column - 1,\n            lineText: lines?.[warning.line - 1],\n          },\n        };\n      });\n    }\n\n    for (const resultMessage of postcssResult.messages) {\n      if (resultMessage.type === 'dependency' && typeof resultMessage['file'] === 'string') {\n        loadResult.watchFiles ??= [];\n        loadResult.watchFiles.push(resultMessage['file']);\n      } else if (\n        resultMessage.type === 'dir-dependency' &&\n        typeof resultMessage['dir'] === 'string' &&\n        typeof resultMessage['glob'] === 'string'\n      ) {\n        loadResult.watchFiles ??= [];\n        const dependencies = await glob(resultMessage['glob'], {\n          absolute: true,\n          cwd: resultMessage['dir'],\n        });\n        loadResult.watchFiles.push(...dependencies);\n      }\n    }\n\n    return loadResult;\n  } catch (error) {\n    postcss ??= (await import('postcss')).default;\n    if (error instanceof postcss.CssSyntaxError) {\n      const lines = error.source?.split(/\\r?\\n/);\n\n      return {\n        errors: [\n          {\n            text: error.reason,\n            location: {\n              file: error.file,\n              line: error.line,\n              column: error.column && error.column - 1,\n              lineText: error.line === undefined ? undefined : lines?.[error.line - 1],\n            },\n          },\n        ],\n      };\n    }\n\n    throw error;\n  }\n}\n"]}
@@ -0,0 +1,11 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ export declare function resolve(specifier: string, context: {}, nextResolve: Function): any;
9
+ export declare function load(url: string, context: {
10
+ format?: string | null;
11
+ }, nextLoad: Function): any;
@@ -0,0 +1,53 @@
1
+ "use strict";
2
+ /**
3
+ * @license
4
+ * Copyright Google LLC All Rights Reserved.
5
+ *
6
+ * Use of this source code is governed by an MIT-style license that can be
7
+ * found in the LICENSE file at https://angular.io/license
8
+ */
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.load = exports.resolve = void 0;
11
+ const node_worker_threads_1 = require("node:worker_threads");
12
+ const url_1 = require("url");
13
+ /**
14
+ * Node.js ESM loader to redirect imports to in memory files.
15
+ * @see: https://nodejs.org/api/esm.html#loaders for more information about loaders.
16
+ */
17
+ const { outputFiles } = node_worker_threads_1.workerData;
18
+ function resolve(specifier, context, nextResolve) {
19
+ if (!isFileProtocol(specifier)) {
20
+ const normalizedSpecifier = specifier.replace(/^\.\//, '');
21
+ if (normalizedSpecifier in outputFiles) {
22
+ return {
23
+ format: 'module',
24
+ shortCircuit: true,
25
+ url: new URL(normalizedSpecifier, 'file:').href,
26
+ };
27
+ }
28
+ }
29
+ // Defer to the next hook in the chain, which would be the
30
+ // Node.js default resolve if this is the last user-specified loader.
31
+ return nextResolve(specifier);
32
+ }
33
+ exports.resolve = resolve;
34
+ function load(url, context, nextLoad) {
35
+ if (isFileProtocol(url)) {
36
+ const source = outputFiles[(0, url_1.fileURLToPath)(url).slice(1)]; // Remove leading slash
37
+ if (source !== undefined) {
38
+ const { format } = context;
39
+ return {
40
+ format,
41
+ shortCircuit: true,
42
+ source,
43
+ };
44
+ }
45
+ }
46
+ // Let Node.js handle all other URLs.
47
+ return nextLoad(url);
48
+ }
49
+ exports.load = load;
50
+ function isFileProtocol(url) {
51
+ return url.startsWith('file://');
52
+ }
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXNtLWluLW1lbW9yeS1maWxlLWxvYWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2FuZ3VsYXJfZGV2a2l0L2J1aWxkX2FuZ3VsYXIvc3JjL3V0aWxzL3NlcnZlci1yZW5kZXJpbmcvZXNtLWluLW1lbW9yeS1maWxlLWxvYWRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7OztHQU1HOzs7QUFFSCw2REFBaUQ7QUFDakQsNkJBQW9DO0FBRXBDOzs7R0FHRztBQUVILE1BQU0sRUFBRSxXQUFXLEVBQUUsR0FBRyxnQ0FFdkIsQ0FBQztBQUVGLFNBQWdCLE9BQU8sQ0FBQyxTQUFpQixFQUFFLE9BQVcsRUFBRSxXQUFxQjtJQUMzRSxJQUFJLENBQUMsY0FBYyxDQUFDLFNBQVMsQ0FBQyxFQUFFO1FBQzlCLE1BQU0sbUJBQW1CLEdBQUcsU0FBUyxDQUFDLE9BQU8sQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFDM0QsSUFBSSxtQkFBbUIsSUFBSSxXQUFXLEVBQUU7WUFDdEMsT0FBTztnQkFDTCxNQUFNLEVBQUUsUUFBUTtnQkFDaEIsWUFBWSxFQUFFLElBQUk7Z0JBQ2xCLEdBQUcsRUFBRSxJQUFJLEdBQUcsQ0FBQyxtQkFBbUIsRUFBRSxPQUFPLENBQUMsQ0FBQyxJQUFJO2FBQ2hELENBQUM7U0FDSDtLQUNGO0lBRUQsMERBQTBEO0lBQzFELHFFQUFxRTtJQUNyRSxPQUFPLFdBQVcsQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUNoQyxDQUFDO0FBZkQsMEJBZUM7QUFFRCxTQUFnQixJQUFJLENBQUMsR0FBVyxFQUFFLE9BQW1DLEVBQUUsUUFBa0I7SUFDdkYsSUFBSSxjQUFjLENBQUMsR0FBRyxDQUFDLEVBQUU7UUFDdkIsTUFBTSxNQUFNLEdBQUcsV0FBVyxDQUFDLElBQUEsbUJBQWEsRUFBQyxHQUFHLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLHVCQUF1QjtRQUNoRixJQUFJLE1BQU0sS0FBSyxTQUFTLEVBQUU7WUFDeEIsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLE9BQU8sQ0FBQztZQUUzQixPQUFPO2dCQUNMLE1BQU07Z0JBQ04sWUFBWSxFQUFFLElBQUk7Z0JBQ2xCLE1BQU07YUFDUCxDQUFDO1NBQ0g7S0FDRjtJQUVELHFDQUFxQztJQUNyQyxPQUFPLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUN2QixDQUFDO0FBaEJELG9CQWdCQztBQUVELFNBQVMsY0FBYyxDQUFDLEdBQVc7SUFDakMsT0FBTyxHQUFHLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0FBQ25DLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHsgd29ya2VyRGF0YSB9IGZyb20gJ25vZGU6d29ya2VyX3RocmVhZHMnO1xuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gJ3VybCc7XG5cbi8qKlxuICogTm9kZS5qcyBFU00gbG9hZGVyIHRvIHJlZGlyZWN0IGltcG9ydHMgdG8gaW4gbWVtb3J5IGZpbGVzLlxuICogQHNlZTogaHR0cHM6Ly9ub2RlanMub3JnL2FwaS9lc20uaHRtbCNsb2FkZXJzIGZvciBtb3JlIGluZm9ybWF0aW9uIGFib3V0IGxvYWRlcnMuXG4gKi9cblxuY29uc3QgeyBvdXRwdXRGaWxlcyB9ID0gd29ya2VyRGF0YSBhcyB7XG4gIG91dHB1dEZpbGVzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmUoc3BlY2lmaWVyOiBzdHJpbmcsIGNvbnRleHQ6IHt9LCBuZXh0UmVzb2x2ZTogRnVuY3Rpb24pIHtcbiAgaWYgKCFpc0ZpbGVQcm90b2NvbChzcGVjaWZpZXIpKSB7XG4gICAgY29uc3Qgbm9ybWFsaXplZFNwZWNpZmllciA9IHNwZWNpZmllci5yZXBsYWNlKC9eXFwuXFwvLywgJycpO1xuICAgIGlmIChub3JtYWxpemVkU3BlY2lmaWVyIGluIG91dHB1dEZpbGVzKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBmb3JtYXQ6ICdtb2R1bGUnLFxuICAgICAgICBzaG9ydENpcmN1aXQ6IHRydWUsXG4gICAgICAgIHVybDogbmV3IFVSTChub3JtYWxpemVkU3BlY2lmaWVyLCAnZmlsZTonKS5ocmVmLFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICAvLyBEZWZlciB0byB0aGUgbmV4dCBob29rIGluIHRoZSBjaGFpbiwgd2hpY2ggd291bGQgYmUgdGhlXG4gIC8vIE5vZGUuanMgZGVmYXVsdCByZXNvbHZlIGlmIHRoaXMgaXMgdGhlIGxhc3QgdXNlci1zcGVjaWZpZWQgbG9hZGVyLlxuICByZXR1cm4gbmV4dFJlc29sdmUoc3BlY2lmaWVyKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGxvYWQodXJsOiBzdHJpbmcsIGNvbnRleHQ6IHsgZm9ybWF0Pzogc3RyaW5nIHwgbnVsbCB9LCBuZXh0TG9hZDogRnVuY3Rpb24pIHtcbiAgaWYgKGlzRmlsZVByb3RvY29sKHVybCkpIHtcbiAgICBjb25zdCBzb3VyY2UgPSBvdXRwdXRGaWxlc1tmaWxlVVJMVG9QYXRoKHVybCkuc2xpY2UoMSldOyAvLyBSZW1vdmUgbGVhZGluZyBzbGFzaFxuICAgIGlmIChzb3VyY2UgIT09IHVuZGVmaW5lZCkge1xuICAgICAgY29uc3QgeyBmb3JtYXQgfSA9IGNvbnRleHQ7XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIGZvcm1hdCxcbiAgICAgICAgc2hvcnRDaXJjdWl0OiB0cnVlLFxuICAgICAgICBzb3VyY2UsXG4gICAgICB9O1xuICAgIH1cbiAgfVxuXG4gIC8vIExldCBOb2RlLmpzIGhhbmRsZSBhbGwgb3RoZXIgVVJMcy5cbiAgcmV0dXJuIG5leHRMb2FkKHVybCk7XG59XG5cbmZ1bmN0aW9uIGlzRmlsZVByb3RvY29sKHVybDogc3RyaW5nKTogYm9vbGVhbiB7XG4gIHJldHVybiB1cmwuc3RhcnRzV2l0aCgnZmlsZTovLycpO1xufVxuIl19
@@ -0,0 +1,22 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ import { OutputFile } from 'esbuild';
9
+ interface PrerenderOptions {
10
+ routesFile?: string;
11
+ discoverRoutes?: boolean;
12
+ routes?: string[];
13
+ }
14
+ interface AppShellOptions {
15
+ route?: string;
16
+ }
17
+ export declare function prerenderPages(tsConfigPath: string, appShellOptions: AppShellOptions | undefined, prerenderOptions: PrerenderOptions | undefined, outputFiles: Readonly<OutputFile[]>, document: string, inlineCriticalCss?: boolean, maxThreads?: number): Promise<{
18
+ output: Record<string, string>;
19
+ warnings: string[];
20
+ errors: string[];
21
+ }>;
22
+ export {};
@@ -0,0 +1,127 @@
1
+ "use strict";
2
+ /**
3
+ * @license
4
+ * Copyright Google LLC All Rights Reserved.
5
+ *
6
+ * Use of this source code is governed by an MIT-style license that can be
7
+ * found in the LICENSE file at https://angular.io/license
8
+ */
9
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ var desc = Object.getOwnPropertyDescriptor(m, k);
12
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
13
+ desc = { enumerable: true, get: function() { return m[k]; } };
14
+ }
15
+ Object.defineProperty(o, k2, desc);
16
+ }) : (function(o, m, k, k2) {
17
+ if (k2 === undefined) k2 = k;
18
+ o[k2] = m[k];
19
+ }));
20
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
21
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
22
+ }) : function(o, v) {
23
+ o["default"] = v;
24
+ });
25
+ var __importStar = (this && this.__importStar) || function (mod) {
26
+ if (mod && mod.__esModule) return mod;
27
+ var result = {};
28
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
29
+ __setModuleDefault(result, mod);
30
+ return result;
31
+ };
32
+ var __importDefault = (this && this.__importDefault) || function (mod) {
33
+ return (mod && mod.__esModule) ? mod : { "default": mod };
34
+ };
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.prerenderPages = void 0;
37
+ const promises_1 = require("node:fs/promises");
38
+ const node_path_1 = require("node:path");
39
+ const piscina_1 = __importDefault(require("piscina"));
40
+ async function prerenderPages(tsConfigPath, appShellOptions = {}, prerenderOptions = {}, outputFiles, document, inlineCriticalCss, maxThreads = 1) {
41
+ const allRoutes = await getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions);
42
+ const outputFilesForWorker = {};
43
+ for (const { text, path } of outputFiles) {
44
+ switch ((0, node_path_1.extname)(path)) {
45
+ case '.mjs': // Contains the server runnable application code.
46
+ case '.css': // Global styles for critical CSS inlining.
47
+ outputFilesForWorker[path] = text;
48
+ break;
49
+ }
50
+ }
51
+ const renderWorker = new piscina_1.default({
52
+ filename: require.resolve('./render-worker'),
53
+ maxThreads: Math.min(allRoutes.size, maxThreads),
54
+ workerData: {
55
+ outputFiles: outputFilesForWorker,
56
+ inlineCriticalCss,
57
+ document,
58
+ },
59
+ execArgv: [
60
+ '--no-warnings',
61
+ '--loader',
62
+ require.resolve('./esm-in-memory-file-loader.js'),
63
+ ],
64
+ });
65
+ const output = {};
66
+ const warnings = [];
67
+ const errors = [];
68
+ try {
69
+ const renderingPromises = [];
70
+ for (const route of allRoutes) {
71
+ const isAppShellRoute = appShellOptions.route === route;
72
+ const serverContext = isAppShellRoute ? 'app-shell' : 'ssg';
73
+ const render = renderWorker.run({ route, serverContext });
74
+ const renderResult = render.then(({ content, warnings, errors }) => {
75
+ if (content !== undefined) {
76
+ const outPath = isAppShellRoute ? 'index.html' : node_path_1.posix.join(route, 'index.html');
77
+ output[outPath] = content;
78
+ }
79
+ if (warnings) {
80
+ warnings.push(...warnings);
81
+ }
82
+ if (errors) {
83
+ errors.push(...errors);
84
+ }
85
+ });
86
+ renderingPromises.push(renderResult);
87
+ }
88
+ await Promise.all(renderingPromises);
89
+ }
90
+ finally {
91
+ void renderWorker.destroy();
92
+ }
93
+ return {
94
+ errors,
95
+ warnings,
96
+ output,
97
+ };
98
+ }
99
+ exports.prerenderPages = prerenderPages;
100
+ async function getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions) {
101
+ const { routesFile, discoverRoutes, routes: existingRoutes } = prerenderOptions;
102
+ const routes = new Set(existingRoutes);
103
+ const { route: appShellRoute } = appShellOptions;
104
+ if (appShellRoute !== undefined) {
105
+ routes.add(appShellRoute);
106
+ }
107
+ if (routesFile) {
108
+ const routesFromFile = (await (0, promises_1.readFile)(routesFile, 'utf8')).split(/\r?\n/);
109
+ for (let route of routesFromFile) {
110
+ route = route.trim();
111
+ if (route) {
112
+ routes.add(route);
113
+ }
114
+ }
115
+ }
116
+ if (discoverRoutes) {
117
+ const { parseAngularRoutes } = await Promise.resolve().then(() => __importStar(require('guess-parser')));
118
+ for (const { path } of parseAngularRoutes(tsConfigPath)) {
119
+ // Exclude dynamic routes as these cannot be pre-rendered.
120
+ if (!/[*:]/.test(path)) {
121
+ routes.add(path);
122
+ }
123
+ }
124
+ }
125
+ return routes;
126
+ }
127
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prerender.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/utils/server-rendering/prerender.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 { OutputFile } from 'esbuild';\nimport { readFile } from 'node:fs/promises';\nimport { extname, posix } from 'node:path';\nimport Piscina from 'piscina';\nimport type { RenderResult, ServerContext } from './render-page';\nimport type { WorkerData } from './render-worker';\n\ninterface PrerenderOptions {\n  routesFile?: string;\n  discoverRoutes?: boolean;\n  routes?: string[];\n}\n\ninterface AppShellOptions {\n  route?: string;\n}\n\nexport async function prerenderPages(\n  tsConfigPath: string,\n  appShellOptions: AppShellOptions = {},\n  prerenderOptions: PrerenderOptions = {},\n  outputFiles: Readonly<OutputFile[]>,\n  document: string,\n  inlineCriticalCss?: boolean,\n  maxThreads = 1,\n): Promise<{\n  output: Record<string, string>;\n  warnings: string[];\n  errors: string[];\n}> {\n  const allRoutes = await getAllRoutes(tsConfigPath, appShellOptions, prerenderOptions);\n  const outputFilesForWorker: Record<string, string> = {};\n\n  for (const { text, path } of outputFiles) {\n    switch (extname(path)) {\n      case '.mjs': // Contains the server runnable application code.\n      case '.css': // Global styles for critical CSS inlining.\n        outputFilesForWorker[path] = text;\n        break;\n    }\n  }\n\n  const renderWorker = new Piscina({\n    filename: require.resolve('./render-worker'),\n    maxThreads: Math.min(allRoutes.size, maxThreads),\n    workerData: {\n      outputFiles: outputFilesForWorker,\n      inlineCriticalCss,\n      document,\n    } as WorkerData,\n    execArgv: [\n      '--no-warnings', // Suppress `ExperimentalWarning: Custom ESM Loaders is an experimental feature...`.\n      '--loader',\n      require.resolve('./esm-in-memory-file-loader.js'),\n    ],\n  });\n\n  const output: Record<string, string> = {};\n  const warnings: string[] = [];\n  const errors: string[] = [];\n\n  try {\n    const renderingPromises: Promise<void>[] = [];\n\n    for (const route of allRoutes) {\n      const isAppShellRoute = appShellOptions.route === route;\n      const serverContext: ServerContext = isAppShellRoute ? 'app-shell' : 'ssg';\n\n      const render: Promise<RenderResult> = renderWorker.run({ route, serverContext });\n      const renderResult: Promise<void> = render.then(({ content, warnings, errors }) => {\n        if (content !== undefined) {\n          const outPath = isAppShellRoute ? 'index.html' : posix.join(route, 'index.html');\n          output[outPath] = content;\n        }\n\n        if (warnings) {\n          warnings.push(...warnings);\n        }\n\n        if (errors) {\n          errors.push(...errors);\n        }\n      });\n\n      renderingPromises.push(renderResult);\n    }\n\n    await Promise.all(renderingPromises);\n  } finally {\n    void renderWorker.destroy();\n  }\n\n  return {\n    errors,\n    warnings,\n    output,\n  };\n}\n\nasync function getAllRoutes(\n  tsConfigPath: string,\n  appShellOptions: AppShellOptions,\n  prerenderOptions: PrerenderOptions,\n): Promise<Set<string>> {\n  const { routesFile, discoverRoutes, routes: existingRoutes } = prerenderOptions;\n  const routes = new Set(existingRoutes);\n\n  const { route: appShellRoute } = appShellOptions;\n  if (appShellRoute !== undefined) {\n    routes.add(appShellRoute);\n  }\n\n  if (routesFile) {\n    const routesFromFile = (await readFile(routesFile, 'utf8')).split(/\\r?\\n/);\n    for (let route of routesFromFile) {\n      route = route.trim();\n      if (route) {\n        routes.add(route);\n      }\n    }\n  }\n\n  if (discoverRoutes) {\n    const { parseAngularRoutes } = await import('guess-parser');\n    for (const { path } of parseAngularRoutes(tsConfigPath)) {\n      // Exclude dynamic routes as these cannot be pre-rendered.\n      if (!/[*:]/.test(path)) {\n        routes.add(path);\n      }\n    }\n  }\n\n  return routes;\n}\n"]}
@@ -0,0 +1,38 @@
1
+ /**
2
+ * @license
3
+ * Copyright Google LLC All Rights Reserved.
4
+ *
5
+ * Use of this source code is governed by an MIT-style license that can be
6
+ * found in the LICENSE file at https://angular.io/license
7
+ */
8
+ import type { ApplicationRef, Type } from '@angular/core';
9
+ import type { renderApplication, renderModule, ɵSERVER_CONTEXT } from '@angular/platform-server';
10
+ export interface RenderOptions {
11
+ route: string;
12
+ serverContext: ServerContext;
13
+ outputFiles: Record<string, string>;
14
+ document: string;
15
+ inlineCriticalCss?: boolean;
16
+ loadBundle?: (path: string) => Promise<MainServerBundleExports>;
17
+ }
18
+ export interface RenderResult {
19
+ errors?: string[];
20
+ warnings?: string[];
21
+ content?: string;
22
+ }
23
+ export type ServerContext = 'app-shell' | 'ssg' | 'ssr';
24
+ interface MainServerBundleExports {
25
+ /** An internal token that allows providing extra information about the server context. */
26
+ ɵSERVER_CONTEXT: typeof ɵSERVER_CONTEXT;
27
+ /** Render an NgModule application. */
28
+ renderModule: typeof renderModule;
29
+ /** Method to render a standalone application. */
30
+ renderApplication: typeof renderApplication;
31
+ /** Standalone application bootstrapping function. */
32
+ default: (() => Promise<ApplicationRef>) | Type<unknown>;
33
+ }
34
+ /**
35
+ * Renders each route in routes and writes them to <outputPath>/<route>/index.html.
36
+ */
37
+ export declare function renderPage({ route, serverContext, document, inlineCriticalCss, outputFiles, loadBundle, }: RenderOptions): Promise<RenderResult>;
38
+ export {};