mirror of
https://github.com/nodejs/node.git
synced 2025-04-29 06:19:07 +00:00

This commit updates the test harness and root test to track when bootstrapping has completed. PR-URL: https://github.com/nodejs/node/pull/46962 Reviewed-By: Moshe Atlow <moshe@atlow.co.il> Reviewed-By: Yagiz Nizipli <yagiz@nizipli.com>
236 lines
6.1 KiB
JavaScript
236 lines
6.1 KiB
JavaScript
'use strict';
|
|
const {
|
|
ArrayPrototypeForEach,
|
|
PromiseResolve,
|
|
SafeMap,
|
|
SafeWeakSet,
|
|
} = primordials;
|
|
const {
|
|
createHook,
|
|
executionAsyncId,
|
|
} = require('async_hooks');
|
|
const {
|
|
codes: {
|
|
ERR_TEST_FAILURE,
|
|
},
|
|
} = require('internal/errors');
|
|
const { exitCodes: { kGenericUserError } } = internalBinding('errors');
|
|
|
|
const { kEmptyObject } = require('internal/util');
|
|
const { kCancelledByParent, Test, Suite } = require('internal/test_runner/test');
|
|
const {
|
|
kAsyncBootstrapFailure,
|
|
parseCommandLine,
|
|
setupTestReporters,
|
|
} = require('internal/test_runner/utils');
|
|
const { bigint: hrtime } = process.hrtime;
|
|
|
|
const testResources = new SafeMap();
|
|
const wasRootSetup = new SafeWeakSet();
|
|
|
|
function createTestTree(options = kEmptyObject) {
|
|
return setup(new Test({ __proto__: null, ...options, name: '<root>' }));
|
|
}
|
|
|
|
function createProcessEventHandler(eventName, rootTest) {
|
|
return (err) => {
|
|
if (err?.failureType === kAsyncBootstrapFailure) {
|
|
// 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.cause;
|
|
}
|
|
|
|
// Check if this error is coming from a test. If it is, fail the test.
|
|
const test = testResources.get(executionAsyncId());
|
|
|
|
if (!test) {
|
|
throw err;
|
|
}
|
|
|
|
if (test.finished) {
|
|
// If the test is already finished, report this as a top level
|
|
// diagnostic since this is a malformed test.
|
|
const msg = `Warning: Test "${test.name}" 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.`;
|
|
|
|
rootTest.diagnostic(msg);
|
|
process.exitCode = kGenericUserError;
|
|
return;
|
|
}
|
|
|
|
test.fail(new ERR_TEST_FAILURE(err, eventName));
|
|
test.postRun();
|
|
};
|
|
}
|
|
|
|
function configureCoverage(rootTest, globalOptions) {
|
|
if (!globalOptions.coverage) {
|
|
return null;
|
|
}
|
|
|
|
const { setupCoverage } = require('internal/test_runner/coverage');
|
|
|
|
try {
|
|
return setupCoverage();
|
|
} catch (err) {
|
|
const msg = `Warning: Code coverage could not be enabled. ${err}`;
|
|
|
|
rootTest.diagnostic(msg);
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
}
|
|
|
|
function collectCoverage(rootTest, coverage) {
|
|
if (!coverage) {
|
|
return null;
|
|
}
|
|
|
|
let summary = null;
|
|
|
|
try {
|
|
summary = coverage.summary();
|
|
coverage.cleanup();
|
|
} catch (err) {
|
|
const op = summary ? 'clean up' : 'report';
|
|
const msg = `Warning: Could not ${op} code coverage. ${err}`;
|
|
|
|
rootTest.diagnostic(msg);
|
|
process.exitCode = kGenericUserError;
|
|
}
|
|
|
|
return summary;
|
|
}
|
|
|
|
function setup(root) {
|
|
if (wasRootSetup.has(root)) {
|
|
return root;
|
|
}
|
|
|
|
// Parse the command line options before the hook is enabled. We don't want
|
|
// global input validation errors to end up in the uncaughtException handler.
|
|
const globalOptions = parseCommandLine();
|
|
|
|
const hook = createHook({
|
|
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 = () => {
|
|
root.harness.coverage = collectCoverage(root, coverage);
|
|
root.postRun(new ERR_TEST_FAILURE(
|
|
'Promise resolution is still pending but the event loop has already resolved',
|
|
kCancelledByParent));
|
|
|
|
hook.disable();
|
|
process.removeListener('unhandledRejection', rejectionHandler);
|
|
process.removeListener('uncaughtException', exceptionHandler);
|
|
};
|
|
|
|
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 = {
|
|
__proto__: null,
|
|
bootstrapComplete: false,
|
|
coverage: null,
|
|
};
|
|
root.startTime = hrtime();
|
|
|
|
wasRootSetup.add(root);
|
|
return root;
|
|
}
|
|
|
|
let globalRoot;
|
|
let reportersSetup;
|
|
function getGlobalRoot() {
|
|
if (!globalRoot) {
|
|
globalRoot = createTestTree();
|
|
globalRoot.reporter.once('test:fail', () => {
|
|
process.exitCode = kGenericUserError;
|
|
});
|
|
reportersSetup = setupTestReporters(globalRoot);
|
|
}
|
|
return globalRoot;
|
|
}
|
|
|
|
async function startSubtest(subtest) {
|
|
await reportersSetup;
|
|
getGlobalRoot().harness.bootstrapComplete = true;
|
|
await subtest.start();
|
|
}
|
|
|
|
function runInParentContext(Factory, addShorthands = true) {
|
|
function run(name, options, fn, overrides) {
|
|
const parent = testResources.get(executionAsyncId()) || getGlobalRoot();
|
|
const subtest = parent.createSubtest(Factory, name, options, fn, overrides);
|
|
if (!(parent instanceof Suite)) {
|
|
return startSubtest(subtest);
|
|
}
|
|
return PromiseResolve();
|
|
}
|
|
|
|
const test = (name, options, fn) => run(name, options, fn);
|
|
if (!addShorthands) {
|
|
return test;
|
|
}
|
|
|
|
ArrayPrototypeForEach(['skip', 'todo', 'only'], (keyword) => {
|
|
test[keyword] = (name, options, fn) => {
|
|
run(name, options, fn, { [keyword]: true });
|
|
};
|
|
});
|
|
return test;
|
|
}
|
|
|
|
function hook(hook) {
|
|
return (fn, options) => {
|
|
const parent = testResources.get(executionAsyncId()) || getGlobalRoot();
|
|
parent.createHook(hook, fn, options);
|
|
};
|
|
}
|
|
|
|
module.exports = {
|
|
createTestTree,
|
|
test: runInParentContext(Test, false),
|
|
describe: runInParentContext(Suite),
|
|
it: runInParentContext(Test),
|
|
before: hook('before'),
|
|
after: hook('after'),
|
|
beforeEach: hook('beforeEach'),
|
|
afterEach: hook('afterEach'),
|
|
};
|