mirror of
https://github.com/nodejs/node.git
synced 2024-11-21 10:59:27 +00:00
bb8cc65edb
PR-URL: https://github.com/nodejs/node/pull/54836 Reviewed-By: Michaël Zasso <targos@protonmail.com> Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com> Reviewed-By: Luigi Pinca <luigipinca@gmail.com> Reviewed-By: Chemi Atlow <chemi@atlow.co.il> Reviewed-By: Moshe Atlow <moshe@atlow.co.il> Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com> Reviewed-By: Robert Nagy <ronagy@icloud.com> Reviewed-By: Trivikram Kamat <trivikr.dev@gmail.com>
356 lines
10 KiB
JavaScript
356 lines
10 KiB
JavaScript
'use strict';
|
|
const {
|
|
ArrayPrototypeForEach,
|
|
ArrayPrototypePush,
|
|
FunctionPrototypeBind,
|
|
PromiseResolve,
|
|
PromiseWithResolvers,
|
|
SafeMap,
|
|
SafePromiseAllReturnVoid,
|
|
} = primordials;
|
|
const { getCallerLocation } = internalBinding('util');
|
|
const {
|
|
createHook,
|
|
executionAsyncId,
|
|
} = require('async_hooks');
|
|
const { relative } = require('path');
|
|
const {
|
|
codes: {
|
|
ERR_TEST_FAILURE,
|
|
},
|
|
} = require('internal/errors');
|
|
const { exitCodes: { kGenericUserError } } = internalBinding('errors');
|
|
const { kCancelledByParent, Test, Suite } = require('internal/test_runner/test');
|
|
const {
|
|
parseCommandLine,
|
|
reporterScope,
|
|
shouldColorizeTestFiles,
|
|
} = require('internal/test_runner/utils');
|
|
const { queueMicrotask } = require('internal/process/task_queues');
|
|
const { bigint: hrtime } = process.hrtime;
|
|
const resolvedPromise = PromiseResolve();
|
|
const testResources = new SafeMap();
|
|
let globalRoot;
|
|
|
|
testResources.set(reporterScope.asyncId(), reporterScope);
|
|
|
|
function createTestTree(rootTestOptions, globalOptions) {
|
|
const buildPhaseDeferred = PromiseWithResolvers();
|
|
const isFilteringByName = globalOptions.testNamePatterns ||
|
|
globalOptions.testSkipPatterns;
|
|
const isFilteringByOnly = (globalOptions.isolation === 'process' || process.env.NODE_TEST_CONTEXT) ?
|
|
globalOptions.only : true;
|
|
const harness = {
|
|
__proto__: null,
|
|
buildPromise: buildPhaseDeferred.promise,
|
|
buildSuites: [],
|
|
isWaitingForBuildPhase: false,
|
|
bootstrapPromise: resolvedPromise,
|
|
watching: false,
|
|
config: globalOptions,
|
|
coverage: null,
|
|
resetCounters() {
|
|
harness.counters = {
|
|
__proto__: null,
|
|
tests: 0,
|
|
failed: 0,
|
|
passed: 0,
|
|
cancelled: 0,
|
|
skipped: 0,
|
|
todo: 0,
|
|
topLevel: 0,
|
|
suites: 0,
|
|
};
|
|
},
|
|
success: true,
|
|
counters: null,
|
|
shouldColorizeTestFiles: shouldColorizeTestFiles(globalOptions.destinations),
|
|
teardown: null,
|
|
snapshotManager: null,
|
|
isFilteringByName,
|
|
isFilteringByOnly,
|
|
async waitForBuildPhase() {
|
|
if (harness.buildSuites.length > 0) {
|
|
await SafePromiseAllReturnVoid(harness.buildSuites);
|
|
}
|
|
|
|
buildPhaseDeferred.resolve();
|
|
},
|
|
};
|
|
|
|
harness.resetCounters();
|
|
globalRoot = new Test({
|
|
__proto__: null,
|
|
...rootTestOptions,
|
|
harness,
|
|
name: '<root>',
|
|
});
|
|
setupProcessState(globalRoot, globalOptions, harness);
|
|
globalRoot.startTime = hrtime();
|
|
return globalRoot;
|
|
}
|
|
|
|
function createProcessEventHandler(eventName, rootTest) {
|
|
return (err) => {
|
|
if (rootTest.harness.bootstrapPromise) {
|
|
// Something went wrong during the asynchronous portion of bootstrapping
|
|
// the test runner. Since the test runner is not setup properly, we can't
|
|
// do anything but throw the error.
|
|
throw err;
|
|
}
|
|
|
|
const test = testResources.get(executionAsyncId());
|
|
|
|
// Check if this error is coming from a reporter. If it is, throw it.
|
|
if (test === reporterScope) {
|
|
throw err;
|
|
}
|
|
|
|
// Check if this error is coming from a test or test hook. If it is, fail the test.
|
|
if (!test || test.finished || test.hookType) {
|
|
// If the test is already finished or the resource that created the error
|
|
// is not mapped to a Test, report this as a top level diagnostic.
|
|
let msg;
|
|
|
|
if (test) {
|
|
const name = test.hookType ? `Test hook "${test.hookType}"` : `Test "${test.name}"`;
|
|
let locInfo = '';
|
|
if (test.loc) {
|
|
const relPath = relative(rootTest.config.cwd, test.loc.file);
|
|
locInfo = ` at ${relPath}:${test.loc.line}:${test.loc.column}`;
|
|
}
|
|
|
|
msg = `Error: ${name}${locInfo} generated asynchronous ` +
|
|
'activity after the test ended. This activity created the error ' +
|
|
`"${err}" and would have caused the test to fail, but instead ` +
|
|
`triggered an ${eventName} event.`;
|
|
} else {
|
|
msg = 'Error: A resource generated asynchronous activity after ' +
|
|
`the test ended. This activity created the error "${err}" which ` +
|
|
`triggered an ${eventName} event, caught by the test runner.`;
|
|
}
|
|
|
|
rootTest.diagnostic(msg);
|
|
rootTest.harness.success = false;
|
|
process.exitCode = kGenericUserError;
|
|
return;
|
|
}
|
|
|
|
test.fail(new ERR_TEST_FAILURE(err, eventName));
|
|
test.abortController.abort();
|
|
};
|
|
}
|
|
|
|
function configureCoverage(rootTest, globalOptions) {
|
|
if (!globalOptions.coverage) {
|
|
return null;
|
|
}
|
|
|
|
const { setupCoverage } = require('internal/test_runner/coverage');
|
|
|
|
try {
|
|
return setupCoverage(globalOptions);
|
|
} catch (err) {
|
|
const msg = `Warning: Code coverage could not be enabled. ${err}`;
|
|
|
|
rootTest.diagnostic(msg);
|
|
rootTest.harness.success = false;
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
}
|
|
|
|
function collectCoverage(rootTest, coverage) {
|
|
if (!coverage) {
|
|
return null;
|
|
}
|
|
|
|
let summary = null;
|
|
|
|
try {
|
|
summary = coverage.summary();
|
|
} catch (err) {
|
|
rootTest.diagnostic(`Warning: Could not report code coverage. ${err}`);
|
|
rootTest.harness.success = false;
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
|
|
try {
|
|
coverage.cleanup();
|
|
} catch (err) {
|
|
rootTest.diagnostic(`Warning: Could not clean up code coverage. ${err}`);
|
|
rootTest.harness.success = false;
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
|
|
return summary;
|
|
}
|
|
|
|
function setupProcessState(root, globalOptions) {
|
|
const hook = createHook({
|
|
__proto__: null,
|
|
init(asyncId, type, triggerAsyncId, resource) {
|
|
if (resource instanceof Test) {
|
|
testResources.set(asyncId, resource);
|
|
return;
|
|
}
|
|
|
|
const parent = testResources.get(triggerAsyncId);
|
|
|
|
if (parent !== undefined) {
|
|
testResources.set(asyncId, parent);
|
|
}
|
|
},
|
|
destroy(asyncId) {
|
|
testResources.delete(asyncId);
|
|
},
|
|
});
|
|
|
|
hook.enable();
|
|
|
|
const exceptionHandler =
|
|
createProcessEventHandler('uncaughtException', root);
|
|
const rejectionHandler =
|
|
createProcessEventHandler('unhandledRejection', root);
|
|
const coverage = configureCoverage(root, globalOptions);
|
|
const exitHandler = async () => {
|
|
if (root.subtests.length === 0 && (root.hooks.before.length > 0 || root.hooks.after.length > 0)) {
|
|
// Run global before/after hooks in case there are no tests
|
|
await root.run();
|
|
}
|
|
root.postRun(new ERR_TEST_FAILURE(
|
|
'Promise resolution is still pending but the event loop has already resolved',
|
|
kCancelledByParent));
|
|
|
|
hook.disable();
|
|
process.removeListener('uncaughtException', exceptionHandler);
|
|
process.removeListener('unhandledRejection', rejectionHandler);
|
|
process.removeListener('beforeExit', exitHandler);
|
|
if (globalOptions.isTestRunner) {
|
|
process.removeListener('SIGINT', terminationHandler);
|
|
process.removeListener('SIGTERM', terminationHandler);
|
|
}
|
|
};
|
|
|
|
const terminationHandler = () => {
|
|
exitHandler();
|
|
process.exit();
|
|
};
|
|
|
|
process.on('uncaughtException', exceptionHandler);
|
|
process.on('unhandledRejection', rejectionHandler);
|
|
process.on('beforeExit', exitHandler);
|
|
// TODO(MoLow): Make it configurable to hook when isTestRunner === false.
|
|
if (globalOptions.isTestRunner) {
|
|
process.on('SIGINT', terminationHandler);
|
|
process.on('SIGTERM', terminationHandler);
|
|
}
|
|
|
|
root.harness.coverage = FunctionPrototypeBind(collectCoverage, null, root, coverage);
|
|
root.harness.teardown = exitHandler;
|
|
}
|
|
|
|
function lazyBootstrapRoot() {
|
|
if (!globalRoot) {
|
|
// This is where the test runner is bootstrapped when node:test is used
|
|
// without the --test flag or the run() API.
|
|
const entryFile = process.argv?.[1];
|
|
const rootTestOptions = {
|
|
__proto__: null,
|
|
entryFile,
|
|
loc: entryFile ? [1, 1, entryFile] : undefined,
|
|
};
|
|
const globalOptions = parseCommandLine();
|
|
globalOptions.cwd = process.cwd();
|
|
createTestTree(rootTestOptions, globalOptions);
|
|
globalRoot.reporter.on('test:summary', (data) => {
|
|
if (!data.success) {
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
});
|
|
globalRoot.harness.bootstrapPromise = globalOptions.setup(globalRoot.reporter);
|
|
}
|
|
return globalRoot;
|
|
}
|
|
|
|
async function startSubtestAfterBootstrap(subtest) {
|
|
if (subtest.root.harness.buildPromise) {
|
|
if (subtest.root.harness.bootstrapPromise) {
|
|
await subtest.root.harness.bootstrapPromise;
|
|
subtest.root.harness.bootstrapPromise = null;
|
|
}
|
|
|
|
if (subtest.buildSuite) {
|
|
ArrayPrototypePush(subtest.root.harness.buildSuites, subtest.buildSuite);
|
|
}
|
|
|
|
if (!subtest.root.harness.isWaitingForBuildPhase) {
|
|
subtest.root.harness.isWaitingForBuildPhase = true;
|
|
queueMicrotask(() => {
|
|
subtest.root.harness.waitForBuildPhase();
|
|
});
|
|
}
|
|
|
|
await subtest.root.harness.buildPromise;
|
|
subtest.root.harness.buildPromise = null;
|
|
}
|
|
|
|
await subtest.start();
|
|
}
|
|
|
|
function runInParentContext(Factory) {
|
|
function run(name, options, fn, overrides) {
|
|
const parent = testResources.get(executionAsyncId()) || lazyBootstrapRoot();
|
|
const subtest = parent.createSubtest(Factory, name, options, fn, overrides);
|
|
if (parent instanceof Suite) {
|
|
return PromiseResolve();
|
|
}
|
|
|
|
return startSubtestAfterBootstrap(subtest);
|
|
}
|
|
|
|
const test = (name, options, fn) => {
|
|
const overrides = {
|
|
__proto__: null,
|
|
loc: getCallerLocation(),
|
|
};
|
|
|
|
return run(name, options, fn, overrides);
|
|
};
|
|
ArrayPrototypeForEach(['skip', 'todo', 'only'], (keyword) => {
|
|
test[keyword] = (name, options, fn) => {
|
|
const overrides = {
|
|
__proto__: null,
|
|
[keyword]: true,
|
|
loc: getCallerLocation(),
|
|
};
|
|
|
|
return run(name, options, fn, overrides);
|
|
};
|
|
});
|
|
return test;
|
|
}
|
|
|
|
function hook(hook) {
|
|
return (fn, options) => {
|
|
const parent = testResources.get(executionAsyncId()) || lazyBootstrapRoot();
|
|
parent.createHook(hook, fn, {
|
|
__proto__: null,
|
|
...options,
|
|
parent,
|
|
hookType: hook,
|
|
loc: getCallerLocation(),
|
|
});
|
|
};
|
|
}
|
|
|
|
module.exports = {
|
|
createTestTree,
|
|
test: runInParentContext(Test),
|
|
suite: runInParentContext(Suite),
|
|
before: hook('before'),
|
|
after: hook('after'),
|
|
beforeEach: hook('beforeEach'),
|
|
afterEach: hook('afterEach'),
|
|
startSubtestAfterBootstrap,
|
|
};
|