UNPKG

@angular-devkit/build-angular

Version:
165 lines • 26.3 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 __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.JavaScriptOptimizerPlugin = void 0; const piscina_1 = __importDefault(require("piscina")); const environment_options_1 = require("../../../utils/environment-options"); const webpack_diagnostics_1 = require("../../../utils/webpack-diagnostics"); const utils_1 = require("../../esbuild/utils"); const esbuild_executor_1 = require("./esbuild-executor"); /** * The maximum number of Workers that will be created to execute optimize tasks. */ const MAX_OPTIMIZE_WORKERS = environment_options_1.maxWorkers; /** * The name of the plugin provided to Webpack when tapping Webpack compiler hooks. */ const PLUGIN_NAME = 'angular-javascript-optimizer'; /** * A Webpack plugin that provides JavaScript optimization capabilities. * * The plugin uses both `esbuild` and `terser` to provide both fast and highly-optimized * code output. `esbuild` is used as an initial pass to remove the majority of unused code * as well as shorten identifiers. `terser` is then used as a secondary pass to apply * optimizations not yet implemented by `esbuild`. */ class JavaScriptOptimizerPlugin { constructor(options) { this.options = options; if (options.supportedBrowsers) { this.targets = (0, utils_1.transformSupportedBrowsersToTargets)(options.supportedBrowsers); } } apply(compiler) { const { OriginalSource, SourceMapSource } = compiler.webpack.sources; compiler.hooks.compilation.tap(PLUGIN_NAME, (compilation) => { const logger = compilation.getLogger('build-angular.JavaScriptOptimizerPlugin'); compilation.hooks.processAssets.tapPromise({ name: PLUGIN_NAME, stage: compiler.webpack.Compilation.PROCESS_ASSETS_STAGE_OPTIMIZE_SIZE, }, async (compilationAssets) => { logger.time('optimize js assets'); const scriptsToOptimize = []; const cache = compilation.options.cache && compilation.getCache('JavaScriptOptimizerPlugin'); // Analyze the compilation assets for scripts that require optimization for (const assetName of Object.keys(compilationAssets)) { if (!assetName.endsWith('.js')) { continue; } const scriptAsset = compilation.getAsset(assetName); // Skip assets that have already been optimized or are verbatim copies (project assets) if (!scriptAsset || scriptAsset.info.minimized || scriptAsset.info.copied) { continue; } const { source: scriptAssetSource, name } = scriptAsset; let cacheItem; if (cache) { const eTag = cache.getLazyHashedEtag(scriptAssetSource); cacheItem = cache.getItemCache(name, eTag); const cachedOutput = await cacheItem.getPromise(); if (cachedOutput) { logger.debug(`${name} restored from cache`); compilation.updateAsset(name, cachedOutput.source, (assetInfo) => ({ ...assetInfo, minimized: true, })); continue; } } const { source, map } = scriptAssetSource.sourceAndMap(); scriptsToOptimize.push({ name: scriptAsset.name, code: typeof source === 'string' ? source : source.toString(), map, cacheItem, }); } if (scriptsToOptimize.length === 0) { return; } // Ensure all replacement values are strings which is the expected type for esbuild let define; if (this.options.define) { define = {}; for (const [key, value] of Object.entries(this.options.define)) { define[key] = String(value); } } // Setup the options used by all worker tasks const optimizeOptions = { sourcemap: this.options.sourcemap, define, keepIdentifierNames: this.options.keepIdentifierNames, target: this.targets, removeLicenses: this.options.removeLicenses, advanced: this.options.advanced, // Perform a single native esbuild support check. // This removes the need for each worker to perform the check which would // otherwise require spawning a separate process per worker. alwaysUseWasm: !(await esbuild_executor_1.EsbuildExecutor.hasNativeSupport()), }; // Sort scripts so larger scripts start first - worker pool uses a FIFO queue scriptsToOptimize.sort((a, b) => a.code.length - b.code.length); // Initialize the task worker pool const workerPath = require.resolve('./javascript-optimizer-worker'); const workerPool = new piscina_1.default({ filename: workerPath, maxThreads: MAX_OPTIMIZE_WORKERS, }); // Enqueue script optimization tasks and update compilation assets as the tasks complete try { const tasks = []; for (const { name, code, map, cacheItem } of scriptsToOptimize) { logger.time(`optimize asset: ${name}`); tasks.push(workerPool .run({ asset: { name, code, map, }, options: optimizeOptions, }) .then(async ({ code, name, map, errors }) => { if (errors?.length) { for (const error of errors) { (0, webpack_diagnostics_1.addError)(compilation, `Optimization error [${name}]: ${error}`); } return; } const optimizedAsset = map ? new SourceMapSource(code, name, map) : new OriginalSource(code, name); compilation.updateAsset(name, optimizedAsset, (assetInfo) => ({ ...assetInfo, minimized: true, })); logger.timeEnd(`optimize asset: ${name}`); return cacheItem?.storePromise({ source: optimizedAsset, }); }, (error) => { (0, webpack_diagnostics_1.addError)(compilation, `Optimization error [${name}]: ${error.stack || error.message}`); })); } await Promise.all(tasks); } finally { void workerPool.destroy(); } logger.timeEnd('optimize js assets'); }); }); } } exports.JavaScriptOptimizerPlugin = JavaScriptOptimizerPlugin; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"javascript-optimizer-plugin.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/webpack/plugins/javascript-optimizer-plugin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport Piscina from 'piscina';\nimport type { Compiler, sources } from 'webpack';\nimport { maxWorkers } from '../../../utils/environment-options';\nimport { addError } from '../../../utils/webpack-diagnostics';\nimport { transformSupportedBrowsersToTargets } from '../../esbuild/utils';\nimport { EsbuildExecutor } from './esbuild-executor';\nimport type { OptimizeRequestOptions } from './javascript-optimizer-worker';\n\n/**\n * The maximum number of Workers that will be created to execute optimize tasks.\n */\nconst MAX_OPTIMIZE_WORKERS = maxWorkers;\n\n/**\n * The name of the plugin provided to Webpack when tapping Webpack compiler hooks.\n */\nconst PLUGIN_NAME = 'angular-javascript-optimizer';\n\n/**\n * The options used to configure the {@link JavaScriptOptimizerPlugin}.\n */\nexport interface JavaScriptOptimizerOptions {\n  /**\n   * Enables advanced optimizations in the underlying JavaScript optimizers.\n   * This currently increases the `terser` passes to 2 and enables the `pure_getters`\n   * option for `terser`.\n   */\n  advanced?: boolean;\n\n  /**\n   * An object record of string keys that will be replaced with their respective values when found\n   * within the code during optimization.\n   */\n  define: Record<string, string | number | boolean>;\n\n  /**\n   * Enables the generation of a sourcemap during optimization.\n   * The output sourcemap will be a full sourcemap containing the merge of the input sourcemap and\n   * all intermediate sourcemaps.\n   */\n  sourcemap?: boolean;\n\n  /**\n   * A list of supported browsers that is used for output code.\n   */\n  supportedBrowsers?: string[];\n\n  /**\n   * Enables the retention of identifier names and ensures that function and class names are\n   * present in the output code.\n   *\n   * **Note**: in some cases symbols are still renamed to avoid collisions.\n   */\n  keepIdentifierNames: boolean;\n\n  /**\n   * Enables the removal of all license comments from the output code.\n   */\n  removeLicenses?: boolean;\n}\n\n/**\n * A Webpack plugin that provides JavaScript optimization capabilities.\n *\n * The plugin uses both `esbuild` and `terser` to provide both fast and highly-optimized\n * code output. `esbuild` is used as an initial pass to remove the majority of unused code\n * as well as shorten identifiers. `terser` is then used as a secondary pass to apply\n * optimizations not yet implemented by `esbuild`.\n */\nexport class JavaScriptOptimizerPlugin {\n  private targets: string[] | undefined;\n\n  constructor(private options: JavaScriptOptimizerOptions) {\n    if (options.supportedBrowsers) {\n      this.targets = transformSupportedBrowsersToTargets(options.supportedBrowsers);\n    }\n  }\n\n  apply(compiler: Compiler) {\n    const { OriginalSource, SourceMapSource } = compiler.webpack.sources;\n\n    compiler.hooks.compilation.tap(PLUGIN_NAME, (compilation) => {\n      const logger = compilation.getLogger('build-angular.JavaScriptOptimizerPlugin');\n\n      compilation.hooks.processAssets.tapPromise(\n        {\n          name: PLUGIN_NAME,\n          stage: compiler.webpack.Compilation.PROCESS_ASSETS_STAGE_OPTIMIZE_SIZE,\n        },\n        async (compilationAssets) => {\n          logger.time('optimize js assets');\n          const scriptsToOptimize = [];\n          const cache =\n            compilation.options.cache && compilation.getCache('JavaScriptOptimizerPlugin');\n\n          // Analyze the compilation assets for scripts that require optimization\n          for (const assetName of Object.keys(compilationAssets)) {\n            if (!assetName.endsWith('.js')) {\n              continue;\n            }\n\n            const scriptAsset = compilation.getAsset(assetName);\n            // Skip assets that have already been optimized or are verbatim copies (project assets)\n            if (!scriptAsset || scriptAsset.info.minimized || scriptAsset.info.copied) {\n              continue;\n            }\n\n            const { source: scriptAssetSource, name } = scriptAsset;\n            let cacheItem;\n\n            if (cache) {\n              const eTag = cache.getLazyHashedEtag(scriptAssetSource);\n              cacheItem = cache.getItemCache(name, eTag);\n              const cachedOutput = await cacheItem.getPromise<\n                { source: sources.Source } | undefined\n              >();\n\n              if (cachedOutput) {\n                logger.debug(`${name} restored from cache`);\n                compilation.updateAsset(name, cachedOutput.source, (assetInfo) => ({\n                  ...assetInfo,\n                  minimized: true,\n                }));\n                continue;\n              }\n            }\n\n            const { source, map } = scriptAssetSource.sourceAndMap();\n            scriptsToOptimize.push({\n              name: scriptAsset.name,\n              code: typeof source === 'string' ? source : source.toString(),\n              map,\n              cacheItem,\n            });\n          }\n\n          if (scriptsToOptimize.length === 0) {\n            return;\n          }\n\n          // Ensure all replacement values are strings which is the expected type for esbuild\n          let define: Record<string, string> | undefined;\n          if (this.options.define) {\n            define = {};\n            for (const [key, value] of Object.entries(this.options.define)) {\n              define[key] = String(value);\n            }\n          }\n\n          // Setup the options used by all worker tasks\n          const optimizeOptions: OptimizeRequestOptions = {\n            sourcemap: this.options.sourcemap,\n            define,\n            keepIdentifierNames: this.options.keepIdentifierNames,\n            target: this.targets,\n            removeLicenses: this.options.removeLicenses,\n            advanced: this.options.advanced,\n            // Perform a single native esbuild support check.\n            // This removes the need for each worker to perform the check which would\n            // otherwise require spawning a separate process per worker.\n            alwaysUseWasm: !(await EsbuildExecutor.hasNativeSupport()),\n          };\n\n          // Sort scripts so larger scripts start first - worker pool uses a FIFO queue\n          scriptsToOptimize.sort((a, b) => a.code.length - b.code.length);\n\n          // Initialize the task worker pool\n          const workerPath = require.resolve('./javascript-optimizer-worker');\n          const workerPool = new Piscina({\n            filename: workerPath,\n            maxThreads: MAX_OPTIMIZE_WORKERS,\n          });\n\n          // Enqueue script optimization tasks and update compilation assets as the tasks complete\n          try {\n            const tasks = [];\n            for (const { name, code, map, cacheItem } of scriptsToOptimize) {\n              logger.time(`optimize asset: ${name}`);\n\n              tasks.push(\n                workerPool\n                  .run({\n                    asset: {\n                      name,\n                      code,\n                      map,\n                    },\n                    options: optimizeOptions,\n                  })\n                  .then(\n                    async ({ code, name, map, errors }) => {\n                      if (errors?.length) {\n                        for (const error of errors) {\n                          addError(compilation, `Optimization error [${name}]: ${error}`);\n                        }\n\n                        return;\n                      }\n\n                      const optimizedAsset = map\n                        ? new SourceMapSource(code, name, map)\n                        : new OriginalSource(code, name);\n                      compilation.updateAsset(name, optimizedAsset, (assetInfo) => ({\n                        ...assetInfo,\n                        minimized: true,\n                      }));\n\n                      logger.timeEnd(`optimize asset: ${name}`);\n\n                      return cacheItem?.storePromise({\n                        source: optimizedAsset,\n                      });\n                    },\n                    (error) => {\n                      addError(\n                        compilation,\n                        `Optimization error [${name}]: ${error.stack || error.message}`,\n                      );\n                    },\n                  ),\n              );\n            }\n\n            await Promise.all(tasks);\n          } finally {\n            void workerPool.destroy();\n          }\n\n          logger.timeEnd('optimize js assets');\n        },\n      );\n    });\n  }\n}\n"]}