Skip to content

Commit

Permalink
test_runner: cleanup test timeout abort listener
Browse files Browse the repository at this point in the history
  • Loading branch information
rluvaton committed Jul 25, 2023
1 parent a0f3ed8 commit a9255e7
Showing 1 changed file with 29 additions and 14 deletions.
43 changes: 29 additions & 14 deletions lib/internal/test_runner/test.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,6 @@ const {
Symbol,
} = primordials;
const { AsyncResource } = require('async_hooks');
const { once } = require('events');
const { AbortController } = require('internal/abort_controller');
const {
codes: {
Expand Down Expand Up @@ -77,14 +76,23 @@ let kResistStopPropagation;

function stopTest(timeout, signal) {
if (timeout === kDefaultTimeout) {
return once(signal, 'abort');
const deferred = createDeferredPromise();

signal.addEventListener('abort', deferred.resolve, { __proto__: null, once: true });
deferred.cleanup = () => signal.removeEventListener('abort', deferred.resolve);
return deferred;
}
return PromisePrototypeThen(setTimeout(timeout, null, { ref: false, signal }), () => {
throw new ERR_TEST_FAILURE(
`test timed out after ${timeout}ms`,
kTestTimeoutFailure,
);
});

return {
__proto__: null,
promise: PromisePrototypeThen(setTimeout(timeout, null, { ref: false, signal }), () => {
throw new ERR_TEST_FAILURE(
`test timed out after ${timeout}ms`,
kTestTimeoutFailure,
);
}),
cleanup: noop,
};
}

class TestContext {
Expand Down Expand Up @@ -549,14 +557,16 @@ class Test extends AsyncResource {
}
});

let testTimeout;

try {
if (this.parent?.hooks.before.length > 0) {
await this.parent.runHook('before', this.parent.getRunArgs());
}
if (this.parent?.hooks.beforeEach.length > 0) {
await this.parent.runHook('beforeEach', { args, ctx });
}
const stopPromise = stopTest(this.timeout, this.signal);
testTimeout = stopTest(this.timeout, this.signal);
const runArgs = ArrayPrototypeSlice(args);
ArrayPrototypeUnshift(runArgs, this.fn, ctx);

Expand All @@ -572,16 +582,18 @@ class Test extends AsyncResource {
'passed a callback but also returned a Promise',
kCallbackAndPromisePresent,
));
await SafePromiseRace([ret, stopPromise]);
await SafePromiseRace([ret, testTimeout.promise]);
} else {
await SafePromiseRace([PromiseResolve(promise), stopPromise]);
await SafePromiseRace([PromiseResolve(promise), testTimeout.promise]);
}
} else {
// This test is synchronous or using Promises.
const promise = ReflectApply(this.runInAsyncScope, this, runArgs);
await SafePromiseRace([PromiseResolve(promise), stopPromise]);
await SafePromiseRace([PromiseResolve(promise), testTimeout.promise]);
}

testTimeout.cleanup();

if (this[kShouldAbort]()) {
this.postRun();
return;
Expand All @@ -591,6 +603,7 @@ class Test extends AsyncResource {
await after();
this.pass();
} catch (err) {
testTimeout?.cleanup();
try { await afterEach(); } catch { /* test is already failing, let's ignore the error */ }
try { await after(); } catch { /* Ignore error. */ }
if (isTestFailureError(err)) {
Expand Down Expand Up @@ -817,6 +830,7 @@ class Suite extends Test {
async run() {
const hookArgs = this.getRunArgs();

let testTimeout;
try {
this.parent.activeSubtests++;
await this.buildSuite;
Expand All @@ -834,15 +848,16 @@ class Suite extends Test {

await this.runHook('before', hookArgs);

const stopPromise = stopTest(this.timeout, this.signal);
testTimeout = stopTest(this.timeout, this.signal);
const subtests = this.skipped || this.error ? [] : this.subtests;
const promise = SafePromiseAll(subtests, (subtests) => subtests.start());

await SafePromiseRace([promise, stopPromise]);
await SafePromiseRace([promise, testTimeout.promise]);
await this.runHook('after', hookArgs);

this.pass();
} catch (err) {
testTimeout?.cleanup();
if (isTestFailureError(err)) {
this.fail(err);
} else {
Expand Down

0 comments on commit a9255e7

Please sign in to comment.