@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
161 lines • 20.2 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 });
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"]}