UNPKG

@angular-devkit/build-angular

Version:
293 lines • 33.6 kB
"use strict"; /** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.isPlatformServerInstalled = exports.getStatsOptions = exports.assetPatterns = exports.globalScriptsByBundleName = exports.getCacheSettings = exports.normalizeGlobalStyles = exports.getInstrumentationExcludedPaths = exports.assetNameTemplateFactory = exports.normalizeExtraEntryPoints = exports.getOutputHashFormat = void 0; const crypto_1 = require("crypto"); const fast_glob_1 = __importDefault(require("fast-glob")); const path = __importStar(require("path")); const schema_1 = require("../../../builders/browser/schema"); const package_version_1 = require("../../../utils/package-version"); function getOutputHashFormat(outputHashing = schema_1.OutputHashing.None, length = 20) { const hashTemplate = `.[contenthash:${length}]`; switch (outputHashing) { case 'media': return { chunk: '', extract: '', file: hashTemplate, script: '', }; case 'bundles': return { chunk: hashTemplate, extract: hashTemplate, file: '', script: hashTemplate, }; case 'all': return { chunk: hashTemplate, extract: hashTemplate, file: hashTemplate, script: hashTemplate, }; case 'none': default: return { chunk: '', extract: '', file: '', script: '', }; } } exports.getOutputHashFormat = getOutputHashFormat; function normalizeExtraEntryPoints(extraEntryPoints, defaultBundleName) { return extraEntryPoints.map((entry) => { if (typeof entry === 'string') { return { input: entry, inject: true, bundleName: defaultBundleName }; } const { inject = true, ...newEntry } = entry; let bundleName; if (entry.bundleName) { bundleName = entry.bundleName; } else if (!inject) { // Lazy entry points use the file name as bundle name. bundleName = path.parse(entry.input).name; } else { bundleName = defaultBundleName; } return { ...newEntry, inject, bundleName }; }); } exports.normalizeExtraEntryPoints = normalizeExtraEntryPoints; function assetNameTemplateFactory(hashFormat) { const visitedFiles = new Map(); return (resourcePath) => { if (hashFormat.file) { // File names are hashed therefore we don't need to handle files with the same file name. return `[name]${hashFormat.file}.[ext]`; } const filename = path.basename(resourcePath); // Check if the file with the same name has already been processed. const visited = visitedFiles.get(filename); if (!visited) { // Not visited. visitedFiles.set(filename, resourcePath); return filename; } else if (visited === resourcePath) { // Same file. return filename; } // File has the same name but it's in a different location. return '[path][name].[ext]'; }; } exports.assetNameTemplateFactory = assetNameTemplateFactory; function getInstrumentationExcludedPaths(root, excludedPaths) { const excluded = new Set(); for (const excludeGlob of excludedPaths) { const excludePath = excludeGlob[0] === '/' ? excludeGlob.slice(1) : excludeGlob; fast_glob_1.default.sync(excludePath, { cwd: root }).forEach((p) => excluded.add(path.join(root, p))); } return excluded; } exports.getInstrumentationExcludedPaths = getInstrumentationExcludedPaths; function normalizeGlobalStyles(styleEntrypoints) { var _a; const entryPoints = {}; const noInjectNames = []; if (styleEntrypoints.length === 0) { return { entryPoints, noInjectNames }; } for (const style of normalizeExtraEntryPoints(styleEntrypoints, 'styles')) { // Add style entry points. entryPoints[_a = style.bundleName] ?? (entryPoints[_a] = []); entryPoints[style.bundleName].push(style.input); // Add non injected styles to the list. if (!style.inject) { noInjectNames.push(style.bundleName); } } return { entryPoints, noInjectNames }; } exports.normalizeGlobalStyles = normalizeGlobalStyles; function getCacheSettings(wco, angularVersion) { const { enabled, path: cacheDirectory } = wco.buildOptions.cache; if (enabled) { return { type: 'filesystem', profile: wco.buildOptions.verbose, cacheDirectory: path.join(cacheDirectory, 'angular-webpack'), maxMemoryGenerations: 1, // We use the versions and build options as the cache name. The Webpack configurations are too // dynamic and shared among different build types: test, build and serve. // None of which are "named". name: (0, crypto_1.createHash)('sha1') .update(angularVersion) .update(package_version_1.VERSION) .update(wco.projectRoot) .update(JSON.stringify(wco.tsConfig)) .update(JSON.stringify({ ...wco.buildOptions, // Needed because outputPath changes on every build when using i18n extraction // https://github.com/angular/angular-cli/blob/736a5f89deaca85f487b78aec9ff66d4118ceb6a/packages/angular_devkit/build_angular/src/utils/i18n-options.ts#L264-L265 outputPath: undefined, })) .digest('hex'), }; } if (wco.buildOptions.watch) { return { type: 'memory', maxGenerations: 1, }; } return false; } exports.getCacheSettings = getCacheSettings; function globalScriptsByBundleName(scripts) { return normalizeExtraEntryPoints(scripts, 'scripts').reduce((prev, curr) => { const { bundleName, inject, input } = curr; const existingEntry = prev.find((el) => el.bundleName === bundleName); if (existingEntry) { if (existingEntry.inject && !inject) { // All entries have to be lazy for the bundle to be lazy. throw new Error(`The ${bundleName} bundle is mixing injected and non-injected scripts.`); } existingEntry.paths.push(input); } else { prev.push({ bundleName, inject, paths: [input], }); } return prev; }, []); } exports.globalScriptsByBundleName = globalScriptsByBundleName; function assetPatterns(root, assets) { return assets.map((asset, index) => { // Resolve input paths relative to workspace root and add slash at the end. // eslint-disable-next-line prefer-const let { input, output, ignore = [], glob } = asset; input = path.resolve(root, input).replace(/\\/g, '/'); input = input.endsWith('/') ? input : input + '/'; output = output.endsWith('/') ? output : output + '/'; if (output.startsWith('..')) { throw new Error('An asset cannot be written to a location outside of the output path.'); } return { context: input, // Now we remove starting slash to make Webpack place it from the output root. to: output.replace(/^\//, ''), from: glob, noErrorOnMissing: true, force: true, globOptions: { dot: true, followSymbolicLinks: !!asset.followSymlinks, ignore: [ '.gitkeep', '**/.DS_Store', '**/Thumbs.db', // Negate patterns needs to be absolute because copy-webpack-plugin uses absolute globs which // causes negate patterns not to match. // See: https://github.com/webpack-contrib/copy-webpack-plugin/issues/498#issuecomment-639327909 ...ignore, ].map((i) => path.posix.join(input, i)), }, priority: index, }; }); } exports.assetPatterns = assetPatterns; function getStatsOptions(verbose = false) { const webpackOutputOptions = { all: false, colors: true, hash: true, timings: true, chunks: true, builtAt: true, warnings: true, errors: true, assets: true, cachedAssets: true, // Needed for markAsyncChunksNonInitial. ids: true, entrypoints: true, }; const verboseWebpackOutputOptions = { // The verbose output will most likely be piped to a file, so colors just mess it up. colors: false, usedExports: true, optimizationBailout: true, reasons: true, children: true, assets: true, version: true, chunkModules: true, errorDetails: true, errorStack: true, moduleTrace: true, logging: 'verbose', modulesSpace: Infinity, }; return verbose ? { ...webpackOutputOptions, ...verboseWebpackOutputOptions } : webpackOutputOptions; } exports.getStatsOptions = getStatsOptions; /** * @param root the workspace root * @returns `true` when `@angular/platform-server` is installed. */ function isPlatformServerInstalled(root) { try { require.resolve('@angular/platform-server', { paths: [root] }); return true; } catch { return false; } } exports.isPlatformServerInstalled = isPlatformServerInstalled; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/webpack/utils/helpers.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 { ObjectPattern } from 'copy-webpack-plugin';\nimport { createHash } from 'crypto';\nimport glob from 'fast-glob';\nimport * as path from 'path';\nimport type { Configuration, WebpackOptionsNormalized } from 'webpack';\nimport {\n  AssetPatternClass,\n  OutputHashing,\n  ScriptElement,\n  StyleElement,\n} from '../../../builders/browser/schema';\nimport { WebpackConfigOptions } from '../../../utils/build-options';\nimport { VERSION } from '../../../utils/package-version';\n\nexport interface HashFormat {\n  chunk: string;\n  extract: string;\n  file: string;\n  script: string;\n}\n\nexport type WebpackStatsOptions = Exclude<Configuration['stats'], string | boolean | undefined>;\n\nexport function getOutputHashFormat(outputHashing = OutputHashing.None, length = 20): HashFormat {\n  const hashTemplate = `.[contenthash:${length}]`;\n\n  switch (outputHashing) {\n    case 'media':\n      return {\n        chunk: '',\n        extract: '',\n        file: hashTemplate,\n        script: '',\n      };\n    case 'bundles':\n      return {\n        chunk: hashTemplate,\n        extract: hashTemplate,\n        file: '',\n        script: hashTemplate,\n      };\n    case 'all':\n      return {\n        chunk: hashTemplate,\n        extract: hashTemplate,\n        file: hashTemplate,\n        script: hashTemplate,\n      };\n    case 'none':\n    default:\n      return {\n        chunk: '',\n        extract: '',\n        file: '',\n        script: '',\n      };\n  }\n}\n\nexport type NormalizedEntryPoint = Required<Exclude<ScriptElement | StyleElement, string>>;\n\nexport function normalizeExtraEntryPoints(\n  extraEntryPoints: (ScriptElement | StyleElement)[],\n  defaultBundleName: string,\n): NormalizedEntryPoint[] {\n  return extraEntryPoints.map((entry) => {\n    if (typeof entry === 'string') {\n      return { input: entry, inject: true, bundleName: defaultBundleName };\n    }\n\n    const { inject = true, ...newEntry } = entry;\n    let bundleName;\n    if (entry.bundleName) {\n      bundleName = entry.bundleName;\n    } else if (!inject) {\n      // Lazy entry points use the file name as bundle name.\n      bundleName = path.parse(entry.input).name;\n    } else {\n      bundleName = defaultBundleName;\n    }\n\n    return { ...newEntry, inject, bundleName };\n  });\n}\n\nexport function assetNameTemplateFactory(hashFormat: HashFormat): (resourcePath: string) => string {\n  const visitedFiles = new Map<string, string>();\n\n  return (resourcePath: string) => {\n    if (hashFormat.file) {\n      // File names are hashed therefore we don't need to handle files with the same file name.\n      return `[name]${hashFormat.file}.[ext]`;\n    }\n\n    const filename = path.basename(resourcePath);\n    // Check if the file with the same name has already been processed.\n    const visited = visitedFiles.get(filename);\n    if (!visited) {\n      // Not visited.\n      visitedFiles.set(filename, resourcePath);\n\n      return filename;\n    } else if (visited === resourcePath) {\n      // Same file.\n      return filename;\n    }\n\n    // File has the same name but it's in a different location.\n    return '[path][name].[ext]';\n  };\n}\n\nexport function getInstrumentationExcludedPaths(\n  root: string,\n  excludedPaths: string[],\n): Set<string> {\n  const excluded = new Set<string>();\n\n  for (const excludeGlob of excludedPaths) {\n    const excludePath = excludeGlob[0] === '/' ? excludeGlob.slice(1) : excludeGlob;\n    glob.sync(excludePath, { cwd: root }).forEach((p) => excluded.add(path.join(root, p)));\n  }\n\n  return excluded;\n}\n\nexport function normalizeGlobalStyles(styleEntrypoints: StyleElement[]): {\n  entryPoints: Record<string, string[]>;\n  noInjectNames: string[];\n} {\n  const entryPoints: Record<string, string[]> = {};\n  const noInjectNames: string[] = [];\n\n  if (styleEntrypoints.length === 0) {\n    return { entryPoints, noInjectNames };\n  }\n\n  for (const style of normalizeExtraEntryPoints(styleEntrypoints, 'styles')) {\n    // Add style entry points.\n    entryPoints[style.bundleName] ??= [];\n    entryPoints[style.bundleName].push(style.input);\n\n    // Add non injected styles to the list.\n    if (!style.inject) {\n      noInjectNames.push(style.bundleName);\n    }\n  }\n\n  return { entryPoints, noInjectNames };\n}\n\nexport function getCacheSettings(\n  wco: WebpackConfigOptions,\n  angularVersion: string,\n): WebpackOptionsNormalized['cache'] {\n  const { enabled, path: cacheDirectory } = wco.buildOptions.cache;\n  if (enabled) {\n    return {\n      type: 'filesystem',\n      profile: wco.buildOptions.verbose,\n      cacheDirectory: path.join(cacheDirectory, 'angular-webpack'),\n      maxMemoryGenerations: 1,\n      // We use the versions and build options as the cache name. The Webpack configurations are too\n      // dynamic and shared among different build types: test, build and serve.\n      // None of which are \"named\".\n      name: createHash('sha1')\n        .update(angularVersion)\n        .update(VERSION)\n        .update(wco.projectRoot)\n        .update(JSON.stringify(wco.tsConfig))\n        .update(\n          JSON.stringify({\n            ...wco.buildOptions,\n            // Needed because outputPath changes on every build when using i18n extraction\n            // https://github.com/angular/angular-cli/blob/736a5f89deaca85f487b78aec9ff66d4118ceb6a/packages/angular_devkit/build_angular/src/utils/i18n-options.ts#L264-L265\n            outputPath: undefined,\n          }),\n        )\n        .digest('hex'),\n    };\n  }\n\n  if (wco.buildOptions.watch) {\n    return {\n      type: 'memory',\n      maxGenerations: 1,\n    };\n  }\n\n  return false;\n}\n\nexport function globalScriptsByBundleName(\n  scripts: ScriptElement[],\n): { bundleName: string; inject: boolean; paths: string[] }[] {\n  return normalizeExtraEntryPoints(scripts, 'scripts').reduce(\n    (prev: { bundleName: string; paths: string[]; inject: boolean }[], curr) => {\n      const { bundleName, inject, input } = curr;\n\n      const existingEntry = prev.find((el) => el.bundleName === bundleName);\n      if (existingEntry) {\n        if (existingEntry.inject && !inject) {\n          // All entries have to be lazy for the bundle to be lazy.\n          throw new Error(`The ${bundleName} bundle is mixing injected and non-injected scripts.`);\n        }\n\n        existingEntry.paths.push(input);\n      } else {\n        prev.push({\n          bundleName,\n          inject,\n          paths: [input],\n        });\n      }\n\n      return prev;\n    },\n    [],\n  );\n}\n\nexport function assetPatterns(root: string, assets: AssetPatternClass[]) {\n  return assets.map((asset: AssetPatternClass, index: number): ObjectPattern => {\n    // Resolve input paths relative to workspace root and add slash at the end.\n    // eslint-disable-next-line prefer-const\n    let { input, output, ignore = [], glob } = asset;\n    input = path.resolve(root, input).replace(/\\\\/g, '/');\n    input = input.endsWith('/') ? input : input + '/';\n    output = output.endsWith('/') ? output : output + '/';\n\n    if (output.startsWith('..')) {\n      throw new Error('An asset cannot be written to a location outside of the output path.');\n    }\n\n    return {\n      context: input,\n      // Now we remove starting slash to make Webpack place it from the output root.\n      to: output.replace(/^\\//, ''),\n      from: glob,\n      noErrorOnMissing: true,\n      force: true,\n      globOptions: {\n        dot: true,\n        followSymbolicLinks: !!asset.followSymlinks,\n        ignore: [\n          '.gitkeep',\n          '**/.DS_Store',\n          '**/Thumbs.db',\n          // Negate patterns needs to be absolute because copy-webpack-plugin uses absolute globs which\n          // causes negate patterns not to match.\n          // See: https://github.com/webpack-contrib/copy-webpack-plugin/issues/498#issuecomment-639327909\n          ...ignore,\n        ].map((i) => path.posix.join(input, i)),\n      },\n      priority: index,\n    };\n  });\n}\n\nexport function getStatsOptions(verbose = false): WebpackStatsOptions {\n  const webpackOutputOptions: WebpackStatsOptions = {\n    all: false, // Fallback value for stats options when an option is not defined. It has precedence over local webpack defaults.\n    colors: true,\n    hash: true, // required by custom stat output\n    timings: true, // required by custom stat output\n    chunks: true, // required by custom stat output\n    builtAt: true, // required by custom stat output\n    warnings: true,\n    errors: true,\n    assets: true, // required by custom stat output\n    cachedAssets: true, // required for bundle size calculators\n\n    // Needed for markAsyncChunksNonInitial.\n    ids: true,\n    entrypoints: true,\n  };\n\n  const verboseWebpackOutputOptions: WebpackStatsOptions = {\n    // The verbose output will most likely be piped to a file, so colors just mess it up.\n    colors: false,\n    usedExports: true,\n    optimizationBailout: true,\n    reasons: true,\n    children: true,\n    assets: true,\n    version: true,\n    chunkModules: true,\n    errorDetails: true,\n    errorStack: true,\n    moduleTrace: true,\n    logging: 'verbose',\n    modulesSpace: Infinity,\n  };\n\n  return verbose\n    ? { ...webpackOutputOptions, ...verboseWebpackOutputOptions }\n    : webpackOutputOptions;\n}\n\n/**\n * @param root the workspace root\n * @returns `true` when `@angular/platform-server` is installed.\n */\nexport function isPlatformServerInstalled(root: string): boolean {\n  try {\n    require.resolve('@angular/platform-server', { paths: [root] });\n\n    return true;\n  } catch {\n    return false;\n  }\n}\n"]}