UNPKG

@angular-devkit/build-angular

Version:
161 lines • 20.2 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 }); const architect_1 = require("@angular-devkit/architect"); const child_process_1 = require("child_process"); const path = __importStar(require("path")); const util_1 = require("util"); const color_1 = require("../../utils/color"); const application_1 = require("../application"); const schema_1 = require("../browser-esbuild/schema"); const options_1 = require("./options"); const test_files_1 = require("./test-files"); const execFile = (0, util_1.promisify)(child_process_1.execFile); /** Main execution function for the Jest builder. */ exports.default = (0, architect_1.createBuilder)(async (schema, context) => { context.logger.warn('NOTE: The Jest builder is currently EXPERIMENTAL and not ready for production use.'); const options = (0, options_1.normalizeOptions)(schema); const testOut = 'dist/test-out'; // TODO(dgp1130): Hide in temp directory. // Verify Jest installation and get the path to it's binary. // We need to `node_modules/.bin/jest`, but there is no means to resolve that directly. Fortunately Jest's `package.json` exports the // same file at `bin/jest`, so we can just resolve that instead. const jest = resolveModule('jest/bin/jest'); if (!jest) { return { success: false, // TODO(dgp1130): Display a more accurate message for non-NPM users. error: 'Jest is not installed, most likely you need to run `npm install jest --save-dev` in your project.', }; } // Verify that JSDom is installed in the project. const environment = resolveModule('jest-environment-jsdom'); if (!environment) { return { success: false, // TODO(dgp1130): Display a more accurate message for non-NPM users. error: '`jest-environment-jsdom` is not installed. Install it with `npm install jest-environment-jsdom --save-dev`.', }; } // Build all the test files. const testFiles = await (0, test_files_1.findTestFiles)(options, context.workspaceRoot); const jestGlobal = path.join(__dirname, 'jest-global.mjs'); const initTestBed = path.join(__dirname, 'init-test-bed.mjs'); const buildResult = await build(context, { // Build all the test files and also the `jest-global` and `init-test-bed` scripts. entryPoints: new Set([...testFiles, jestGlobal, initTestBed]), tsConfig: options.tsConfig, polyfills: options.polyfills ?? ['zone.js', 'zone.js/testing'], outputPath: testOut, aot: false, index: false, outputHashing: schema_1.OutputHashing.None, outExtension: 'mjs', optimization: false, sourceMap: { scripts: true, styles: false, vendor: false, }, }); if (!buildResult.success) { return buildResult; } // Execute Jest on the built output directory. const jestProc = execFile(process.execPath, [ '--experimental-vm-modules', jest, `--rootDir="${testOut}"`, '--testEnvironment=jsdom', // TODO(dgp1130): Enable cache once we have a mechanism for properly clearing / disabling it. '--no-cache', // Run basically all files in the output directory, any excluded files were already dropped by the build. `--testMatch="<rootDir>/**/*.mjs"`, // Load polyfills and initialize the environment before executing each test file. // IMPORTANT: Order matters here. // First, we execute `jest-global.mjs` to initialize the `jest` global variable. // Second, we execute user polyfills, including `zone.js` and `zone.js/testing`. This is dependent on the Jest global so it can patch // the environment for fake async to work correctly. // Third, we initialize `TestBed`. This is dependent on fake async being set up correctly beforehand. `--setupFilesAfterEnv="<rootDir>/jest-global.mjs"`, ...(options.polyfills ? [`--setupFilesAfterEnv="<rootDir>/polyfills.mjs"`] : []), `--setupFilesAfterEnv="<rootDir>/init-test-bed.mjs"`, // Don't run any infrastructure files as tests, they are manually loaded where needed. `--testPathIgnorePatterns="<rootDir>/jest-global\\.mjs"`, ...(options.polyfills ? [`--testPathIgnorePatterns="<rootDir>/polyfills\\.mjs"`] : []), `--testPathIgnorePatterns="<rootDir>/init-test-bed\\.mjs"`, // Skip shared chunks, as they are not entry points to tests. `--testPathIgnorePatterns="<rootDir>/chunk-.*\\.mjs"`, // Optionally enable color. ...(color_1.colors.enabled ? ['--colors'] : []), ]); // Stream test output to the terminal. jestProc.child.stdout?.on('data', (chunk) => { context.logger.info(chunk); }); jestProc.child.stderr?.on('data', (chunk) => { // Write to stderr directly instead of `context.logger.error(chunk)` because the logger will overwrite Jest's coloring information. process.stderr.write(chunk); }); try { await jestProc; } catch (error) { // No need to propagate error message, already piped to terminal output. // TODO(dgp1130): Handle process spawning failures. return { success: false }; } return { success: true }; }); async function build(context, options) { try { for await (const _ of (0, application_1.buildApplicationInternal)(options, context)) { // Nothing to do for each event, just wait for the whole build. } return { success: true }; } catch (err) { return { success: false, error: err.message, }; } } /** Safely resolves the given Node module string. */ function resolveModule(module) { try { return require.resolve(module); } catch { return undefined; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/jest/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 { execFile as execFileCb } from 'child_process';\nimport * as path from 'path';\nimport { promisify } from 'util';\nimport { colors } from '../../utils/color';\nimport { buildApplicationInternal } from '../application';\nimport { ApplicationBuilderInternalOptions } from '../application/options';\nimport { OutputHashing } from '../browser-esbuild/schema';\nimport { normalizeOptions } from './options';\nimport { Schema as JestBuilderSchema } from './schema';\nimport { findTestFiles } from './test-files';\n\nconst execFile = promisify(execFileCb);\n\n/** Main execution function for the Jest builder. */\nexport default createBuilder(\n  async (schema: JestBuilderSchema, context: BuilderContext): Promise<BuilderOutput> => {\n    context.logger.warn(\n      'NOTE: The Jest builder is currently EXPERIMENTAL and not ready for production use.',\n    );\n\n    const options = normalizeOptions(schema);\n    const testOut = 'dist/test-out'; // TODO(dgp1130): Hide in temp directory.\n\n    // Verify Jest installation and get the path to it's binary.\n    // We need to `node_modules/.bin/jest`, but there is no means to resolve that directly. Fortunately Jest's `package.json` exports the\n    // same file at `bin/jest`, so we can just resolve that instead.\n    const jest = resolveModule('jest/bin/jest');\n    if (!jest) {\n      return {\n        success: false,\n        // TODO(dgp1130): Display a more accurate message for non-NPM users.\n        error:\n          'Jest is not installed, most likely you need to run `npm install jest --save-dev` in your project.',\n      };\n    }\n\n    // Verify that JSDom is installed in the project.\n    const environment = resolveModule('jest-environment-jsdom');\n    if (!environment) {\n      return {\n        success: false,\n        // TODO(dgp1130): Display a more accurate message for non-NPM users.\n        error:\n          '`jest-environment-jsdom` is not installed. Install it with `npm install jest-environment-jsdom --save-dev`.',\n      };\n    }\n\n    // Build all the test files.\n    const testFiles = await findTestFiles(options, context.workspaceRoot);\n    const jestGlobal = path.join(__dirname, 'jest-global.mjs');\n    const initTestBed = path.join(__dirname, 'init-test-bed.mjs');\n    const buildResult = await build(context, {\n      // Build all the test files and also the `jest-global` and `init-test-bed` scripts.\n      entryPoints: new Set([...testFiles, jestGlobal, initTestBed]),\n      tsConfig: options.tsConfig,\n      polyfills: options.polyfills ?? ['zone.js', 'zone.js/testing'],\n      outputPath: testOut,\n      aot: false,\n      index: false,\n      outputHashing: OutputHashing.None,\n      outExtension: 'mjs', // Force native ESM.\n      optimization: false,\n      sourceMap: {\n        scripts: true,\n        styles: false,\n        vendor: false,\n      },\n    });\n    if (!buildResult.success) {\n      return buildResult;\n    }\n\n    // Execute Jest on the built output directory.\n    const jestProc = execFile(process.execPath, [\n      '--experimental-vm-modules',\n      jest,\n\n      `--rootDir=\"${testOut}\"`,\n      '--testEnvironment=jsdom',\n\n      // TODO(dgp1130): Enable cache once we have a mechanism for properly clearing / disabling it.\n      '--no-cache',\n\n      // Run basically all files in the output directory, any excluded files were already dropped by the build.\n      `--testMatch=\"<rootDir>/**/*.mjs\"`,\n\n      // Load polyfills and initialize the environment before executing each test file.\n      // IMPORTANT: Order matters here.\n      // First, we execute `jest-global.mjs` to initialize the `jest` global variable.\n      // Second, we execute user polyfills, including `zone.js` and `zone.js/testing`. This is dependent on the Jest global so it can patch\n      // the environment for fake async to work correctly.\n      // Third, we initialize `TestBed`. This is dependent on fake async being set up correctly beforehand.\n      `--setupFilesAfterEnv=\"<rootDir>/jest-global.mjs\"`,\n      ...(options.polyfills ? [`--setupFilesAfterEnv=\"<rootDir>/polyfills.mjs\"`] : []),\n      `--setupFilesAfterEnv=\"<rootDir>/init-test-bed.mjs\"`,\n\n      // Don't run any infrastructure files as tests, they are manually loaded where needed.\n      `--testPathIgnorePatterns=\"<rootDir>/jest-global\\\\.mjs\"`,\n      ...(options.polyfills ? [`--testPathIgnorePatterns=\"<rootDir>/polyfills\\\\.mjs\"`] : []),\n      `--testPathIgnorePatterns=\"<rootDir>/init-test-bed\\\\.mjs\"`,\n\n      // Skip shared chunks, as they are not entry points to tests.\n      `--testPathIgnorePatterns=\"<rootDir>/chunk-.*\\\\.mjs\"`,\n\n      // Optionally enable color.\n      ...(colors.enabled ? ['--colors'] : []),\n    ]);\n\n    // Stream test output to the terminal.\n    jestProc.child.stdout?.on('data', (chunk) => {\n      context.logger.info(chunk);\n    });\n    jestProc.child.stderr?.on('data', (chunk) => {\n      // Write to stderr directly instead of `context.logger.error(chunk)` because the logger will overwrite Jest's coloring information.\n      process.stderr.write(chunk);\n    });\n\n    try {\n      await jestProc;\n    } catch (error) {\n      // No need to propagate error message, already piped to terminal output.\n      // TODO(dgp1130): Handle process spawning failures.\n      return { success: false };\n    }\n\n    return { success: true };\n  },\n);\n\nasync function build(\n  context: BuilderContext,\n  options: ApplicationBuilderInternalOptions,\n): Promise<BuilderOutput> {\n  try {\n    for await (const _ of buildApplicationInternal(options, context)) {\n      // Nothing to do for each event, just wait for the whole build.\n    }\n\n    return { success: true };\n  } catch (err) {\n    return {\n      success: false,\n      error: (err as Error).message,\n    };\n  }\n}\n\n/** Safely resolves the given Node module string. */\nfunction resolveModule(module: string): string | undefined {\n  try {\n    return require.resolve(module);\n  } catch {\n    return undefined;\n  }\n}\n"]}