@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
158 lines • 20.6 kB
JavaScript
;
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.bundleComponentStylesheet = exports.createStylesheetBundleOptions = void 0;
const node_crypto_1 = require("node:crypto");
const node_path_1 = __importDefault(require("node:path"));
const bundler_context_1 = require("../bundler-context");
const css_language_1 = require("./css-language");
const css_resource_plugin_1 = require("./css-resource-plugin");
const less_language_1 = require("./less-language");
const sass_language_1 = require("./sass-language");
const stylesheet_plugin_factory_1 = require("./stylesheet-plugin-factory");
/**
* A counter for component styles used to generate unique build-time identifiers for each stylesheet.
*/
let componentStyleCounter = 0;
function createStylesheetBundleOptions(options, cache, inlineComponentData) {
// Ensure preprocessor include paths are absolute based on the workspace root
const includePaths = options.includePaths?.map((includePath) => node_path_1.default.resolve(options.workspaceRoot, includePath));
const pluginFactory = new stylesheet_plugin_factory_1.StylesheetPluginFactory({
sourcemap: !!options.sourcemap,
includePaths,
inlineComponentData,
tailwindConfiguration: options.tailwindConfiguration,
}, cache);
return {
absWorkingDir: options.workspaceRoot,
bundle: true,
entryNames: options.outputNames.bundles,
assetNames: options.outputNames.media,
logLevel: 'silent',
minify: options.optimization,
metafile: true,
sourcemap: options.sourcemap,
outdir: options.workspaceRoot,
write: false,
platform: 'browser',
target: options.target,
preserveSymlinks: options.preserveSymlinks,
external: options.externalDependencies,
conditions: ['style', 'sass'],
mainFields: ['style', 'sass'],
plugins: [
pluginFactory.create(sass_language_1.SassStylesheetLanguage),
pluginFactory.create(less_language_1.LessStylesheetLanguage),
pluginFactory.create(css_language_1.CssStylesheetLanguage),
(0, css_resource_plugin_1.createCssResourcePlugin)(cache),
],
};
}
exports.createStylesheetBundleOptions = createStylesheetBundleOptions;
/**
* Bundles a component stylesheet. The stylesheet can be either an inline stylesheet that
* is contained within the Component's metadata definition or an external file referenced
* from the Component's metadata definition.
*
* @param identifier A unique string identifier for the component stylesheet.
* @param language The language of the stylesheet such as `css` or `scss`.
* @param data The string content of the stylesheet.
* @param filename The filename representing the source of the stylesheet content.
* @param inline If true, the stylesheet source is within the component metadata;
* if false, the source is a stylesheet file.
* @param options An object containing the stylesheet bundling options.
* @returns An object containing the output of the bundling operation.
*/
async function bundleComponentStylesheet(language, data, filename, inline, options, cache) {
const namespace = 'angular:styles/component';
// Use a hash of the inline stylesheet content to ensure a consistent identifier. External stylesheets will resolve
// to the actual stylesheet file path.
// TODO: Consider xxhash instead for hashing
const id = inline ? (0, node_crypto_1.createHash)('sha256').update(data).digest('hex') : componentStyleCounter++;
const entry = [language, id, filename].join(';');
const buildOptions = createStylesheetBundleOptions(options, cache, { [entry]: data });
buildOptions.entryPoints = [`${namespace};${entry}`];
buildOptions.plugins.push({
name: 'angular-component-styles',
setup(build) {
build.onResolve({ filter: /^angular:styles\/component;/ }, (args) => {
if (args.kind !== 'entry-point') {
return null;
}
if (inline) {
return {
path: entry,
namespace,
};
}
else {
return {
path: filename,
};
}
});
build.onLoad({ filter: /^css;/, namespace }, async () => {
return {
contents: data,
loader: 'css',
resolveDir: node_path_1.default.dirname(filename),
};
});
},
});
// Execute esbuild
const context = new bundler_context_1.BundlerContext(options.workspaceRoot, false, buildOptions);
const result = await context.bundle();
// Extract the result of the bundling from the output files
let contents = '';
let map;
let outputPath;
const resourceFiles = [];
if (!result.errors) {
for (const outputFile of result.outputFiles) {
const filename = node_path_1.default.basename(outputFile.path);
if (filename.endsWith('.css')) {
outputPath = outputFile.path;
contents = outputFile.text;
}
else if (filename.endsWith('.css.map')) {
map = outputFile.text;
}
else {
// The output files could also contain resources (images/fonts/etc.) that were referenced
resourceFiles.push(outputFile);
}
}
}
let metafile;
if (!result.errors) {
metafile = result.metafile;
// Remove entryPoint fields from outputs to prevent the internal component styles from being
// treated as initial files. Also mark the entry as a component resource for stat reporting.
Object.values(metafile.outputs).forEach((output) => {
delete output.entryPoint;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
output['ng-component'] = true;
});
}
return {
errors: result.errors,
warnings: result.warnings,
contents,
map,
path: outputPath,
resourceFiles,
metafile,
};
}
exports.bundleComponentStylesheet = bundleComponentStylesheet;
//# 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":";AAAA;;;;;;GAMG;;;;;;AAGH,6CAAyC;AACzC,0DAA6B;AAC7B,wDAAoD;AAEpD,iDAAuD;AACvD,+DAAgE;AAChE,mDAAyD;AACzD,mDAAyD;AACzD,2EAAsE;AAEtE;;GAEG;AACH,IAAI,qBAAqB,GAAG,CAAC,CAAC;AAc9B,SAAgB,6BAA6B,CAC3C,OAAgC,EAChC,KAAuB,EACvB,mBAA4C;IAE5C,6EAA6E;IAC7E,MAAM,YAAY,GAAG,OAAO,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC,WAAW,EAAE,EAAE,CAC7D,mBAAI,CAAC,OAAO,CAAC,OAAO,CAAC,aAAa,EAAE,WAAW,CAAC,CACjD,CAAC;IAEF,MAAM,aAAa,GAAG,IAAI,mDAAuB,CAC/C;QACE,SAAS,EAAE,CAAC,CAAC,OAAO,CAAC,SAAS;QAC9B,YAAY;QACZ,mBAAmB;QACnB,qBAAqB,EAAE,OAAO,CAAC,qBAAqB;KACrD,EACD,KAAK,CACN,CAAC;IAEF,OAAO;QACL,aAAa,EAAE,OAAO,CAAC,aAAa;QACpC,MAAM,EAAE,IAAI;QACZ,UAAU,EAAE,OAAO,CAAC,WAAW,CAAC,OAAO;QACvC,UAAU,EAAE,OAAO,CAAC,WAAW,CAAC,KAAK;QACrC,QAAQ,EAAE,QAAQ;QAClB,MAAM,EAAE,OAAO,CAAC,YAAY;QAC5B,QAAQ,EAAE,IAAI;QACd,SAAS,EAAE,OAAO,CAAC,SAAS;QAC5B,MAAM,EAAE,OAAO,CAAC,aAAa;QAC7B,KAAK,EAAE,KAAK;QACZ,QAAQ,EAAE,SAAS;QACnB,MAAM,EAAE,OAAO,CAAC,MAAM;QACtB,gBAAgB,EAAE,OAAO,CAAC,gBAAgB;QAC1C,QAAQ,EAAE,OAAO,CAAC,oBAAoB;QACtC,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC;QAC7B,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC;QAC7B,OAAO,EAAE;YACP,aAAa,CAAC,MAAM,CAAC,sCAAsB,CAAC;YAC5C,aAAa,CAAC,MAAM,CAAC,sCAAsB,CAAC;YAC5C,aAAa,CAAC,MAAM,CAAC,oCAAqB,CAAC;YAC3C,IAAA,6CAAuB,EAAC,KAAK,CAAC;SAC/B;KACF,CAAC;AACJ,CAAC;AA5CD,sEA4CC;AAED;;;;;;;;;;;;;GAaG;AACI,KAAK,UAAU,yBAAyB,CAC7C,QAAgB,EAChB,IAAY,EACZ,QAAgB,EAChB,MAAe,EACf,OAAgC,EAChC,KAAuB;IAEvB,MAAM,SAAS,GAAG,0BAA0B,CAAC;IAC7C,mHAAmH;IACnH,sCAAsC;IACtC,4CAA4C;IAC5C,MAAM,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,IAAA,wBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,qBAAqB,EAAE,CAAC;IAC9F,MAAM,KAAK,GAAG,CAAC,QAAQ,EAAE,EAAE,EAAE,QAAQ,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAEjD,MAAM,YAAY,GAAG,6BAA6B,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;IACtF,YAAY,CAAC,WAAW,GAAG,CAAC,GAAG,SAAS,IAAI,KAAK,EAAE,CAAC,CAAC;IACrD,YAAY,CAAC,OAAO,CAAC,IAAI,CAAC;QACxB,IAAI,EAAE,0BAA0B;QAChC,KAAK,CAAC,KAAK;YACT,KAAK,CAAC,SAAS,CAAC,EAAE,MAAM,EAAE,6BAA6B,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE;gBAClE,IAAI,IAAI,CAAC,IAAI,KAAK,aAAa,EAAE;oBAC/B,OAAO,IAAI,CAAC;iBACb;gBAED,IAAI,MAAM,EAAE;oBACV,OAAO;wBACL,IAAI,EAAE,KAAK;wBACX,SAAS;qBACV,CAAC;iBACH;qBAAM;oBACL,OAAO;wBACL,IAAI,EAAE,QAAQ;qBACf,CAAC;iBACH;YACH,CAAC,CAAC,CAAC;YACH,KAAK,CAAC,MAAM,CAAC,EAAE,MAAM,EAAE,OAAO,EAAE,SAAS,EAAE,EAAE,KAAK,IAAI,EAAE;gBACtD,OAAO;oBACL,QAAQ,EAAE,IAAI;oBACd,MAAM,EAAE,KAAK;oBACb,UAAU,EAAE,mBAAI,CAAC,OAAO,CAAC,QAAQ,CAAC;iBACnC,CAAC;YACJ,CAAC,CAAC,CAAC;QACL,CAAC;KACF,CAAC,CAAC;IAEH,kBAAkB;IAClB,MAAM,OAAO,GAAG,IAAI,gCAAc,CAAC,OAAO,CAAC,aAAa,EAAE,KAAK,EAAE,YAAY,CAAC,CAAC;IAC/E,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,MAAM,EAAE,CAAC;IAEtC,2DAA2D;IAC3D,IAAI,QAAQ,GAAG,EAAE,CAAC;IAClB,IAAI,GAAG,CAAC;IACR,IAAI,UAAU,CAAC;IACf,MAAM,aAAa,GAAiB,EAAE,CAAC;IACvC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE;QAClB,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,WAAW,EAAE;YAC3C,MAAM,QAAQ,GAAG,mBAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;YAChD,IAAI,QAAQ,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE;gBAC7B,UAAU,GAAG,UAAU,CAAC,IAAI,CAAC;gBAC7B,QAAQ,GAAG,UAAU,CAAC,IAAI,CAAC;aAC5B;iBAAM,IAAI,QAAQ,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;gBACxC,GAAG,GAAG,UAAU,CAAC,IAAI,CAAC;aACvB;iBAAM;gBACL,yFAAyF;gBACzF,aAAa,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aAChC;SACF;KACF;IAED,IAAI,QAAQ,CAAC;IACb,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE;QAClB,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;QAC3B,4FAA4F;QAC5F,4FAA4F;QAC5F,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE;YACjD,OAAO,MAAM,CAAC,UAAU,CAAC;YACzB,8DAA8D;YAC7D,MAAc,CAAC,cAAc,CAAC,GAAG,IAAI,CAAC;QACzC,CAAC,CAAC,CAAC;KACJ;IAED,OAAO;QACL,MAAM,EAAE,MAAM,CAAC,MAAM;QACrB,QAAQ,EAAE,MAAM,CAAC,QAAQ;QACzB,QAAQ;QACR,GAAG;QACH,IAAI,EAAE,UAAU;QAChB,aAAa;QACb,QAAQ;KACT,CAAC;AACJ,CAAC;AA3FD,8DA2FC","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"]}