projen
Version:
CDK for software projects
325 lines • 68.8 kB
JavaScript
"use strict";
var _a, _b, _c, _d;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Jest = exports.JestReporter = exports.UpdateSnapshot = exports.WatchPlugin = exports.Transform = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const path = require("path");
const component_1 = require("../component");
const javascript_1 = require("../javascript");
const json_1 = require("../json");
const util_1 = require("../util");
const constructs_1 = require("../util/constructs");
const DEFAULT_TEST_REPORTS_DIR = "test-reports";
class Transform {
constructor(name, options) {
this.name = name;
this.options = options;
}
/**
* @jsii ignore
* @internal
*/
toJSON() {
if (this.options != null) {
return [this.name, this.options];
}
return this.name;
}
}
exports.Transform = Transform;
_a = JSII_RTTI_SYMBOL_1;
Transform[_a] = { fqn: "projen.javascript.Transform", version: "0.99.51" };
class WatchPlugin {
constructor(name, options) {
this.name = name;
this.options = options;
}
/**
* @jsii ignore
* @internal
*/
toJSON() {
if (this.options != null) {
return [this.name, this.options];
}
return this.name;
}
}
exports.WatchPlugin = WatchPlugin;
_b = JSII_RTTI_SYMBOL_1;
WatchPlugin[_b] = { fqn: "projen.javascript.WatchPlugin", version: "0.99.51" };
var UpdateSnapshot;
(function (UpdateSnapshot) {
/**
* Always update snapshots in "test" task.
*/
UpdateSnapshot["ALWAYS"] = "always";
/**
* Never update snapshots in "test" task and create a separate "test:update" task.
*/
UpdateSnapshot["NEVER"] = "never";
})(UpdateSnapshot || (exports.UpdateSnapshot = UpdateSnapshot = {}));
class JestReporter {
constructor(name, options) {
this.name = name;
this.options = options;
}
/**
* @jsii ignore
* @internal
*/
toJSON() {
if (this.options == null) {
return this.name;
}
return [this.name, this.options];
}
}
exports.JestReporter = JestReporter;
_c = JSII_RTTI_SYMBOL_1;
JestReporter[_c] = { fqn: "projen.javascript.JestReporter", version: "0.99.51" };
/**
* Installs the following npm scripts:
*
* - `test`, intended for testing locally and in CI. Will update snapshots unless `updateSnapshot: UpdateSnapshot: NEVER` is set.
* - `test:watch`, intended for automatically rerunning tests when files change.
* - `test:update`, intended for testing locally and updating snapshots to match the latest unit under test. Only available when `updateSnapshot: UpdateSnapshot: NEVER`.
*
*/
class Jest extends component_1.Component {
/**
* Returns the singleton Jest component of a project or undefined if there is none.
*/
static of(project) {
const isJest = (c) => c instanceof Jest;
return project.components.find(isJest);
}
constructor(scope, options = {}) {
super(scope);
this.testMatch = new Array();
this.project = (0, constructs_1.closestProjectMustBe)(scope, javascript_1.NodeProject, new.target.name);
// hard deprecation
if (options.typescriptConfig) {
throw new Error('"jestOptions.typescriptConfig" is deprecated. Use "typescriptProject.tsconfigDev" instead');
}
// Jest snapshot files are generated files!
this.project.root.annotateGenerated("*.snap");
this.jestVersion = options.jestVersion ? `@${options.jestVersion}` : "";
this.project.addDevDeps(`jest${this.jestVersion}`);
// use native v8 coverage collection as default
// https://jestjs.io/docs/en/cli#--coverageproviderprovider
const coverageProvider = this.jestConfig?.coverageProvider ?? "v8";
this.jestConfig = {
coverageProvider,
...options.jestConfig,
additionalOptions: undefined,
...options.jestConfig?.additionalOptions,
};
this.extraCliOptions = options.extraCliOptions ?? [];
this.passWithNoTests = options.passWithNoTests ?? true;
this.ignorePatterns = this.jestConfig?.testPathIgnorePatterns ??
options.ignorePatterns ?? ["/node_modules/"];
this.watchIgnorePatterns = this.jestConfig?.watchPathIgnorePatterns ?? [
"/node_modules/",
];
this.coverageReporters = this.jestConfig?.coverageReporters ?? [
"json",
"lcov",
"clover",
"cobertura",
];
if (this.jestConfig?.testMatch && this.jestConfig.testMatch.length > 0) {
this.jestConfig.testMatch.forEach((pattern) => this.addTestMatch(pattern));
}
const coverageDirectory = this.jestConfig?.coverageDirectory ?? "coverage";
this.reporters = [];
if (options.preserveDefaultReporters ?? true) {
this.reporters.unshift(new JestReporter("default"));
}
this.config = {
...this.jestConfig,
clearMocks: this.jestConfig?.clearMocks ?? true,
collectCoverage: options.coverage ?? this.jestConfig?.collectCoverage ?? true,
coverageReporters: this.coverageReporters,
coverageDirectory: coverageDirectory,
coveragePathIgnorePatterns: this.jestConfig?.coveragePathIgnorePatterns ?? this.ignorePatterns,
testPathIgnorePatterns: this.ignorePatterns,
watchPathIgnorePatterns: this.watchIgnorePatterns,
// @ts-expect-error - lazily loading the testMatch in order to only apply defaults if none are ever added
testMatch: () => this.testMatch.length > 0
? this.testMatch
: [`**/__tests__/**/*.[jt]s?(x)`, `**/*(*.)@(spec|test).[jt]s?(x)`], // Jest defaults
reporters: this.reporters,
snapshotResolver: (() => this._snapshotResolver),
};
if (options.junitReporting ?? true) {
const reportsDir = DEFAULT_TEST_REPORTS_DIR;
this.addReporter(new JestReporter("jest-junit", { outputDirectory: reportsDir }));
this.project.addDevDeps("jest-junit@^16");
this.project.gitignore.exclude("# jest-junit artifacts", `/${reportsDir}/`, "junit.xml");
this.project.npmignore?.exclude("# jest-junit artifacts", `/${reportsDir}/`, "junit.xml");
}
if (this.jestConfig?.reporters) {
for (const reporter of this.jestConfig.reporters) {
this.addReporter(reporter);
}
}
if (this.jestConfig?.coverageThreshold) {
this.config.coverageThreshold = {
global: this.jestConfig?.coverageThreshold,
};
}
this.configureTestCommand(options.updateSnapshot ?? UpdateSnapshot.ALWAYS);
if (options.configFilePath) {
this.file = new json_1.JsonFile(this.project, options.configFilePath, {
obj: this.config,
});
this.project.npmignore?.addPatterns(`/${this.file.path}`);
}
else {
this.project.addFields({ jest: this.config });
}
const coverageDirectoryPath = path.posix.join("/", coverageDirectory, "/");
this.project.npmignore?.exclude(coverageDirectoryPath);
this.project.gitignore.exclude(coverageDirectoryPath);
if (options.coverageText ?? true) {
this.coverageReporters.push("text");
}
}
/**
* Adds a test match pattern.
* @param pattern glob pattern to match for tests
*/
addTestMatch(pattern) {
this.testMatch.push(pattern);
}
/**
* Build standard test match patterns for a directory.
* @param dirs The directories to add test matches for. Matches any folder if not specified or an empty array.
* @param options Options for building test match patterns.
*/
discoverTestMatchPatternsForDirs(dirs, options) {
const testPatterns = this.buildTestMatchPatternsForDirs(dirs, options);
testPatterns.forEach((pattern) => this.addTestMatch(pattern));
}
/**
* Build standard test match patterns for a directory.
* @param dirs The directories to add test matches for. Matches any folder if not specified.
* @param fileExtensionPattern The file extension pattern to use. Defaults to "[jt]s?(x)".
* @returns The test match patterns.
*/
buildTestMatchPatternsForDirs(dirs, options) {
const fileExtensionPattern = options?.fileExtensionPattern ?? "[jt]s?(x)";
return [
`<rootDir>/@(${dirs.join("|")})/**/*(*.)@(spec|test).${fileExtensionPattern}`,
`<rootDir>/@(${dirs.join("|")})/**/__tests__/**/*.${fileExtensionPattern}`,
];
}
/**
* Adds a watch ignore pattern.
* @param pattern The pattern (regular expression).
*/
addWatchIgnorePattern(pattern) {
this.watchIgnorePatterns.push(pattern);
}
addIgnorePattern(pattern) {
this.ignorePatterns.push(pattern);
}
addReporter(reporter) {
this.reporters.push(reporter);
}
/**
* Adds a a setup file to Jest's setupFiles configuration.
* @param file File path to setup file
*/
addSetupFile(file) {
if (!this.config.setupFiles) {
this.config.setupFiles = [];
}
this.config.setupFiles.push(file);
}
/**
* Adds a a setup file to Jest's setupFilesAfterEnv configuration.
* @param file File path to setup file
*/
addSetupFileAfterEnv(file) {
if (!this.config.setupFilesAfterEnv) {
this.config.setupFilesAfterEnv = [];
}
this.config.setupFilesAfterEnv.push(file);
}
addSnapshotResolver(file) {
const normalized = (0, util_1.normalizePersistedPath)(file);
this._snapshotResolver = normalized;
}
/**
* Adds one or more moduleNameMapper entries to Jest's configuration.
* Will overwrite if the same key is used as a pre-existing one.
*
* @param moduleNameMapperAdditions - A map from regular expressions to module names or to arrays of module names that allow to stub out resources, like images or styles with a single module.
*/
addModuleNameMappers(moduleNameMapperAdditions) {
const existingModuleNameMapper = this.config.moduleNameMapper ?? {};
this.config.moduleNameMapper = {
...existingModuleNameMapper,
...moduleNameMapperAdditions,
};
}
/**
* Adds one or more modulePaths to Jest's configuration.
*
* @param modulePaths - An array of absolute paths to additional locations to search when resolving modules *
*/
addModulePaths(...modulePaths) {
const existingModulePaths = this.config.modulePaths ?? [];
this.config.modulePaths = [
...new Set([...existingModulePaths, ...modulePaths]),
];
}
/**
* Adds one or more roots to Jest's configuration.
*
* @param roots - A list of paths to directories that Jest should use to search for files in.
*/
addRoots(...roots) {
const existingRoots = this.config.roots ?? [];
this.config.roots = [...new Set([...existingRoots, ...roots])];
}
configureTestCommand(updateSnapshot) {
const jestOpts = this.extraCliOptions;
const jestConfigOpts = this.file && this.file.path != "jest.config.json"
? ` -c ${this.file.path}`
: "";
if (this.passWithNoTests) {
jestOpts.push("--passWithNoTests");
}
if (updateSnapshot === UpdateSnapshot.ALWAYS) {
jestOpts.push("--updateSnapshot");
}
else {
jestOpts.push("--ci"); // to prevent accepting new snapshots
const testUpdate = this.project.tasks.tryFind("test:update");
if (!testUpdate) {
this.project.addTask("test:update", {
description: "Update jest snapshots",
exec: `jest --updateSnapshot ${jestOpts.join(" ")}${jestConfigOpts}`,
receiveArgs: true,
});
}
}
this.project.testTask.exec(`jest ${jestOpts.join(" ")}${jestConfigOpts}`, {
receiveArgs: true,
});
const testWatch = this.project.tasks.tryFind("test:watch");
if (!testWatch) {
this.project.addTask("test:watch", {
description: "Run jest in watch mode",
exec: `jest --watch${jestConfigOpts}`,
});
}
}
}
exports.Jest = Jest;
_d = JSII_RTTI_SYMBOL_1;
Jest[_d] = { fqn: "projen.javascript.Jest", version: "0.99.51" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"jest.js","sourceRoot":"","sources":["../../src/javascript/jest.ts"],"names":[],"mappings":";;;;;AAAA,6BAA6B;AAE7B,4CAAyC;AACzC,8CAA4C;AAC5C,kCAAmC;AAEnC,kCAAiD;AACjD,mDAA0D;AAE1D,MAAM,wBAAwB,GAAG,cAAc,CAAC;AA2ehD,MAAa,SAAS;IACpB,YACmB,IAAY,EACZ,OAAa;QADb,SAAI,GAAJ,IAAI,CAAQ;QACZ,YAAO,GAAP,OAAO,CAAM;IAC7B,CAAC;IAEJ;;;OAGG;IACI,MAAM;QACX,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;YACzB,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QACnC,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;;AAfH,8BAgBC;;;AAED,MAAa,WAAW;IACtB,YACmB,IAAY,EACZ,OAAa;QADb,SAAI,GAAJ,IAAI,CAAQ;QACZ,YAAO,GAAP,OAAO,CAAM;IAC7B,CAAC;IAEJ;;;OAGG;IACI,MAAM;QACX,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;YACzB,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QACnC,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;;AAfH,kCAgBC;;;AA6FD,IAAY,cAUX;AAVD,WAAY,cAAc;IACxB;;OAEG;IACH,mCAAiB,CAAA;IAEjB;;OAEG;IACH,iCAAe,CAAA;AACjB,CAAC,EAVW,cAAc,8BAAd,cAAc,QAUzB;AAUD,MAAa,YAAY;IACvB,YACmB,IAAY,EACZ,OAAgC;QADhC,SAAI,GAAJ,IAAI,CAAQ;QACZ,YAAO,GAAP,OAAO,CAAyB;IAChD,CAAC;IAEJ;;;OAGG;IACI,MAAM;QACX,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;YACzB,OAAO,IAAI,CAAC,IAAI,CAAC;QACnB,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;IACnC,CAAC;;AAfH,oCAgBC;;;AAED;;;;;;;GAOG;AACH,MAAa,IAAK,SAAQ,qBAAS;IACjC;;OAEG;IACI,MAAM,CAAC,EAAE,CAAC,OAAgB;QAC/B,MAAM,MAAM,GAAG,CAAC,CAAY,EAAa,EAAE,CAAC,CAAC,YAAY,IAAI,CAAC;QAC9D,OAAO,OAAO,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IACzC,CAAC;IAgCD,YAAY,KAAiB,EAAE,UAAuB,EAAE;QACtD,KAAK,CAAC,KAAK,CAAC,CAAC;QAdE,cAAS,GAAG,IAAI,KAAK,EAAU,CAAC;QAe/C,IAAI,CAAC,OAAO,GAAG,IAAA,iCAAoB,EAAC,KAAK,EAAE,wBAAW,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAEzE,mBAAmB;QACnB,IAAK,OAAe,CAAC,gBAAgB,EAAE,CAAC;YACtC,MAAM,IAAI,KAAK,CACb,2FAA2F,CAC5F,CAAC;QACJ,CAAC;QAED,2CAA2C;QAC3C,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;QAC9C,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACxE,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,OAAO,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QAEnD,+CAA+C;QAC/C,2DAA2D;QAC3D,MAAM,gBAAgB,GAAG,IAAI,CAAC,UAAU,EAAE,gBAAgB,IAAI,IAAI,CAAC;QAEnE,IAAI,CAAC,UAAU,GAAG;YAChB,gBAAgB;YAChB,GAAG,OAAO,CAAC,UAAU;YACrB,iBAAiB,EAAE,SAAS;YAC5B,GAAG,OAAO,CAAC,UAAU,EAAE,iBAAiB;SACzC,CAAC;QACF,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,EAAE,CAAC;QACrD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,IAAI,CAAC;QAEvD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,EAAE,sBAAsB;YAC3D,OAAO,CAAC,cAAc,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC/C,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,UAAU,EAAE,uBAAuB,IAAI;YACrE,gBAAgB;SACjB,CAAC;QACF,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,UAAU,EAAE,iBAAiB,IAAI;YAC7D,MAAM;YACN,MAAM;YACN,QAAQ;YACR,WAAW;SACZ,CAAC;QAEF,IAAI,IAAI,CAAC,UAAU,EAAE,SAAS,IAAI,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvE,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAC5C,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAC3B,CAAC;QACJ,CAAC;QAED,MAAM,iBAAiB,GAAG,IAAI,CAAC,UAAU,EAAE,iBAAiB,IAAI,UAAU,CAAC;QAE3E,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;QAEpB,IAAI,OAAO,CAAC,wBAAwB,IAAI,IAAI,EAAE,CAAC;YAC7C,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,CAAC,MAAM,GAAG;YACZ,GAAG,IAAI,CAAC,UAAU;YAClB,UAAU,EAAE,IAAI,CAAC,UAAU,EAAE,UAAU,IAAI,IAAI;YAC/C,eAAe,EACb,OAAO,CAAC,QAAQ,IAAI,IAAI,CAAC,UAAU,EAAE,eAAe,IAAI,IAAI;YAC9D,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;YACzC,iBAAiB,EAAE,iBAAiB;YACpC,0BAA0B,EACxB,IAAI,CAAC,UAAU,EAAE,0BAA0B,IAAI,IAAI,CAAC,cAAc;YACpE,sBAAsB,EAAE,IAAI,CAAC,cAAc;YAC3C,uBAAuB,EAAE,IAAI,CAAC,mBAAmB;YACjD,yGAAyG;YACzG,SAAS,EAAE,GAAG,EAAE,CACd,IAAI,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC;gBACvB,CAAC,CAAC,IAAI,CAAC,SAAS;gBAChB,CAAC,CAAC,CAAC,6BAA6B,EAAE,gCAAgC,CAAC,EAAE,gBAAgB;YACzF,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,gBAAgB,EAAE,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAQ;SAC5B,CAAC;QAE9B,IAAI,OAAO,CAAC,cAAc,IAAI,IAAI,EAAE,CAAC;YACnC,MAAM,UAAU,GAAG,wBAAwB,CAAC;YAE5C,IAAI,CAAC,WAAW,CACd,IAAI,YAAY,CAAC,YAAY,EAAE,EAAE,eAAe,EAAE,UAAU,EAAE,CAAC,CAChE,CAAC;YAEF,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;YAE1C,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,CAC5B,wBAAwB,EACxB,IAAI,UAAU,GAAG,EACjB,WAAW,CACZ,CAAC;YACF,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAC7B,wBAAwB,EACxB,IAAI,UAAU,GAAG,EACjB,WAAW,CACZ,CAAC;QACJ,CAAC;QAED,IAAI,IAAI,CAAC,UAAU,EAAE,SAAS,EAAE,CAAC;YAC/B,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,CAAC;gBACjD,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;YAC7B,CAAC;QACH,CAAC;QAED,IAAI,IAAI,CAAC,UAAU,EAAE,iBAAiB,EAAE,CAAC;YACvC,IAAI,CAAC,MAAM,CAAC,iBAAiB,GAAG;gBAC9B,MAAM,EAAE,IAAI,CAAC,UAAU,EAAE,iBAAiB;aAC3C,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,cAAc,IAAI,cAAc,CAAC,MAAM,CAAC,CAAC;QAE3E,IAAI,OAAO,CAAC,cAAc,EAAE,CAAC;YAC3B,IAAI,CAAC,IAAI,GAAG,IAAI,eAAQ,CAAC,IAAI,CAAC,OAAO,EAAE,OAAO,CAAC,cAAc,EAAE;gBAC7D,GAAG,EAAE,IAAI,CAAC,MAAM;aACjB,CAAC,CAAC;YACH,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,WAAW,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAC5D,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAChD,CAAC;QAED,MAAM,qBAAqB,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,iBAAiB,EAAE,GAAG,CAAC,CAAC;QAC3E,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,OAAO,CAAC,qBAAqB,CAAC,CAAC;QACvD,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,CAAC,qBAAqB,CAAC,CAAC;QAEtD,IAAI,OAAO,CAAC,YAAY,IAAI,IAAI,EAAE,CAAC;YACjC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACtC,CAAC;IACH,CAAC;IAED;;;OAGG;IACI,YAAY,CAAC,OAAe;QACjC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACI,gCAAgC,CACrC,IAAc,EACd,OAAqD;QAErD,MAAM,YAAY,GAAG,IAAI,CAAC,6BAA6B,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;QAEvE,YAAY,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;IAChE,CAAC;IAED;;;;;OAKG;IACK,6BAA6B,CACnC,IAAc,EACd,OAAqD;QAErD,MAAM,oBAAoB,GAAG,OAAO,EAAE,oBAAoB,IAAI,WAAW,CAAC;QAC1E,OAAO;YACL,eAAe,IAAI,CAAC,IAAI,CACtB,GAAG,CACJ,0BAA0B,oBAAoB,EAAE;YACjD,eAAe,IAAI,CAAC,IAAI,CACtB,GAAG,CACJ,uBAAuB,oBAAoB,EAAE;SAC/C,CAAC;IACJ,CAAC;IAED;;;OAGG;IACI,qBAAqB,CAAC,OAAe;QAC1C,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACzC,CAAC;IAEM,gBAAgB,CAAC,OAAe;QACrC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IACpC,CAAC;IAEM,WAAW,CAAC,QAAsB;QACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAChC,CAAC;IAED;;;OAGG;IACI,YAAY,CAAC,IAAY;QAC9B,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;YAC5B,IAAI,CAAC,MAAM,CAAC,UAAU,GAAG,EAAE,CAAC;QAC9B,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACpC,CAAC;IAED;;;OAGG;IACI,oBAAoB,CAAC,IAAY;QACtC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;YACpC,IAAI,CAAC,MAAM,CAAC,kBAAkB,GAAG,EAAE,CAAC;QACtC,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC5C,CAAC;IAEM,mBAAmB,CAAC,IAAY;QACrC,MAAM,UAAU,GAAG,IAAA,6BAAsB,EAAC,IAAI,CAAC,CAAC;QAEhD,IAAI,CAAC,iBAAiB,GAAG,UAAU,CAAC;IACtC,CAAC;IAED;;;;;OAKG;IACI,oBAAoB,CAAC,yBAE3B;QACC,MAAM,wBAAwB,GAAG,IAAI,CAAC,MAAM,CAAC,gBAAgB,IAAI,EAAE,CAAC;QACpE,IAAI,CAAC,MAAM,CAAC,gBAAgB,GAAG;YAC7B,GAAG,wBAAwB;YAC3B,GAAG,yBAAyB;SAC7B,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACI,cAAc,CAAC,GAAG,WAAqB;QAC5C,MAAM,mBAAmB,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,IAAI,EAAE,CAAC;QAC1D,IAAI,CAAC,MAAM,CAAC,WAAW,GAAG;YACxB,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,mBAAmB,EAAE,GAAG,WAAW,CAAC,CAAC;SACrD,CAAC;IACJ,CAAC;IAED;;;;OAIG;IACI,QAAQ,CAAC,GAAG,KAAe;QAChC,MAAM,aAAa,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;QAC9C,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,aAAa,EAAE,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;IACjE,CAAC;IAEO,oBAAoB,CAAC,cAA8B;QACzD,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC;QACtC,MAAM,cAAc,GAClB,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,CAAC,IAAI,IAAI,kBAAkB;YAC/C,CAAC,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE;YACzB,CAAC,CAAC,EAAE,CAAC;QAET,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,QAAQ,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;QACrC,CAAC;QACD,IAAI,cAAc,KAAK,cAAc,CAAC,MAAM,EAAE,CAAC;YAC7C,QAAQ,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,qCAAqC;YAE5D,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;YAC7D,IAAI,CAAC,UAAU,EAAE,CAAC;gBAChB,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,aAAa,EAAE;oBAClC,WAAW,EAAE,uBAAuB;oBACpC,IAAI,EAAE,yBAAyB,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,cAAc,EAAE;oBACpE,WAAW,EAAE,IAAI;iBAClB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,cAAc,EAAE,EAAE;YACxE,WAAW,EAAE,IAAI;SAClB,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;QAC3D,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,YAAY,EAAE;gBACjC,WAAW,EAAE,wBAAwB;gBACrC,IAAI,EAAE,eAAe,cAAc,EAAE;aACtC,CAAC,CAAC;QACL,CAAC;IACH,CAAC;;AAxUH,oBAyUC","sourcesContent":["import * as path from \"path\";\nimport type { IConstruct } from \"constructs\";\nimport { Component } from \"../component\";\nimport { NodeProject } from \"../javascript\";\nimport { JsonFile } from \"../json\";\nimport type { Project } from \"../project\";\nimport { normalizePersistedPath } from \"../util\";\nimport { closestProjectMustBe } from \"../util/constructs\";\n\nconst DEFAULT_TEST_REPORTS_DIR = \"test-reports\";\n\n// Pulled from https://jestjs.io/docs/en/configuration\nexport interface JestConfigOptions {\n  /**\n   * This option tells Jest that all imported modules in your tests should be mocked automatically.\n   * All modules used in your tests will have a replacement implementation, keeping the API surface\n   * @default - false\n   */\n  readonly automock?: boolean;\n\n  /**\n   * By default, Jest runs all tests and produces all errors into the console upon completion.\n   * The bail config option can be used here to have Jest stop running tests after n failures.\n   * Setting bail to true is the same as setting bail to 1.\n   * @default - 0\n   */\n  readonly bail?: boolean | number;\n\n  /**\n   * The directory where Jest should store its cached dependency information\n   * @default - \"/tmp/<path>\"\n   */\n  readonly cacheDirectory?: string;\n\n  /**\n   * Automatically clear mock calls and instances before every test.\n   * Equivalent to calling jest.clearAllMocks() before each test.\n   * This does not remove any mock implementation that may have been provided\n   * @default true\n   */\n  readonly clearMocks?: boolean;\n\n  /**\n   * Indicates whether the coverage information should be collected while executing the test.\n   * Because this retrofits all executed files with coverage collection statements,\n   * it may significantly slow down your tests\n   * @default true\n   */\n  readonly collectCoverage?: boolean;\n\n  /**\n   * An array of glob patterns indicating a set of files for which coverage information should be collected.\n   * @default - undefined\n   */\n  readonly collectCoverageFrom?: string[];\n\n  /**\n   * The directory where Jest should output its coverage files.\n   * @default \"coverage\"\n   */\n  readonly coverageDirectory?: string;\n\n  /**\n   * An array of regexp pattern strings that are matched against all file paths before executing the test.\n   * If the file path matches any of the patterns, coverage information will be skipped\n   * @default \"/node_modules/\"\n   */\n  readonly coveragePathIgnorePatterns?: string[];\n\n  /**\n   * Indicates which provider should be used to instrument code for coverage.\n   * Allowed values are v8 (default) or babel\n   * @default - \"v8\"\n   */\n  readonly coverageProvider?: \"babel\" | \"v8\";\n\n  /**\n   * A list of reporter names that Jest uses when writing coverage reports. Any istanbul reporter can be used\n   * @default - [\"json\", \"lcov\", \"text\", \"clover\", \"cobertura\"]\n   */\n  readonly coverageReporters?: string[];\n\n  /**\n   * Specify the global coverage thresholds. This will be used to configure minimum threshold enforcement\n   * for coverage results. Thresholds can be specified as global, as a glob, and as a directory or file path.\n   * If thresholds aren't met, jest will fail.\n   * @default - undefined\n   */\n  readonly coverageThreshold?: CoverageThreshold;\n\n  /**\n   * This option allows the use of a custom dependency extractor.\n   * It must be a node module that exports an object with an extract function\n   * @default - undefined\n   */\n  readonly dependencyExtractor?: string;\n\n  /**\n   * Allows for a label to be printed alongside a test while it is running.\n   * @default - undefined\n   */\n  readonly displayName?: string | any;\n\n  /**\n   * Make calling deprecated APIs throw helpful error messages. Useful for easing the upgrade process.\n   * @default - false\n   */\n  readonly errorOnDeprecated?: boolean;\n\n  /**\n   * Test files run inside a vm, which slows calls to global context properties (e.g. Math).\n   * With this option you can specify extra properties to be defined inside the vm for faster lookups.\n   * @default - undefined\n   */\n  readonly extraGlobals?: string[];\n\n  /**\n   * Test files are normally ignored from collecting code coverage.\n   * With this option, you can overwrite this behavior and include otherwise ignored files in code coverage.\n   * @default - ['']\n   */\n  readonly forceCoverageMatch?: string[];\n\n  /**\n   * A set of global variables that need to be available in all test environments.\n   * @default - {}\n   */\n  readonly globals?: any;\n\n  /**\n   * This option allows the use of a custom global setup module which exports an async function that is\n   * triggered once before all test suites. This function gets Jest's globalConfig object as a parameter.\n   * @default - undefined\n   */\n  readonly globalSetup?: string;\n\n  /**\n   * This option allows the use of a custom global teardown module which exports an async function that is\n   * triggered once after all test suites. This function gets Jest's globalConfig object as a parameter.\n   * @default - undefined\n   */\n  readonly globalTeardown?: string;\n\n  /**\n   * This will be used to configure the behavior of jest-haste-map, Jest's internal file crawler/cache system.\n   * @default - {}\n   */\n  readonly haste?: HasteConfig;\n\n  /**\n   * Insert Jest's globals (expect, test, describe, beforeEach etc.) into the global environment.\n   * If you set this to false, you should import from @jest/globals\n   * @default - true\n   */\n  readonly injectGlobals?: boolean;\n\n  /**\n   * A number limiting the number of tests that are allowed to run at the same time when using test.concurrent.\n   * Any test above this limit will be queued and executed once a slot is released.\n   * @default - 5\n   */\n  readonly maxConcurrency?: number;\n\n  /**\n   * Specifies the maximum number of workers the worker-pool will spawn for running tests. In single run mode,\n   * this defaults to the number of the cores available on your machine minus one for the main thread\n   * In watch mode, this defaults to half of the available cores on your machine.\n   * For environments with variable CPUs available, you can use percentage based configuration: \"maxWorkers\": \"50%\"\n   * @default - the number of the cores available on your machine minus one for the main thread\n   */\n  readonly maxWorkers?: number | string;\n\n  /**\n   * An array of directory names to be searched recursively up from the requiring module's location.\n   * Setting this option will override the default, if you wish to still search node_modules for packages\n   * include it along with any other options: [\"node_modules\", \"bower_components\"]\n   * @default - [\"node_modules\"]\n   */\n  readonly moduleDirectories?: string[];\n\n  /**\n   * An array of file extensions your modules use. If you require modules without specifying a file extension,\n   * these are the extensions Jest will look for, in left-to-right order.\n   * @default - [\"js\", \"json\", \"jsx\", \"ts\", \"tsx\", \"node\"]\n   */\n  readonly moduleFileExtensions?: string[];\n\n  /**\n   * A map from regular expressions to module names or to arrays of module names that allow to stub out\n   * resources, like images or styles with a single module.\n   * @default - null\n   */\n  readonly moduleNameMapper?: { [key: string]: string | string[] };\n\n  /**\n   * An array of regexp pattern strings that are matched against all module paths before those paths are\n   * to be considered 'visible' to the module loader. If a given module's path matches any of the patterns,\n   * it will not be require()-able in the test environment.\n   * @default - []\n   */\n  readonly modulePathIgnorePatterns?: string[];\n\n  /**\n   * An alternative API to setting the NODE_PATH env variable, modulePaths is an array of absolute paths\n   * to additional locations to search when resolving modules. Use the <rootDir> string token to include\n   * the path to your project's root directory. Example: [\"<rootDir>/app/\"].\n   * @default - []\n   */\n  readonly modulePaths?: string[];\n\n  /**\n   * Activates notifications for test results.\n   * @default - false\n   */\n  readonly notify?: boolean;\n\n  /**\n   * Specifies notification mode. Requires notify: true\n   * @default - failure-change\n   */\n  readonly notifyMode?:\n    | \"always\"\n    | \"failure\"\n    | \"success\"\n    | \"change\"\n    | \"success-change\"\n    | \"failure-change\";\n\n  /**\n   * A preset that is used as a base for Jest's configuration. A preset should point to an npm module\n   * that has a jest-preset.json or jest-preset.js file at the root.\n   * @default - undefined\n   */\n  readonly preset?: string;\n\n  /**\n   * Sets the path to the prettier node module used to update inline snapshots.\n   * @default - \"prettier\"\n   */\n  readonly prettierPath?: string;\n\n  /**\n   * When the projects configuration is provided with an array of paths or glob patterns, Jest will\n   * run tests in all of the specified projects at the same time. This is great for monorepos or\n   * when working on multiple projects at the same time.\n   * @default - undefined\n   */\n  readonly projects?: Array<string | { [key: string]: any }>;\n\n  /**\n   * Use this configuration option to add custom reporters to Jest. A custom reporter is a class\n   * that implements onRunStart, onTestStart, onTestResult, onRunComplete methods that will be\n   * called when any of those events occurs.\n   * @default - undefined\n   */\n  readonly reporters?: JestReporter[];\n\n  /**\n   * Automatically reset mock state before every test. Equivalent to calling jest.resetAllMocks()\n   * before each test. This will lead to any mocks having their fake implementations removed but\n   * does not restore their initial implementation.\n   * @default - false\n   */\n  readonly resetMocks?: boolean;\n\n  /**\n   * By default, each test file gets its own independent module registry. Enabling resetModules\n   * goes a step further and resets the module registry before running each individual test.\n   * @default - false\n   */\n  readonly resetModules?: boolean;\n\n  /**\n   * This option allows the use of a custom resolver.\n   * https://jestjs.io/docs/en/configuration#resolver-string\n   * @default - undefined\n   */\n  readonly resolver?: string;\n\n  /**\n   * Automatically restore mock state before every test. Equivalent to calling jest.restoreAllMocks()\n   * before each test. This will lead to any mocks having their fake implementations removed and\n   * restores their initial implementation.\n   * @default - false\n   */\n  readonly restoreMocks?: boolean;\n\n  /**\n   * The root directory that Jest should scan for tests and modules within. If you put your Jest\n   * config inside your package.json and want the root directory to be the root of your repo, the\n   * value for this config param will default to the directory of the package.json.\n   * @default - directory of the package.json\n   */\n  readonly rootDir?: string;\n\n  /**\n   * A list of paths to directories that Jest should use to search for files in.\n   * @default - [\"<rootDir>\"]\n   */\n  readonly roots?: string[];\n\n  /**\n   * This option allows you to use a custom runner instead of Jest's default test runner.\n   * @default - \"jest-runner\"\n   */\n  readonly runner?: string;\n\n  /**\n   * A list of paths to modules that run some code to configure or set up the testing environment.\n   * Each setupFile will be run once per test file. Since every test runs in its own environment,\n   * these scripts will be executed in the testing environment immediately before executing the\n   * test code itself.\n   * @default - []\n   */\n  readonly setupFiles?: string[];\n\n  /**\n   * A list of paths to modules that run some code to configure or set up the testing framework\n   * before each test file in the suite is executed. Since setupFiles executes before the test\n   * framework is installed in the environment, this script file presents you the opportunity of\n   * running some code immediately after the test framework has been installed in the environment.\n   * @default - []\n   */\n  readonly setupFilesAfterEnv?: string[];\n\n  /**\n   * The number of seconds after which a test is considered as slow and reported as such in the results.\n   * @default - 5\n   */\n  readonly slowTestThreshold?: number;\n\n  /**\n   * The path to a module that can resolve test<->snapshot path. This config option lets you customize\n   * where Jest stores snapshot files on disk.\n   * @default - undefined\n   */\n  readonly snapshotResolver?: string;\n\n  /**\n   * A list of paths to snapshot serializer modules Jest should use for snapshot testing.\n   * @default = []\n   */\n  readonly snapshotSerializers?: string[];\n\n  /**\n   * The test environment that will be used for testing. The default environment in Jest is a\n   * browser-like environment through jsdom. If you are building a node service, you can use the node\n   * option to use a node-like environment instead.\n   * @default - \"jsdom\"\n   */\n  readonly testEnvironment?: string;\n\n  /**\n   * Test environment options that will be passed to the testEnvironment.\n   * The relevant options depend on the environment.\n   * @default - {}\n   */\n  readonly testEnvironmentOptions?: any;\n\n  /**\n   * The exit code Jest returns on test failure.\n   * @default - 1\n   */\n  readonly testFailureExitCode?: number;\n\n  /**\n   * The glob patterns Jest uses to detect test files. By default it looks for .js, .jsx, .ts and .tsx\n   * files inside of __tests__ folders, as well as any files with a suffix of .test or .spec\n   * (e.g. Component.test.js or Component.spec.js). It will also find files called test.js or spec.js.\n   * @default ['**\\/__tests__/**\\/*.[jt]s?(x)', '**\\/*(*.)@(spec|test).[tj]s?(x)']\n   */\n  readonly testMatch?: string[];\n\n  /**\n   * An array of regexp pattern strings that are matched against all test paths before executing the test.\n   * If the test path matches any of the patterns, it will be skipped.\n   * @default - [\"/node_modules/\"]\n   */\n  readonly testPathIgnorePatterns?: string[];\n\n  /**\n   * The pattern or patterns Jest uses to detect test files. By default it looks for .js, .jsx, .ts and .tsx\n   * files inside of __tests__ folders, as well as any files with a suffix of .test or .spec\n   * (e.g. Component.test.js or Component.spec.js). It will also find files called test.js or spec.js.\n   * @default - (/__tests__/.*|(\\\\.|/)(test|spec))\\\\.[jt]sx?$\n   */\n  readonly testRegex?: string | string[];\n\n  /**\n   * This option allows the use of a custom results processor.\n   * @default - undefined\n   */\n  readonly testResultsProcessor?: string;\n\n  /**\n   * This option allows the use of a custom test runner. The default is jasmine2. A custom test runner\n   * can be provided by specifying a path to a test runner implementation.\n   * @default - \"jasmine2\"\n   */\n  readonly testRunner?: string;\n\n  /**\n   * This option allows you to use a custom sequencer instead of Jest's default.\n   * Sort may optionally return a Promise.\n   * @default - \"@jest/test-sequencer\"\n   */\n  readonly testSequencer?: string;\n\n  /**\n   * Default timeout of a test in milliseconds.\n   * @default - 5000\n   */\n  readonly testTimeout?: number;\n\n  /**\n   * This option sets the URL for the jsdom environment. It is reflected in properties such as location.href.\n   * @default - \"http://localhost\"\n   */\n  readonly testURL?: string;\n\n  /**\n   * Setting this value to legacy or fake allows the use of fake timers for functions such as setTimeout.\n   * Fake timers are useful when a piece of code sets a long timeout that we don't want to wait for in a test.\n   * @default - \"real\"\n   */\n  readonly timers?: string;\n\n  /**\n   * A map from regular expressions to paths to transformers. A transformer is a module that provides a\n   * synchronous function for transforming source files.\n   * @default - {\"\\\\.[jt]sx?$\": \"babel-jest\"}\n   */\n  readonly transform?: { [key: string]: Transform };\n\n  /**\n   * An array of regexp pattern strings that are matched against all source file paths before transformation.\n   * If the test path matches any of the patterns, it will not be transformed.\n   * @default - [\"/node_modules/\", \"\\\\.pnp\\\\.[^\\\\\\/]+$\"]\n   */\n  readonly transformIgnorePatterns?: string[];\n\n  /**\n   * An array of regexp pattern strings that are matched against all modules before the module loader will\n   * automatically return a mock for them. If a module's path matches any of the patterns in this list, it\n   * will not be automatically mocked by the module loader.\n   * @default - []\n   */\n  readonly unmockedModulePathPatterns?: string[];\n\n  /**\n   * Indicates whether each individual test should be reported during the run. All errors will also\n   * still be shown on the bottom after execution. Note that if there is only one test file being run\n   * it will default to true.\n   * @default - false\n   */\n  readonly verbose?: boolean;\n\n  /**\n   * An array of RegExp patterns that are matched against all source file paths before re-running tests\n   * in watch mode. If the file path matches any of the patterns, when it is updated, it will not trigger\n   * a re-run of tests.\n   * @default - []\n   */\n  readonly watchPathIgnorePatterns?: string[];\n\n  /**\n   *\n   * @default -\n   */\n  readonly watchPlugins?: WatchPlugin[];\n\n  /**\n   * Whether to use watchman for file crawling.\n   * @default - true\n   */\n  readonly watchman?: boolean;\n\n  /**\n   * Escape hatch to allow any value\n   */\n  readonly additionalOptions?: { [name: string]: any };\n\n  /**\n   * Escape hatch to allow any value (JS/TS only)\n   *\n   * @deprecated use `additionalOptions` instead.\n   *\n   * @jsii ignore\n   */\n  readonly [name: string]: any;\n}\n\n/**\n * Options for discoverTestMatchPatternsForDirs.\n */\nexport interface JestDiscoverTestMatchPatternsForDirsOptions {\n  /** The file extension pattern to use. Defaults to \"[jt]s?(x)\". */\n  readonly fileExtensionPattern?: string;\n}\n\nexport class Transform {\n  public constructor(\n    private readonly name: string,\n    private readonly options?: any,\n  ) {}\n\n  /**\n   * @jsii ignore\n   * @internal\n   */\n  public toJSON() {\n    if (this.options != null) {\n      return [this.name, this.options];\n    }\n    return this.name;\n  }\n}\n\nexport class WatchPlugin {\n  public constructor(\n    private readonly name: string,\n    private readonly options?: any,\n  ) {}\n\n  /**\n   * @jsii ignore\n   * @internal\n   */\n  public toJSON() {\n    if (this.options != null) {\n      return [this.name, this.options];\n    }\n    return this.name;\n  }\n}\n\nexport interface JestOptions {\n  /**\n   * Collect coverage. Deprecated\n   * @default true\n   * @deprecated use jestConfig.collectCoverage\n   */\n  readonly coverage?: boolean;\n\n  /**\n   * Include the `text` coverage reporter, which means that coverage summary is printed\n   * at the end of the jest execution.\n   *\n   * @default true\n   */\n  readonly coverageText?: boolean;\n\n  /**\n   * Defines `testPathIgnorePatterns` and `coveragePathIgnorePatterns`\n   * @default [\"