From 959df1b722914f48bd4a9c2d8efbc740606739a3 Mon Sep 17 00:00:00 2001 From: Colin Ihrig Date: Sun, 4 Aug 2024 11:56:04 -0400 Subject: [PATCH] test_runner: run after hooks even if test is aborted If a test is run, but aborted, any after hooks should still be run, as they may need to perform cleanup. PR-URL: https://github.com/nodejs/node/pull/54151 Reviewed-By: Chemi Atlow Reviewed-By: Benjamin Gruenbaum Reviewed-By: Moshe Atlow Reviewed-By: Jake Yuesong Li Reviewed-By: Matteo Collina Reviewed-By: Raz Luvaton --- lib/internal/test_runner/test.js | 10 ++------ .../output/abort-runs-after-hook.js | 14 +++++++++++ .../output/abort-runs-after-hook.snapshot | 23 +++++++++++++++++++ test/parallel/test-runner-output.mjs | 1 + 4 files changed, 40 insertions(+), 8 deletions(-) create mode 100644 test/fixtures/test-runner/output/abort-runs-after-hook.js create mode 100644 test/fixtures/test-runner/output/abort-runs-after-hook.snapshot diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 58490ecac84b2c..1549b81f6859ce 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -694,10 +694,7 @@ class Test extends AsyncResource { } [kShouldAbort]() { - if (this.signal.aborted) { - return true; - } - if (this.outerSignal?.aborted) { + if (this.signal.aborted || this.outerSignal?.aborted) { this.#abortHandler(); return true; } @@ -790,10 +787,7 @@ class Test extends AsyncResource { await SafePromiseRace([PromiseResolve(promise), stopPromise]); } - if (this[kShouldAbort]()) { - this.postRun(); - return; - } + this[kShouldAbort](); this.plan?.check(); this.pass(); await afterEach(); diff --git a/test/fixtures/test-runner/output/abort-runs-after-hook.js b/test/fixtures/test-runner/output/abort-runs-after-hook.js new file mode 100644 index 00000000000000..693ea535ca60ba --- /dev/null +++ b/test/fixtures/test-runner/output/abort-runs-after-hook.js @@ -0,0 +1,14 @@ +'use strict'; +const { test } = require('node:test'); + +test('test that aborts', (t, done) => { + t.after(() => { + // This should still run. + console.log('AFTER'); + }); + + setImmediate(() => { + // This creates an uncaughtException, which aborts the test. + throw new Error('boom'); + }); +}); diff --git a/test/fixtures/test-runner/output/abort-runs-after-hook.snapshot b/test/fixtures/test-runner/output/abort-runs-after-hook.snapshot new file mode 100644 index 00000000000000..c734a264b0f07f --- /dev/null +++ b/test/fixtures/test-runner/output/abort-runs-after-hook.snapshot @@ -0,0 +1,23 @@ +TAP version 13 +AFTER +# Subtest: test that aborts +not ok 1 - test that aborts + --- + duration_ms: * + location: '/test/fixtures/test-runner/output/abort-runs-after-hook.js:(LINE):1' + failureType: 'uncaughtException' + error: 'boom' + code: 'ERR_TEST_FAILURE' + stack: |- + * + * + ... +1..1 +# tests 1 +# suites 0 +# pass 0 +# fail 1 +# cancelled 0 +# skipped 0 +# todo 0 +# duration_ms * diff --git a/test/parallel/test-runner-output.mjs b/test/parallel/test-runner-output.mjs index 774c94bda6f55b..9fc6a6bfb44e9a 100644 --- a/test/parallel/test-runner-output.mjs +++ b/test/parallel/test-runner-output.mjs @@ -93,6 +93,7 @@ const lcovTransform = snapshot.transform( const tests = [ { name: 'test-runner/output/abort.js' }, + { name: 'test-runner/output/abort-runs-after-hook.js' }, { name: 'test-runner/output/abort_suite.js' }, { name: 'test-runner/output/abort_hooks.js' }, { name: 'test-runner/output/describe_it.js' },