UNPKG

@angular-devkit/build-angular

Version:
180 lines • 24.1 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; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.ScriptsWebpackPlugin = void 0; const loader_utils_1 = require("loader-utils"); const path = __importStar(require("path")); const webpack_1 = require("webpack"); const error_1 = require("../../../utils/error"); const webpack_diagnostics_1 = require("../../../utils/webpack-diagnostics"); const Entrypoint = require('webpack/lib/Entrypoint'); /** * The name of the plugin provided to Webpack when tapping Webpack compiler hooks. */ const PLUGIN_NAME = 'scripts-webpack-plugin'; function addDependencies(compilation, scripts) { for (const script of scripts) { compilation.fileDependencies.add(script); } } class ScriptsWebpackPlugin { constructor(options) { this.options = options; } async shouldSkip(compilation, scripts) { if (this._lastBuildTime == undefined) { this._lastBuildTime = Date.now(); return false; } for (const script of scripts) { const scriptTime = await new Promise((resolve, reject) => { compilation.fileSystemInfo.getFileTimestamp(script, (error, entry) => { if (error) { reject(error); return; } resolve(entry && typeof entry !== 'string' ? entry.safeTime : undefined); }); }); if (!scriptTime || scriptTime > this._lastBuildTime) { this._lastBuildTime = Date.now(); return false; } } return true; } _insertOutput(compilation, { filename, source }, cached = false) { const chunk = new webpack_1.Chunk(this.options.name); chunk.rendered = !cached; chunk.id = this.options.name; chunk.ids = [chunk.id]; chunk.files.add(filename); const entrypoint = new Entrypoint(this.options.name); entrypoint.pushChunk(chunk); chunk.addGroup(entrypoint); compilation.entrypoints.set(this.options.name, entrypoint); compilation.chunks.add(chunk); compilation.assets[filename] = source; compilation.hooks.chunkAsset.call(chunk, filename); } apply(compiler) { if (this.options.scripts.length === 0) { return; } compiler.hooks.thisCompilation.tap(PLUGIN_NAME, (compilation) => { // Use the resolver from the compilation instead of compiler. // Using the latter will causes a lot of `DescriptionFileUtils.loadDescriptionFile` calls. // See: https://github.com/angular/angular-cli/issues/24634#issuecomment-1425782668 const resolver = compilation.resolverFactory.get('normal', { preferRelative: true, useSyncFileSystemCalls: true, // Caching must be disabled because it causes the resolver to become async after a rebuild. cache: false, }); const scripts = []; for (const script of this.options.scripts) { try { const resolvedPath = resolver.resolveSync({}, this.options.basePath, script); if (resolvedPath) { scripts.push(resolvedPath); } else { (0, webpack_diagnostics_1.addError)(compilation, `Cannot resolve '${script}'.`); } } catch (error) { (0, error_1.assertIsError)(error); (0, webpack_diagnostics_1.addError)(compilation, error.message); } } compilation.hooks.additionalAssets.tapPromise(PLUGIN_NAME, async () => { if (await this.shouldSkip(compilation, scripts)) { if (this._cachedOutput) { this._insertOutput(compilation, this._cachedOutput, true); } addDependencies(compilation, scripts); return; } const sourceGetters = scripts.map((fullPath) => { return new Promise((resolve, reject) => { compilation.inputFileSystem.readFile(fullPath, (err, data) => { if (err) { reject(err); return; } const content = data?.toString() ?? ''; let source; if (this.options.sourceMap) { // TODO: Look for source map file (for '.min' scripts, etc.) let adjustedPath = fullPath; if (this.options.basePath) { adjustedPath = path.relative(this.options.basePath, fullPath); } source = new webpack_1.sources.OriginalSource(content, adjustedPath); } else { source = new webpack_1.sources.RawSource(content); } resolve(source); }); }); }); const sources = await Promise.all(sourceGetters); const concatSource = new webpack_1.sources.ConcatSource(); sources.forEach((source) => { concatSource.add(source); concatSource.add('\n;'); }); const combinedSource = new webpack_1.sources.CachedSource(concatSource); const output = { filename: this.options.filename, source: combinedSource }; this._insertOutput(compilation, output); this._cachedOutput = output; addDependencies(compilation, scripts); }); compilation.hooks.processAssets.tapPromise({ name: PLUGIN_NAME, stage: compiler.webpack.Compilation.PROCESS_ASSETS_STAGE_DEV_TOOLING, }, async () => { const assetName = this.options.filename; const asset = compilation.getAsset(assetName); if (asset) { const interpolatedFilename = (0, loader_utils_1.interpolateName)({ resourcePath: 'scripts.js' }, assetName, { content: asset.source.source() }); if (assetName !== interpolatedFilename) { compilation.renameAsset(assetName, interpolatedFilename); } } }); }); } } exports.ScriptsWebpackPlugin = ScriptsWebpackPlugin; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scripts-webpack-plugin.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/webpack/plugins/scripts-webpack-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 { interpolateName } from 'loader-utils';\nimport * as path from 'path';\nimport { Chunk, Compilation, Compiler, sources as webpackSources } from 'webpack';\nimport { assertIsError } from '../../../utils/error';\nimport { addError } from '../../../utils/webpack-diagnostics';\n\nconst Entrypoint = require('webpack/lib/Entrypoint');\n\n/**\n * The name of the plugin provided to Webpack when tapping Webpack compiler hooks.\n */\nconst PLUGIN_NAME = 'scripts-webpack-plugin';\n\nexport interface ScriptsWebpackPluginOptions {\n  name: string;\n  sourceMap?: boolean;\n  scripts: string[];\n  filename: string;\n  basePath: string;\n}\n\ninterface ScriptOutput {\n  filename: string;\n  source: webpackSources.CachedSource;\n}\n\nfunction addDependencies(compilation: Compilation, scripts: string[]): void {\n  for (const script of scripts) {\n    compilation.fileDependencies.add(script);\n  }\n}\n\nexport class ScriptsWebpackPlugin {\n  private _lastBuildTime?: number;\n  private _cachedOutput?: ScriptOutput;\n\n  constructor(private options: ScriptsWebpackPluginOptions) {}\n\n  async shouldSkip(compilation: Compilation, scripts: string[]): Promise<boolean> {\n    if (this._lastBuildTime == undefined) {\n      this._lastBuildTime = Date.now();\n\n      return false;\n    }\n\n    for (const script of scripts) {\n      const scriptTime = await new Promise<number | undefined>((resolve, reject) => {\n        compilation.fileSystemInfo.getFileTimestamp(script, (error, entry) => {\n          if (error) {\n            reject(error);\n\n            return;\n          }\n\n          resolve(entry && typeof entry !== 'string' ? entry.safeTime : undefined);\n        });\n      });\n\n      if (!scriptTime || scriptTime > this._lastBuildTime) {\n        this._lastBuildTime = Date.now();\n\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private _insertOutput(\n    compilation: Compilation,\n    { filename, source }: ScriptOutput,\n    cached = false,\n  ) {\n    const chunk = new Chunk(this.options.name);\n    chunk.rendered = !cached;\n    chunk.id = this.options.name;\n    chunk.ids = [chunk.id];\n    chunk.files.add(filename);\n\n    const entrypoint = new Entrypoint(this.options.name);\n    entrypoint.pushChunk(chunk);\n    chunk.addGroup(entrypoint);\n    compilation.entrypoints.set(this.options.name, entrypoint);\n    compilation.chunks.add(chunk);\n\n    compilation.assets[filename] = source;\n    compilation.hooks.chunkAsset.call(chunk, filename);\n  }\n\n  apply(compiler: Compiler): void {\n    if (this.options.scripts.length === 0) {\n      return;\n    }\n\n    compiler.hooks.thisCompilation.tap(PLUGIN_NAME, (compilation) => {\n      // Use the resolver from the compilation instead of compiler.\n      // Using the latter will causes a lot of `DescriptionFileUtils.loadDescriptionFile` calls.\n      // See: https://github.com/angular/angular-cli/issues/24634#issuecomment-1425782668\n      const resolver = compilation.resolverFactory.get('normal', {\n        preferRelative: true,\n        useSyncFileSystemCalls: true,\n        // Caching must be disabled because it causes the resolver to become async after a rebuild.\n        cache: false,\n      });\n\n      const scripts: string[] = [];\n\n      for (const script of this.options.scripts) {\n        try {\n          const resolvedPath = resolver.resolveSync({}, this.options.basePath, script);\n          if (resolvedPath) {\n            scripts.push(resolvedPath);\n          } else {\n            addError(compilation, `Cannot resolve '${script}'.`);\n          }\n        } catch (error) {\n          assertIsError(error);\n          addError(compilation, error.message);\n        }\n      }\n\n      compilation.hooks.additionalAssets.tapPromise(PLUGIN_NAME, async () => {\n        if (await this.shouldSkip(compilation, scripts)) {\n          if (this._cachedOutput) {\n            this._insertOutput(compilation, this._cachedOutput, true);\n          }\n\n          addDependencies(compilation, scripts);\n\n          return;\n        }\n\n        const sourceGetters = scripts.map((fullPath) => {\n          return new Promise<webpackSources.Source>((resolve, reject) => {\n            compilation.inputFileSystem.readFile(\n              fullPath,\n              (err?: Error | null, data?: string | Buffer) => {\n                if (err) {\n                  reject(err);\n\n                  return;\n                }\n\n                const content = data?.toString() ?? '';\n\n                let source;\n                if (this.options.sourceMap) {\n                  // TODO: Look for source map file (for '.min' scripts, etc.)\n\n                  let adjustedPath = fullPath;\n                  if (this.options.basePath) {\n                    adjustedPath = path.relative(this.options.basePath, fullPath);\n                  }\n                  source = new webpackSources.OriginalSource(content, adjustedPath);\n                } else {\n                  source = new webpackSources.RawSource(content);\n                }\n\n                resolve(source);\n              },\n            );\n          });\n        });\n\n        const sources = await Promise.all(sourceGetters);\n        const concatSource = new webpackSources.ConcatSource();\n        sources.forEach((source) => {\n          concatSource.add(source);\n          concatSource.add('\\n;');\n        });\n\n        const combinedSource = new webpackSources.CachedSource(concatSource);\n\n        const output = { filename: this.options.filename, source: combinedSource };\n        this._insertOutput(compilation, output);\n        this._cachedOutput = output;\n        addDependencies(compilation, scripts);\n      });\n      compilation.hooks.processAssets.tapPromise(\n        {\n          name: PLUGIN_NAME,\n          stage: compiler.webpack.Compilation.PROCESS_ASSETS_STAGE_DEV_TOOLING,\n        },\n        async () => {\n          const assetName = this.options.filename;\n          const asset = compilation.getAsset(assetName);\n          if (asset) {\n            const interpolatedFilename = interpolateName(\n              { resourcePath: 'scripts.js' },\n              assetName,\n              { content: asset.source.source() },\n            );\n            if (assetName !== interpolatedFilename) {\n              compilation.renameAsset(assetName, interpolatedFilename);\n            }\n          }\n        },\n      );\n    });\n  }\n}\n"]}