@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
152 lines • 21.4 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 __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.getIndexInputFile = exports.getIndexOutputFile = exports.generateBrowserWebpackConfigFromContext = exports.generateI18nBrowserWebpackConfigFromContext = exports.generateWebpackConfig = void 0;
const path = __importStar(require("path"));
const webpack_1 = require("webpack");
const webpack_merge_1 = require("webpack-merge");
const builder_watch_plugin_1 = require("../tools/webpack/plugins/builder-watch-plugin");
const utils_1 = require("../utils");
const read_tsconfig_1 = require("../utils/read-tsconfig");
const i18n_options_1 = require("./i18n-options");
async function generateWebpackConfig(workspaceRoot, projectRoot, sourceRoot, projectName, options, webpackPartialGenerator, logger, extraBuildOptions) {
// Ensure Build Optimizer is only used with AOT.
if (options.buildOptimizer && !options.aot) {
throw new Error(`The 'buildOptimizer' option cannot be used without 'aot'.`);
}
const tsConfigPath = path.resolve(workspaceRoot, options.tsConfig);
const tsConfig = await (0, read_tsconfig_1.readTsconfig)(tsConfigPath);
const buildOptions = { ...options, ...extraBuildOptions };
const wco = {
root: workspaceRoot,
logger: logger.createChild('webpackConfigOptions'),
projectRoot,
sourceRoot,
buildOptions,
tsConfig,
tsConfigPath,
projectName,
};
wco.buildOptions.progress = (0, utils_1.defaultProgress)(wco.buildOptions.progress);
const partials = await Promise.all(webpackPartialGenerator(wco));
const webpackConfig = (0, webpack_merge_1.merge)(partials);
return webpackConfig;
}
exports.generateWebpackConfig = generateWebpackConfig;
async function generateI18nBrowserWebpackConfigFromContext(options, context, webpackPartialGenerator, extraBuildOptions = {}) {
const { buildOptions, i18n } = await (0, i18n_options_1.configureI18nBuild)(context, options);
const result = await generateBrowserWebpackConfigFromContext(buildOptions, context, (wco) => {
return webpackPartialGenerator(wco);
}, extraBuildOptions);
const config = result.config;
if (i18n.shouldInline) {
// Remove localize "polyfill" if in AOT mode
if (buildOptions.aot) {
if (!config.resolve) {
config.resolve = {};
}
if (Array.isArray(config.resolve.alias)) {
config.resolve.alias.push({
name: '@angular/localize/init',
alias: false,
});
}
else {
if (!config.resolve.alias) {
config.resolve.alias = {};
}
config.resolve.alias['@angular/localize/init'] = false;
}
}
// Update file hashes to include translation file content
const i18nHash = Object.values(i18n.locales).reduce((data, locale) => data + locale.files.map((file) => file.integrity || '').join('|'), '');
config.plugins ?? (config.plugins = []);
config.plugins.push({
apply(compiler) {
compiler.hooks.compilation.tap('build-angular', (compilation) => {
webpack_1.javascript.JavascriptModulesPlugin.getCompilationHooks(compilation).chunkHash.tap('build-angular', (_, hash) => {
hash.update('$localize' + i18nHash);
});
});
},
});
}
return { ...result, i18n };
}
exports.generateI18nBrowserWebpackConfigFromContext = generateI18nBrowserWebpackConfigFromContext;
async function generateBrowserWebpackConfigFromContext(options, context, webpackPartialGenerator, extraBuildOptions = {}) {
const projectName = context.target && context.target.project;
if (!projectName) {
throw new Error('The builder requires a target.');
}
const workspaceRoot = context.workspaceRoot;
const projectMetadata = await context.getProjectMetadata(projectName);
const projectRoot = path.join(workspaceRoot, projectMetadata.root ?? '');
const sourceRoot = projectMetadata.sourceRoot;
const projectSourceRoot = sourceRoot ? path.join(workspaceRoot, sourceRoot) : undefined;
const normalizedOptions = (0, utils_1.normalizeBrowserSchema)(workspaceRoot, projectRoot, projectSourceRoot, options, projectMetadata, context.logger);
const config = await generateWebpackConfig(workspaceRoot, projectRoot, projectSourceRoot, projectName, normalizedOptions, webpackPartialGenerator, context.logger, extraBuildOptions);
// If builder watch support is present in the context, add watch plugin
// This is internal only and currently only used for testing
const watcherFactory = context.watcherFactory;
if (watcherFactory) {
if (!config.plugins) {
config.plugins = [];
}
config.plugins.push(new builder_watch_plugin_1.BuilderWatchPlugin(watcherFactory));
}
return {
config,
projectRoot,
projectSourceRoot,
};
}
exports.generateBrowserWebpackConfigFromContext = generateBrowserWebpackConfigFromContext;
function getIndexOutputFile(index) {
if (typeof index === 'string') {
return path.basename(index);
}
else {
return index.output || 'index.html';
}
}
exports.getIndexOutputFile = getIndexOutputFile;
function getIndexInputFile(index) {
if (typeof index === 'string') {
return index;
}
else {
return index.input;
}
}
exports.getIndexInputFile = getIndexInputFile;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webpack-browser-config.js","sourceRoot":"","sources":["../../../../../../../../packages/angular_devkit/build_angular/src/utils/webpack-browser-config.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 { BuilderContext } from '@angular-devkit/architect';\nimport { logging } from '@angular-devkit/core';\nimport * as path from 'path';\nimport { Configuration, javascript } from 'webpack';\nimport { merge as webpackMerge } from 'webpack-merge';\nimport { Schema as BrowserBuilderSchema } from '../builders/browser/schema';\nimport {\n  BuilderWatchPlugin,\n  BuilderWatcherFactory,\n} from '../tools/webpack/plugins/builder-watch-plugin';\nimport { NormalizedBrowserBuilderSchema, defaultProgress, normalizeBrowserSchema } from '../utils';\nimport { WebpackConfigOptions } from '../utils/build-options';\nimport { readTsconfig } from '../utils/read-tsconfig';\nimport { I18nOptions, configureI18nBuild } from './i18n-options';\n\nexport type BrowserWebpackConfigOptions = WebpackConfigOptions<NormalizedBrowserBuilderSchema>;\n\nexport type WebpackPartialGenerator = (\n  configurationOptions: BrowserWebpackConfigOptions,\n) => (Promise<Configuration> | Configuration)[];\n\nexport async function generateWebpackConfig(\n  workspaceRoot: string,\n  projectRoot: string,\n  sourceRoot: string | undefined,\n  projectName: string,\n  options: NormalizedBrowserBuilderSchema,\n  webpackPartialGenerator: WebpackPartialGenerator,\n  logger: logging.LoggerApi,\n  extraBuildOptions: Partial<NormalizedBrowserBuilderSchema>,\n): Promise<Configuration> {\n  // Ensure Build Optimizer is only used with AOT.\n  if (options.buildOptimizer && !options.aot) {\n    throw new Error(`The 'buildOptimizer' option cannot be used without 'aot'.`);\n  }\n\n  const tsConfigPath = path.resolve(workspaceRoot, options.tsConfig);\n  const tsConfig = await readTsconfig(tsConfigPath);\n\n  const buildOptions: NormalizedBrowserBuilderSchema = { ...options, ...extraBuildOptions };\n  const wco: BrowserWebpackConfigOptions = {\n    root: workspaceRoot,\n    logger: logger.createChild('webpackConfigOptions'),\n    projectRoot,\n    sourceRoot,\n    buildOptions,\n    tsConfig,\n    tsConfigPath,\n    projectName,\n  };\n\n  wco.buildOptions.progress = defaultProgress(wco.buildOptions.progress);\n\n  const partials = await Promise.all(webpackPartialGenerator(wco));\n  const webpackConfig = webpackMerge(partials);\n\n  return webpackConfig;\n}\n\nexport async function generateI18nBrowserWebpackConfigFromContext(\n  options: BrowserBuilderSchema,\n  context: BuilderContext,\n  webpackPartialGenerator: WebpackPartialGenerator,\n  extraBuildOptions: Partial<NormalizedBrowserBuilderSchema> = {},\n): Promise<{\n  config: Configuration;\n  projectRoot: string;\n  projectSourceRoot?: string;\n  i18n: I18nOptions;\n}> {\n  const { buildOptions, i18n } = await configureI18nBuild(context, options);\n  const result = await generateBrowserWebpackConfigFromContext(\n    buildOptions,\n    context,\n    (wco) => {\n      return webpackPartialGenerator(wco);\n    },\n    extraBuildOptions,\n  );\n  const config = result.config;\n\n  if (i18n.shouldInline) {\n    // Remove localize \"polyfill\" if in AOT mode\n    if (buildOptions.aot) {\n      if (!config.resolve) {\n        config.resolve = {};\n      }\n      if (Array.isArray(config.resolve.alias)) {\n        config.resolve.alias.push({\n          name: '@angular/localize/init',\n          alias: false,\n        });\n      } else {\n        if (!config.resolve.alias) {\n          config.resolve.alias = {};\n        }\n        config.resolve.alias['@angular/localize/init'] = false;\n      }\n    }\n\n    // Update file hashes to include translation file content\n    const i18nHash = Object.values(i18n.locales).reduce(\n      (data, locale) => data + locale.files.map((file) => file.integrity || '').join('|'),\n      '',\n    );\n\n    config.plugins ??= [];\n    config.plugins.push({\n      apply(compiler) {\n        compiler.hooks.compilation.tap('build-angular', (compilation) => {\n          javascript.JavascriptModulesPlugin.getCompilationHooks(compilation).chunkHash.tap(\n            'build-angular',\n            (_, hash) => {\n              hash.update('$localize' + i18nHash);\n            },\n          );\n        });\n      },\n    });\n  }\n\n  return { ...result, i18n };\n}\nexport async function generateBrowserWebpackConfigFromContext(\n  options: BrowserBuilderSchema,\n  context: BuilderContext,\n  webpackPartialGenerator: WebpackPartialGenerator,\n  extraBuildOptions: Partial<NormalizedBrowserBuilderSchema> = {},\n): Promise<{ config: Configuration; projectRoot: string; projectSourceRoot?: string }> {\n  const projectName = context.target && context.target.project;\n  if (!projectName) {\n    throw new Error('The builder requires a target.');\n  }\n\n  const workspaceRoot = context.workspaceRoot;\n  const projectMetadata = await context.getProjectMetadata(projectName);\n  const projectRoot = path.join(workspaceRoot, (projectMetadata.root as string | undefined) ?? '');\n  const sourceRoot = projectMetadata.sourceRoot as string | undefined;\n  const projectSourceRoot = sourceRoot ? path.join(workspaceRoot, sourceRoot) : undefined;\n\n  const normalizedOptions = normalizeBrowserSchema(\n    workspaceRoot,\n    projectRoot,\n    projectSourceRoot,\n    options,\n    projectMetadata,\n    context.logger,\n  );\n\n  const config = await generateWebpackConfig(\n    workspaceRoot,\n    projectRoot,\n    projectSourceRoot,\n    projectName,\n    normalizedOptions,\n    webpackPartialGenerator,\n    context.logger,\n    extraBuildOptions,\n  );\n\n  // If builder watch support is present in the context, add watch plugin\n  // This is internal only and currently only used for testing\n  const watcherFactory = (\n    context as {\n      watcherFactory?: BuilderWatcherFactory;\n    }\n  ).watcherFactory;\n  if (watcherFactory) {\n    if (!config.plugins) {\n      config.plugins = [];\n    }\n    config.plugins.push(new BuilderWatchPlugin(watcherFactory));\n  }\n\n  return {\n    config,\n    projectRoot,\n    projectSourceRoot,\n  };\n}\n\nexport function getIndexOutputFile(index: BrowserBuilderSchema['index']): string {\n  if (typeof index === 'string') {\n    return path.basename(index);\n  } else {\n    return index.output || 'index.html';\n  }\n}\n\nexport function getIndexInputFile(index: BrowserBuilderSchema['index']): string {\n  if (typeof index === 'string') {\n    return index;\n  } else {\n    return index.input;\n  }\n}\n"]}