UNPKG

@angular-devkit/build-angular

Version:
145 lines (143 loc) 6.92 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.dev/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 () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); Object.defineProperty(exports, "__esModule", { value: true }); exports.execute = execute; const private_1 = require("@angular/build/private"); const path = __importStar(require("node:path")); const rxjs_1 = require("rxjs"); const configs_1 = require("../../tools/webpack/configs"); 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"); function execute(options, context, karmaOptions, transforms = {}) { return (0, rxjs_1.from)(initializeBrowser(options, context, transforms.webpackConfiguration)).pipe((0, rxjs_1.switchMap)(async ([karma, webpackConfig]) => { const projectName = context.target?.project; if (!projectName) { throw new Error(`The 'karma' builder requires a target to be specified.`); } const projectMetadata = await context.getProjectMetadata(projectName); const sourceRoot = (projectMetadata.sourceRoot ?? projectMetadata.root ?? ''); if (!options.main) { 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()]; } } } 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) => { // Pass onto Karma to emit BuildEvents. karmaConfig.buildWebpack ??= {}; if (typeof karmaConfig.buildWebpack === 'object') { // eslint-disable-next-line @typescript-eslint/no-explicit-any karmaConfig.buildWebpack.failureCb ??= () => subscriber.next({ success: false }); // eslint-disable-next-line @typescript-eslint/no-explicit-any karmaConfig.buildWebpack.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 })); } async function initializeBrowser(options, context, webpackConfigurationTransformer) { // Purge old build disk cache. await (0, private_1.purgeStaleBuildCache)(context); const karma = await Promise.resolve().then(() => __importStar(require('karma'))); 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: options.aot, 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)]); return [karma, (await webpackConfigurationTransformer?.(config)) ?? config]; } function getBuiltInMainFile() { const content = Buffer.from(` import { getTestBed } from '@angular/core/testing'; import { BrowserTestingModule, platformBrowserTesting } from '@angular/platform-browser/testing'; // Initialize the Angular testing environment. getTestBed().initTestEnvironment(BrowserTestingModule, platformBrowserTesting(), { errorOnUnknownElements: true, errorOnUnknownProperties: true }); `).toString('base64'); return `ng-virtual-main.js!=!data:text/javascript;base64,${content}`; }