@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
217 lines (216 loc) • 31.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.execute = void 0;
const architect_1 = require("@angular-devkit/architect");
const core_1 = require("@angular-devkit/core");
const module_1 = require("module");
const path = __importStar(require("path"));
const rxjs_1 = require("rxjs");
const configs_1 = require("../../tools/webpack/configs");
const purge_cache_1 = require("../../utils/purge-cache");
const version_1 = require("../../utils/version");
const webpack_browser_config_1 = require("../../utils/webpack-browser-config");
const schema_1 = require("../browser/schema");
const find_tests_plugin_1 = require("./find-tests-plugin");
async function initialize(options, context, webpackConfigurationTransformer) {
// Purge old build disk cache.
await (0, purge_cache_1.purgeStaleBuildCache)(context);
const { config } = await (0, webpack_browser_config_1.generateBrowserWebpackConfigFromContext)(
// only two properties are missing:
// * `outputPath` which is fixed for tests
// * `budgets` which might be incorrect due to extra dev libs
{
...options,
outputPath: '',
budgets: undefined,
optimization: false,
buildOptimizer: false,
aot: false,
vendorChunk: true,
namedChunks: true,
extractLicenses: false,
outputHashing: schema_1.OutputHashing.None,
// The webpack tier owns the watch behavior so we want to force it in the config.
// When not in watch mode, webpack-dev-middleware will call `compiler.watch` anyway.
// https://github.com/webpack/webpack-dev-middleware/blob/698c9ae5e9bb9a013985add6189ff21c1a1ec185/src/index.js#L65
// https://github.com/webpack/webpack/blob/cde1b73e12eb8a77eb9ba42e7920c9ec5d29c2c9/lib/Compiler.js#L379-L388
watch: true,
}, context, (wco) => [(0, configs_1.getCommonConfig)(wco), (0, configs_1.getStylesConfig)(wco)]);
const karma = await Promise.resolve().then(() => __importStar(require('karma')));
return [karma, (await webpackConfigurationTransformer?.(config)) ?? config];
}
/**
* @experimental Direct usage of this function is considered experimental.
*/
function execute(options, context, transforms = {}) {
// Check Angular version.
(0, version_1.assertCompatibleAngularVersion)(context.workspaceRoot);
let singleRun;
if (options.watch !== undefined) {
singleRun = !options.watch;
}
return (0, rxjs_1.from)(initialize(options, context, transforms.webpackConfiguration)).pipe((0, rxjs_1.switchMap)(async ([karma, webpackConfig]) => {
// Determine project name from builder context target
const projectName = context.target?.project;
if (!projectName) {
throw new Error(`The 'karma' builder requires a target to be specified.`);
}
const karmaOptions = options.karmaConfig
? {}
: getBuiltInKarmaConfig(context.workspaceRoot, projectName);
karmaOptions.singleRun = singleRun;
// Convert browsers from a string to an array
if (options.browsers) {
karmaOptions.browsers = options.browsers.split(',');
}
if (options.reporters) {
// Split along commas to make it more natural, and remove empty strings.
const reporters = options.reporters
.reduce((acc, curr) => acc.concat(curr.split(',')), [])
.filter((x) => !!x);
if (reporters.length > 0) {
karmaOptions.reporters = reporters;
}
}
if (!options.main) {
webpackConfig.entry ?? (webpackConfig.entry = {});
if (typeof webpackConfig.entry === 'object' && !Array.isArray(webpackConfig.entry)) {
if (Array.isArray(webpackConfig.entry['main'])) {
webpackConfig.entry['main'].push(getBuiltInMainFile());
}
else {
webpackConfig.entry['main'] = [getBuiltInMainFile()];
}
}
}
const projectMetadata = await context.getProjectMetadata(projectName);
const sourceRoot = (projectMetadata.sourceRoot ?? projectMetadata.root ?? '');
webpackConfig.plugins ?? (webpackConfig.plugins = []);
webpackConfig.plugins.push(new find_tests_plugin_1.FindTestsPlugin({
include: options.include,
exclude: options.exclude,
workspaceRoot: context.workspaceRoot,
projectSourceRoot: path.join(context.workspaceRoot, sourceRoot),
}));
karmaOptions.buildWebpack = {
options,
webpackConfig,
logger: context.logger,
};
const parsedKarmaConfig = await karma.config.parseConfig(options.karmaConfig && path.resolve(context.workspaceRoot, options.karmaConfig), transforms.karmaOptions ? transforms.karmaOptions(karmaOptions) : karmaOptions, { promiseConfig: true, throwErrors: true });
return [karma, parsedKarmaConfig];
}), (0, rxjs_1.switchMap)(([karma, karmaConfig]) => new rxjs_1.Observable((subscriber) => {
var _a, _b;
// Pass onto Karma to emit BuildEvents.
karmaConfig.buildWebpack ?? (karmaConfig.buildWebpack = {});
if (typeof karmaConfig.buildWebpack === 'object') {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
(_a = karmaConfig.buildWebpack).failureCb ?? (_a.failureCb = () => subscriber.next({ success: false }));
// eslint-disable-next-line @typescript-eslint/no-explicit-any
(_b = karmaConfig.buildWebpack).successCb ?? (_b.successCb = () => subscriber.next({ success: true }));
}
// Complete the observable once the Karma server returns.
const karmaServer = new karma.Server(karmaConfig, (exitCode) => {
subscriber.next({ success: exitCode === 0 });
subscriber.complete();
});
const karmaStart = karmaServer.start();
// Cleanup, signal Karma to exit.
return () => {
void karmaStart.then(() => karmaServer.stop());
};
})), (0, rxjs_1.defaultIfEmpty)({ success: false }));
}
exports.execute = execute;
function getBuiltInKarmaConfig(workspaceRoot, projectName) {
let coverageFolderName = projectName.charAt(0) === '@' ? projectName.slice(1) : projectName;
if (/[A-Z]/.test(coverageFolderName)) {
coverageFolderName = core_1.strings.dasherize(coverageFolderName);
}
const workspaceRootRequire = (0, module_1.createRequire)(workspaceRoot + '/');
// Any changes to the config here need to be synced to: packages/schematics/angular/config/files/karma.conf.js.template
return {
basePath: '',
frameworks: ['jasmine', '@angular-devkit/build-angular'],
plugins: [
'karma-jasmine',
'karma-chrome-launcher',
'karma-jasmine-html-reporter',
'karma-coverage',
'@angular-devkit/build-angular/plugins/karma',
].map((p) => workspaceRootRequire(p)),
client: {
clearContext: false, // leave Jasmine Spec Runner output visible in browser
},
jasmineHtmlReporter: {
suppressAll: true, // removes the duplicated traces
},
coverageReporter: {
dir: path.join(workspaceRoot, 'coverage', coverageFolderName),
subdir: '.',
reporters: [{ type: 'html' }, { type: 'text-summary' }],
},
reporters: ['progress', 'kjhtml'],
browsers: ['Chrome'],
customLaunchers: {
// Chrome configured to run in a bazel sandbox.
// Disable the use of the gpu and `/dev/shm` because it causes Chrome to
// crash on some environments.
// See:
// https://github.com/puppeteer/puppeteer/blob/v1.0.0/docs/troubleshooting.md#tips
// https://stackoverflow.com/questions/50642308/webdriverexception-unknown-error-devtoolsactiveport-file-doesnt-exist-while-t
ChromeHeadlessNoSandbox: {
base: 'ChromeHeadless',
flags: ['--no-sandbox', '--headless', '--disable-gpu', '--disable-dev-shm-usage'],
},
},
restartOnFileChange: true,
};
}
exports.default = (0, architect_1.createBuilder)(execute);
function getBuiltInMainFile() {
const content = Buffer.from(`
import { getTestBed } from '@angular/core/testing';
import {
BrowserDynamicTestingModule,
platformBrowserDynamicTesting,
} from '@angular/platform-browser-dynamic/testing';
// Initialize the Angular testing environment.
getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting(), {
errorOnUnknownElements: true,
errorOnUnknownProperties: true
});
`).toString('base64');
return `ng-virtual-main.js!=!data:text/javascript;base64,${content}`;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/karma/index.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, BuilderOutput, createBuilder } from '@angular-devkit/architect';\nimport { strings } from '@angular-devkit/core';\nimport type { Config, ConfigOptions } from 'karma';\nimport { createRequire } from 'module';\nimport * as path from 'path';\nimport { Observable, defaultIfEmpty, from, switchMap } from 'rxjs';\nimport { Configuration } from 'webpack';\nimport { getCommonConfig, getStylesConfig } from '../../tools/webpack/configs';\nimport { ExecutionTransformer } from '../../transforms';\nimport { purgeStaleBuildCache } from '../../utils/purge-cache';\nimport { assertCompatibleAngularVersion } from '../../utils/version';\nimport { generateBrowserWebpackConfigFromContext } from '../../utils/webpack-browser-config';\nimport { Schema as BrowserBuilderOptions, OutputHashing } from '../browser/schema';\nimport { FindTestsPlugin } from './find-tests-plugin';\nimport { Schema as KarmaBuilderOptions } from './schema';\n\nexport type KarmaConfigOptions = ConfigOptions & {\n  buildWebpack?: unknown;\n  configFile?: string;\n};\n\nasync function initialize(\n  options: KarmaBuilderOptions,\n  context: BuilderContext,\n  webpackConfigurationTransformer?: ExecutionTransformer<Configuration>,\n): Promise<[typeof import('karma'), Configuration]> {\n  // Purge old build disk cache.\n  await purgeStaleBuildCache(context);\n\n  const { config } = await generateBrowserWebpackConfigFromContext(\n    // only two properties are missing:\n    // * `outputPath` which is fixed for tests\n    // * `budgets` which might be incorrect due to extra dev libs\n    {\n      ...(options as unknown as BrowserBuilderOptions),\n      outputPath: '',\n      budgets: undefined,\n      optimization: false,\n      buildOptimizer: false,\n      aot: false,\n      vendorChunk: true,\n      namedChunks: true,\n      extractLicenses: false,\n      outputHashing: OutputHashing.None,\n      // The webpack tier owns the watch behavior so we want to force it in the config.\n      // When not in watch mode, webpack-dev-middleware will call `compiler.watch` anyway.\n      // https://github.com/webpack/webpack-dev-middleware/blob/698c9ae5e9bb9a013985add6189ff21c1a1ec185/src/index.js#L65\n      // https://github.com/webpack/webpack/blob/cde1b73e12eb8a77eb9ba42e7920c9ec5d29c2c9/lib/Compiler.js#L379-L388\n      watch: true,\n    },\n    context,\n    (wco) => [getCommonConfig(wco), getStylesConfig(wco)],\n  );\n\n  const karma = await import('karma');\n\n  return [karma, (await webpackConfigurationTransformer?.(config)) ?? config];\n}\n\n/**\n * @experimental Direct usage of this function is considered experimental.\n */\nexport function execute(\n  options: KarmaBuilderOptions,\n  context: BuilderContext,\n  transforms: {\n    webpackConfiguration?: ExecutionTransformer<Configuration>;\n    // The karma options transform cannot be async without a refactor of the builder implementation\n    karmaOptions?: (options: KarmaConfigOptions) => KarmaConfigOptions;\n  } = {},\n): Observable<BuilderOutput> {\n  // Check Angular version.\n  assertCompatibleAngularVersion(context.workspaceRoot);\n\n  let singleRun: boolean | undefined;\n  if (options.watch !== undefined) {\n    singleRun = !options.watch;\n  }\n\n  return from(initialize(options, context, transforms.webpackConfiguration)).pipe(\n    switchMap(async ([karma, webpackConfig]) => {\n      // Determine project name from builder context target\n      const projectName = context.target?.project;\n      if (!projectName) {\n        throw new Error(`The 'karma' builder requires a target to be specified.`);\n      }\n\n      const karmaOptions: KarmaConfigOptions = options.karmaConfig\n        ? {}\n        : getBuiltInKarmaConfig(context.workspaceRoot, projectName);\n\n      karmaOptions.singleRun = singleRun;\n\n      // Convert browsers from a string to an array\n      if (options.browsers) {\n        karmaOptions.browsers = options.browsers.split(',');\n      }\n\n      if (options.reporters) {\n        // Split along commas to make it more natural, and remove empty strings.\n        const reporters = options.reporters\n          .reduce<string[]>((acc, curr) => acc.concat(curr.split(',')), [])\n          .filter((x) => !!x);\n\n        if (reporters.length > 0) {\n          karmaOptions.reporters = reporters;\n        }\n      }\n\n      if (!options.main) {\n        webpackConfig.entry ??= {};\n        if (typeof webpackConfig.entry === 'object' && !Array.isArray(webpackConfig.entry)) {\n          if (Array.isArray(webpackConfig.entry['main'])) {\n            webpackConfig.entry['main'].push(getBuiltInMainFile());\n          } else {\n            webpackConfig.entry['main'] = [getBuiltInMainFile()];\n          }\n        }\n      }\n\n      const projectMetadata = await context.getProjectMetadata(projectName);\n      const sourceRoot = (projectMetadata.sourceRoot ?? projectMetadata.root ?? '') as string;\n\n      webpackConfig.plugins ??= [];\n      webpackConfig.plugins.push(\n        new FindTestsPlugin({\n          include: options.include,\n          exclude: options.exclude,\n          workspaceRoot: context.workspaceRoot,\n          projectSourceRoot: path.join(context.workspaceRoot, sourceRoot),\n        }),\n      );\n\n      karmaOptions.buildWebpack = {\n        options,\n        webpackConfig,\n        logger: context.logger,\n      };\n\n      const parsedKarmaConfig = await karma.config.parseConfig(\n        options.karmaConfig && path.resolve(context.workspaceRoot, options.karmaConfig),\n        transforms.karmaOptions ? transforms.karmaOptions(karmaOptions) : karmaOptions,\n        { promiseConfig: true, throwErrors: true },\n      );\n\n      return [karma, parsedKarmaConfig] as [typeof karma, KarmaConfigOptions];\n    }),\n    switchMap(\n      ([karma, karmaConfig]) =>\n        new Observable<BuilderOutput>((subscriber) => {\n          // Pass onto Karma to emit BuildEvents.\n          karmaConfig.buildWebpack ??= {};\n          if (typeof karmaConfig.buildWebpack === 'object') {\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            (karmaConfig.buildWebpack as any).failureCb ??= () =>\n              subscriber.next({ success: false });\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            (karmaConfig.buildWebpack as any).successCb ??= () =>\n              subscriber.next({ success: true });\n          }\n\n          // Complete the observable once the Karma server returns.\n          const karmaServer = new karma.Server(karmaConfig as Config, (exitCode) => {\n            subscriber.next({ success: exitCode === 0 });\n            subscriber.complete();\n          });\n\n          const karmaStart = karmaServer.start();\n\n          // Cleanup, signal Karma to exit.\n          return () => {\n            void karmaStart.then(() => karmaServer.stop());\n          };\n        }),\n    ),\n    defaultIfEmpty({ success: false }),\n  );\n}\n\nfunction getBuiltInKarmaConfig(\n  workspaceRoot: string,\n  projectName: string,\n): ConfigOptions & Record<string, unknown> {\n  let coverageFolderName = projectName.charAt(0) === '@' ? projectName.slice(1) : projectName;\n  if (/[A-Z]/.test(coverageFolderName)) {\n    coverageFolderName = strings.dasherize(coverageFolderName);\n  }\n\n  const workspaceRootRequire = createRequire(workspaceRoot + '/');\n\n  // Any changes to the config here need to be synced to: packages/schematics/angular/config/files/karma.conf.js.template\n  return {\n    basePath: '',\n    frameworks: ['jasmine', '@angular-devkit/build-angular'],\n    plugins: [\n      'karma-jasmine',\n      'karma-chrome-launcher',\n      'karma-jasmine-html-reporter',\n      'karma-coverage',\n      '@angular-devkit/build-angular/plugins/karma',\n    ].map((p) => workspaceRootRequire(p)),\n    client: {\n      clearContext: false, // leave Jasmine Spec Runner output visible in browser\n    },\n    jasmineHtmlReporter: {\n      suppressAll: true, // removes the duplicated traces\n    },\n    coverageReporter: {\n      dir: path.join(workspaceRoot, 'coverage', coverageFolderName),\n      subdir: '.',\n      reporters: [{ type: 'html' }, { type: 'text-summary' }],\n    },\n    reporters: ['progress', 'kjhtml'],\n    browsers: ['Chrome'],\n    customLaunchers: {\n      // Chrome configured to run in a bazel sandbox.\n      // Disable the use of the gpu and `/dev/shm` because it causes Chrome to\n      // crash on some environments.\n      // See:\n      //   https://github.com/puppeteer/puppeteer/blob/v1.0.0/docs/troubleshooting.md#tips\n      //   https://stackoverflow.com/questions/50642308/webdriverexception-unknown-error-devtoolsactiveport-file-doesnt-exist-while-t\n      ChromeHeadlessNoSandbox: {\n        base: 'ChromeHeadless',\n        flags: ['--no-sandbox', '--headless', '--disable-gpu', '--disable-dev-shm-usage'],\n      },\n    },\n    restartOnFileChange: true,\n  };\n}\n\nexport { KarmaBuilderOptions };\nexport default createBuilder<Record<string, string> & KarmaBuilderOptions>(execute);\n\nfunction getBuiltInMainFile(): string {\n  const content = Buffer.from(\n    `\n  import { getTestBed } from '@angular/core/testing';\n  import {\n    BrowserDynamicTestingModule,\n    platformBrowserDynamicTesting,\n   } from '@angular/platform-browser-dynamic/testing';\n\n  // Initialize the Angular testing environment.\n  getTestBed().initTestEnvironment(BrowserDynamicTestingModule, platformBrowserDynamicTesting(), {\n    errorOnUnknownElements: true,\n    errorOnUnknownProperties: true\n  });\n`,\n  ).toString('base64');\n\n  return `ng-virtual-main.js!=!data:text/javascript;base64,${content}`;\n}\n"]}