From 928179a4fcb1fdfe457381204beb1037fe7c9dc1 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Mon, 20 Jun 2022 12:51:53 +0300 Subject: [PATCH 01/13] test_runner: support timeout for tests --- doc/api/test.md | 6 ++++++ lib/internal/test_runner/test.js | 30 +++++++++++++++++++++++++---- test/message/test_runner_output.js | 10 ++++++++++ test/message/test_runner_output.out | 24 +++++++++++++++++++---- 4 files changed, 62 insertions(+), 8 deletions(-) diff --git a/doc/api/test.md b/doc/api/test.md index b1068411ec600f..96698d4503d087 100644 --- a/doc/api/test.md +++ b/doc/api/test.md @@ -339,6 +339,9 @@ added: v18.0.0 * `todo` {boolean|string} If truthy, the test marked as `TODO`. If a string is provided, that string is displayed in the test results as the reason why the test is `TODO`. **Default:** `false`. + * `timeout` {number} A number of milliseconds the test will fail after. + If unspecified, subtests inherit this value from their parent. + **Default:** `30_000`. * `fn` {Function|AsyncFunction} The function under test. The first argument to this function is a [`TestContext`][] object. If the test uses callbacks, the callback function is passed as the second argument. **Default:** A no-op @@ -538,6 +541,9 @@ added: v18.0.0 * `todo` {boolean|string} If truthy, the test marked as `TODO`. If a string is provided, that string is displayed in the test results as the reason why the test is `TODO`. **Default:** `false`. + * `timeout` {number} A number of milliseconds the test will fail after. + If unspecified, subtests inherit this value from their parent. + **Default:** `30_000`. * `fn` {Function|AsyncFunction} The function under test. The first argument to this function is a [`TestContext`][] object. If the test uses callbacks, the callback function is passed as the second argument. **Default:** A no-op diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 0f7c31ec808e28..81e62e20487dad 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -9,6 +9,7 @@ const { PromiseResolve, ReflectApply, SafeMap, + PromiseRace, } = primordials; const { AsyncResource } = require('async_hooks'); const { @@ -33,12 +34,27 @@ const kCancelledByParent = 'cancelledByParent'; const kParentAlreadyFinished = 'parentAlreadyFinished'; const kSubtestsFailed = 'subtestsFailed'; const kTestCodeFailure = 'testCodeFailure'; +const kTestTimeoutFailure = 'testTimeoutFailure'; const kDefaultIndent = ' '; const noop = FunctionPrototype; const isTestRunner = getOptionValue('--test'); const testOnlyFlag = !isTestRunner && getOptionValue('--test-only'); // TODO(cjihrig): Use uv_available_parallelism() once it lands. const rootConcurrency = isTestRunner ? cpus().length : 1; +const rootTimeout = 30_0000; + + +function PromiseTimeout(promise, timeout) { + const deferred = createDeferredPromise(); + setTimeout(() => deferred.reject(new ERR_TEST_FAILURE( + `test timed out after ${timeout}ms`, + kTestTimeoutFailure + )), timeout).unref(); + return PromiseRace([ + promise, + deferred.promise, + ]); +} class TestContext { #test; @@ -76,7 +92,7 @@ class Test extends AsyncResource { super('Test'); let { fn, name, parent, skip } = options; - const { concurrency, only, todo } = options; + const { concurrency, only, todo, timeout } = options; if (typeof fn !== 'function') { fn = noop; @@ -98,6 +114,7 @@ class Test extends AsyncResource { this.reporter = new TapStream(); this.runOnlySubtests = this.only; this.testNumber = 0; + this.timeout = rootTimeout; } else { const indent = parent.parent === null ? parent.indent : parent.indent + parent.indentString; @@ -109,12 +126,17 @@ class Test extends AsyncResource { this.reporter = parent.reporter; this.runOnlySubtests = !this.only; this.testNumber = parent.subtests.length + 1; + this.timeout = parent.timeout; } if (isUint32(concurrency) && concurrency !== 0) { this.concurrency = concurrency; } + if (isUint32(timeout) && timeout !== 0) { + this.timeout = timeout; + } + if (testOnlyFlag && !this.only) { skip = '\'only\' option not set'; } @@ -329,13 +351,13 @@ class Test extends AsyncResource { 'passed a callback but also returned a Promise', kCallbackAndPromisePresent )); - await ret; + await PromiseTimeout(ret, this.timeout); } else { - await promise; + await PromiseTimeout(promise, this.timeout); } } else { // This test is synchronous or using Promises. - await ReflectApply(this.runInAsyncScope, this, args); + await PromiseTimeout(ReflectApply(this.runInAsyncScope, this, args), this.timeout); } this.pass(); diff --git a/test/message/test_runner_output.js b/test/message/test_runner_output.js index c586199f0d9d31..341675ee383722 100644 --- a/test/message/test_runner_output.js +++ b/test/message/test_runner_output.js @@ -328,3 +328,13 @@ test('subtest sync throw fails', async (t) => { throw new Error('thrown from subtest sync throw fails at second'); }); }); + +test('timed out test', { timeout: 500 }, async (t) => { + return new Promise((resolve) => { + setTimeout(resolve, 1000) + }); +}); + +test('callback timed out test', { timeout: 500 }, (t, done) => { + setTimeout(done, 1000) +}); \ No newline at end of file diff --git a/test/message/test_runner_output.out b/test/message/test_runner_output.out index bada2fdacae9a9..38964a09429a02 100644 --- a/test/message/test_runner_output.out +++ b/test/message/test_runner_output.out @@ -562,8 +562,24 @@ not ok 56 - subtest sync throw fails error: '2 subtests failed' code: 'ERR_TEST_FAILURE' ... +# Subtest: timed out test +not ok 57 - timed out test + --- + duration_ms: * + failureType: 'testTimeoutFailure' + error: 'test timed out after 500ms' + code: 'ERR_TEST_FAILURE' + ... +# Subtest: callback timed out test +not ok 58 - callback timed out test + --- + duration_ms: * + failureType: 'testTimeoutFailure' + error: 'test timed out after 500ms' + code: 'ERR_TEST_FAILURE' + ... # Subtest: invalid subtest fail -not ok 57 - invalid subtest fail +not ok 59 - invalid subtest fail --- duration_ms: * failureType: 'parentAlreadyFinished' @@ -572,16 +588,16 @@ not ok 57 - invalid subtest fail stack: |- * ... -1..57 +1..59 # Warning: Test "unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "async unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from async unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "immediate throw - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from immediate throw fail" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "immediate reject - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from immediate reject fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "callback called twice in different ticks" generated asynchronous activity after the test ended. This activity created the error "Error [ERR_TEST_FAILURE]: callback invoked multiple times" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "callback async throw after done" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from callback async throw after done" and would have caused the test to fail, but instead triggered an uncaughtException event. -# tests 57 +# tests 59 # pass 24 -# fail 18 +# fail 20 # cancelled 0 # skipped 10 # todo 5 From 56981696034c5745ee2f095633888c6d9b96b187 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Mon, 20 Jun 2022 13:03:22 +0300 Subject: [PATCH 02/13] lintr --- lib/internal/test_runner/test.js | 3 ++- test/message/test_runner_output.js | 6 +++--- 2 files changed, 5 insertions(+), 4 deletions(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 81e62e20487dad..8cc0c8607ab58f 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -27,6 +27,7 @@ const { } = require('internal/util'); const { isPromise } = require('internal/util/types'); const { isUint32 } = require('internal/validators'); +const { setTimeout } = require('timers'); const { cpus } = require('os'); const { bigint: hrtime } = process.hrtime; const kCallbackAndPromisePresent = 'callbackAndPromisePresent'; @@ -51,7 +52,7 @@ function PromiseTimeout(promise, timeout) { kTestTimeoutFailure )), timeout).unref(); return PromiseRace([ - promise, + promise, deferred.promise, ]); } diff --git a/test/message/test_runner_output.js b/test/message/test_runner_output.js index 341675ee383722..da73674d9460b4 100644 --- a/test/message/test_runner_output.js +++ b/test/message/test_runner_output.js @@ -331,10 +331,10 @@ test('subtest sync throw fails', async (t) => { test('timed out test', { timeout: 500 }, async (t) => { return new Promise((resolve) => { - setTimeout(resolve, 1000) + setTimeout(resolve, 1000); }); }); test('callback timed out test', { timeout: 500 }, (t, done) => { - setTimeout(done, 1000) -}); \ No newline at end of file + setTimeout(done, 1000); +}); From 391d302fd4b9eebc8cafc7bfd8d3eccdeffc9f92 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Mon, 20 Jun 2022 15:17:46 +0300 Subject: [PATCH 03/13] CR --- doc/api/test.md | 8 ++++++++ lib/internal/test_runner/test.js | 14 +++++++------- 2 files changed, 15 insertions(+), 7 deletions(-) diff --git a/doc/api/test.md b/doc/api/test.md index 96698d4503d087..8b0ef257486998 100644 --- a/doc/api/test.md +++ b/doc/api/test.md @@ -320,6 +320,10 @@ internally. * `name` {string} The name of the test, which is displayed when reporting test @@ -522,6 +526,10 @@ test('top level test', (t) => { * `name` {string} The name of the subtest, which is displayed when reporting diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 8cc0c8607ab58f..b46a76a4d55947 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -27,7 +27,7 @@ const { } = require('internal/util'); const { isPromise } = require('internal/util/types'); const { isUint32 } = require('internal/validators'); -const { setTimeout } = require('timers'); +const { setTimeout } = require('timers/promises'); const { cpus } = require('os'); const { bigint: hrtime } = process.hrtime; const kCallbackAndPromisePresent = 'callbackAndPromisePresent'; @@ -46,14 +46,14 @@ const rootTimeout = 30_0000; function PromiseTimeout(promise, timeout) { - const deferred = createDeferredPromise(); - setTimeout(() => deferred.reject(new ERR_TEST_FAILURE( - `test timed out after ${timeout}ms`, - kTestTimeoutFailure - )), timeout).unref(); return PromiseRace([ promise, - deferred.promise, + setTimeout(timeout, null, { ref: false }).then(() => { + throw new ERR_TEST_FAILURE( + `test timed out after ${timeout}ms`, + kTestTimeoutFailure + ); + }), ]); } From 79c33fa1f6970d34cdac5812450721d17060b55d Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Mon, 20 Jun 2022 15:32:05 +0300 Subject: [PATCH 04/13] linter --- doc/api/test.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/doc/api/test.md b/doc/api/test.md index 8b0ef257486998..5970106e11da3e 100644 --- a/doc/api/test.md +++ b/doc/api/test.md @@ -323,7 +323,7 @@ added: v18.0.0 changes: - version: REPLACEME pr-url: https://github.com/nodejs/node/pull/43505 - description: add a timeout to tests and allow setting it in options + description: add a timeout to tests and allow setting it in options. --> * `name` {string} The name of the test, which is displayed when reporting test @@ -529,7 +529,7 @@ added: v18.0.0 changes: - version: REPLACEME pr-url: https://github.com/nodejs/node/pull/43505 - description: add a timeout to tests and allow setting it in options + description: add a timeout to tests and allow setting it in options. --> * `name` {string} The name of the subtest, which is displayed when reporting From d9acf5e355de7c3e7dcf800cf048fb77b7577f13 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Tue, 21 Jun 2022 07:03:29 +0300 Subject: [PATCH 05/13] more tests --- test/message/test_runner_output.js | 15 ++++++++++++-- test/message/test_runner_output.out | 32 +++++++++++++++++++---------- 2 files changed, 34 insertions(+), 13 deletions(-) diff --git a/test/message/test_runner_output.js b/test/message/test_runner_output.js index da73674d9460b4..08034b8a2420d9 100644 --- a/test/message/test_runner_output.js +++ b/test/message/test_runner_output.js @@ -329,12 +329,23 @@ test('subtest sync throw fails', async (t) => { }); }); -test('timed out test', { timeout: 500 }, async (t) => { +test('timed out async test', { timeout: 5 }, async (t) => { return new Promise((resolve) => { setTimeout(resolve, 1000); }); }); -test('callback timed out test', { timeout: 500 }, (t, done) => { +test('timed out callback test', { timeout: 5 }, (t, done) => { setTimeout(done, 1000); }); + + +test('large timeout async test is ok', { timeout: Number.MAX_VALUE }, async (t) => { + return new Promise((resolve) => { + setTimeout(resolve, 10); + }); +}); + +test('large timeout callback test is ok', { timeout: Number.MAX_VALUE }, (t, done) => { + setTimeout(done, 10); +}); \ No newline at end of file diff --git a/test/message/test_runner_output.out b/test/message/test_runner_output.out index 38964a09429a02..2a0381ca6b1d4c 100644 --- a/test/message/test_runner_output.out +++ b/test/message/test_runner_output.out @@ -562,41 +562,51 @@ not ok 56 - subtest sync throw fails error: '2 subtests failed' code: 'ERR_TEST_FAILURE' ... -# Subtest: timed out test -not ok 57 - timed out test +# Subtest: timed out async test +not ok 57 - timed out async test --- duration_ms: * failureType: 'testTimeoutFailure' - error: 'test timed out after 500ms' + error: 'test timed out after 5ms' code: 'ERR_TEST_FAILURE' ... -# Subtest: callback timed out test -not ok 58 - callback timed out test +# Subtest: timed out callback test +not ok 58 - timed out callback test --- duration_ms: * failureType: 'testTimeoutFailure' - error: 'test timed out after 500ms' + error: 'test timed out after 5ms' code: 'ERR_TEST_FAILURE' ... +# Subtest: large timeout async test is ok +ok 59 - large timeout async test is ok + --- + duration_ms: * + ... +# Subtest: large timeout callback test is ok +ok 60 - large timeout callback test is ok + --- + duration_ms: * + ... # Subtest: invalid subtest fail -not ok 59 - invalid subtest fail +not ok 61 - invalid subtest fail --- duration_ms: * failureType: 'parentAlreadyFinished' error: 'test could not be started because its parent finished' code: 'ERR_TEST_FAILURE' stack: |- - * + Immediate. (/Users/mosheatlow/repos/node/test/message/test_runner_output.js:173:7) ... -1..59 +1..61 # Warning: Test "unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "async unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from async unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "immediate throw - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from immediate throw fail" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "immediate reject - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from immediate reject fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "callback called twice in different ticks" generated asynchronous activity after the test ended. This activity created the error "Error [ERR_TEST_FAILURE]: callback invoked multiple times" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "callback async throw after done" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from callback async throw after done" and would have caused the test to fail, but instead triggered an uncaughtException event. -# tests 59 -# pass 24 +# tests 61 +# pass 26 # fail 20 # cancelled 0 # skipped 10 From 554d2b2929f45e4517b84b912591852998d6e6d8 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Tue, 21 Jun 2022 07:06:56 +0300 Subject: [PATCH 06/13] cr --- lib/internal/test_runner/test.js | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index b46a76a4d55947..352e4fd1056e37 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -37,15 +37,15 @@ const kSubtestsFailed = 'subtestsFailed'; const kTestCodeFailure = 'testCodeFailure'; const kTestTimeoutFailure = 'testTimeoutFailure'; const kDefaultIndent = ' '; +const kDefaultTimeout = 30_0000; const noop = FunctionPrototype; const isTestRunner = getOptionValue('--test'); const testOnlyFlag = !isTestRunner && getOptionValue('--test-only'); // TODO(cjihrig): Use uv_available_parallelism() once it lands. const rootConcurrency = isTestRunner ? cpus().length : 1; -const rootTimeout = 30_0000; -function PromiseTimeout(promise, timeout) { +function testTimeout(promise, timeout) { return PromiseRace([ promise, setTimeout(timeout, null, { ref: false }).then(() => { @@ -93,7 +93,7 @@ class Test extends AsyncResource { super('Test'); let { fn, name, parent, skip } = options; - const { concurrency, only, todo, timeout } = options; + const { concurrency, only, timeout, todo } = options; if (typeof fn !== 'function') { fn = noop; @@ -115,7 +115,7 @@ class Test extends AsyncResource { this.reporter = new TapStream(); this.runOnlySubtests = this.only; this.testNumber = 0; - this.timeout = rootTimeout; + this.timeout = kDefaultTimeout; } else { const indent = parent.parent === null ? parent.indent : parent.indent + parent.indentString; @@ -134,7 +134,7 @@ class Test extends AsyncResource { this.concurrency = concurrency; } - if (isUint32(timeout) && timeout !== 0) { + if (isUint32(timeout)) { this.timeout = timeout; } @@ -352,13 +352,13 @@ class Test extends AsyncResource { 'passed a callback but also returned a Promise', kCallbackAndPromisePresent )); - await PromiseTimeout(ret, this.timeout); + await testTimeout(ret, this.timeout); } else { - await PromiseTimeout(promise, this.timeout); + await testTimeout(promise, this.timeout); } } else { // This test is synchronous or using Promises. - await PromiseTimeout(ReflectApply(this.runInAsyncScope, this, args), this.timeout); + await testTimeout(ReflectApply(this.runInAsyncScope, this, args), this.timeout); } this.pass(); From 88ab08ef5918a8e519fef0774307a851753b4209 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Tue, 21 Jun 2022 08:26:47 +0300 Subject: [PATCH 07/13] lint --- test/message/test_runner_output.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/message/test_runner_output.js b/test/message/test_runner_output.js index 08034b8a2420d9..e8c00227e1da82 100644 --- a/test/message/test_runner_output.js +++ b/test/message/test_runner_output.js @@ -347,5 +347,5 @@ test('large timeout async test is ok', { timeout: Number.MAX_VALUE }, async (t) }); test('large timeout callback test is ok', { timeout: Number.MAX_VALUE }, (t, done) => { - setTimeout(done, 10); -}); \ No newline at end of file + setTimeout(done, 10); +}); From d02fc1f26bd7c91a7f87ebcd7d06b8e9b5b632ec Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Tue, 21 Jun 2022 08:32:11 +0300 Subject: [PATCH 08/13] whoops --- test/message/test_runner_output.out | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/message/test_runner_output.out b/test/message/test_runner_output.out index 2a0381ca6b1d4c..22d97940a36f9c 100644 --- a/test/message/test_runner_output.out +++ b/test/message/test_runner_output.out @@ -596,7 +596,7 @@ not ok 61 - invalid subtest fail error: 'test could not be started because its parent finished' code: 'ERR_TEST_FAILURE' stack: |- - Immediate. (/Users/mosheatlow/repos/node/test/message/test_runner_output.js:173:7) + * ... 1..61 # Warning: Test "unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. From 0e0a64c39071e40bf06cf18f9cc07b229a44c2c6 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Thu, 23 Jun 2022 20:32:07 +0300 Subject: [PATCH 09/13] CR --- lib/internal/test_runner/test.js | 5 ++++- test/message/test_runner_output.js | 4 ++-- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 352e4fd1056e37..d24dbe515cc37b 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -37,7 +37,7 @@ const kSubtestsFailed = 'subtestsFailed'; const kTestCodeFailure = 'testCodeFailure'; const kTestTimeoutFailure = 'testTimeoutFailure'; const kDefaultIndent = ' '; -const kDefaultTimeout = 30_0000; +const kDefaultTimeout = null; const noop = FunctionPrototype; const isTestRunner = getOptionValue('--test'); const testOnlyFlag = !isTestRunner && getOptionValue('--test-only'); @@ -46,6 +46,9 @@ const rootConcurrency = isTestRunner ? cpus().length : 1; function testTimeout(promise, timeout) { + if (timeout === kDefaultTimeout) { + return promise; + } return PromiseRace([ promise, setTimeout(timeout, null, { ref: false }).then(() => { diff --git a/test/message/test_runner_output.js b/test/message/test_runner_output.js index e8c00227e1da82..33e17e6a082ca7 100644 --- a/test/message/test_runner_output.js +++ b/test/message/test_runner_output.js @@ -340,12 +340,12 @@ test('timed out callback test', { timeout: 5 }, (t, done) => { }); -test('large timeout async test is ok', { timeout: Number.MAX_VALUE }, async (t) => { +test('large timeout async test is ok', { timeout: 30_000_000 }, async (t) => { return new Promise((resolve) => { setTimeout(resolve, 10); }); }); -test('large timeout callback test is ok', { timeout: Number.MAX_VALUE }, (t, done) => { +test('large timeout callback test is ok', { timeout: 30_000_000 }, (t, done) => { setTimeout(done, 10); }); From 402fd562316df56902b327e2486bdd2af50e134f Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Wed, 29 Jun 2022 23:14:51 +0300 Subject: [PATCH 10/13] timeout this as well --- lib/internal/test_runner/test.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index d24dbe515cc37b..2ecebde20ce34b 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -506,10 +506,10 @@ class Suite extends Test { this.parent.activeSubtests++; this.startTime = hrtime(); const subtests = this.skipped || this.error ? [] : this.subtests; - await ArrayPrototypeReduce(subtests, async (prev, subtest) => { + await testTimeout(ArrayPrototypeReduce(subtests, async (prev, subtest) => { await prev; await subtest.run(); - }, PromiseResolve()); + }, PromiseResolve()), this.timeout); this.pass(); this.postRun(); } From d327bfff3bc949aed6dfd652eba1fecdddd8a223 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Thu, 30 Jun 2022 08:15:45 +0300 Subject: [PATCH 11/13] this is probably wrong since timeouts inherit --- lib/internal/test_runner/test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 2ecebde20ce34b..6134cff42806c9 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -509,7 +509,7 @@ class Suite extends Test { await testTimeout(ArrayPrototypeReduce(subtests, async (prev, subtest) => { await prev; await subtest.run(); - }, PromiseResolve()), this.timeout); + }, PromiseResolve()); this.pass(); this.postRun(); } From 6c58a500a82323a69fa098e4181004967e4eb5b3 Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Thu, 30 Jun 2022 08:18:18 +0300 Subject: [PATCH 12/13] tests --- test/message/test_runner_describe_it.js | 23 +++++++++++++++ test/message/test_runner_describe_it.out | 37 ++++++++++++++++++++++++ 2 files changed, 60 insertions(+) diff --git a/test/message/test_runner_describe_it.js b/test/message/test_runner_describe_it.js index a8023f8e9bf350..156fecddaf6401 100644 --- a/test/message/test_runner_describe_it.js +++ b/test/message/test_runner_describe_it.js @@ -316,3 +316,26 @@ describe('describe async throw fails', async () => { it('should not run', () => {}); throw new Error('thrown from describe'); }); + +describe('timeouts', () => { + it('timed out async test', { timeout: 5 }, async () => { + return new Promise((resolve) => { + setTimeout(resolve, 1000); + }); + }); + + it('timed out callback test', { timeout: 5 }, (done) => { + setTimeout(done, 1000); + }); + + + it('large timeout async test is ok', { timeout: 30_000_000 }, async () => { + return new Promise((resolve) => { + setTimeout(resolve, 10); + }); + }); + + it('large timeout callback test is ok', { timeout: 30_000_000 }, (done) => { + setTimeout(done, 10); + }); +}); diff --git a/test/message/test_runner_describe_it.out b/test/message/test_runner_describe_it.out index f60553d4b7ff09..9eeecf3c3bdbd6 100644 --- a/test/message/test_runner_describe_it.out +++ b/test/message/test_runner_describe_it.out @@ -553,6 +553,43 @@ not ok 56 - describe async throw fails ... # Subtest: invalid subtest fail not ok 57 - invalid subtest fail +# Subtest: timeouts + # Subtest: timed out async test + not ok 1 - timed out async test + --- + duration_ms: * + failureType: 'testTimeoutFailure' + error: 'test timed out after 5ms' + code: 'ERR_TEST_FAILURE' + ... + # Subtest: timed out callback test + not ok 2 - timed out callback test + --- + duration_ms: * + failureType: 'testTimeoutFailure' + error: 'test timed out after 5ms' + code: 'ERR_TEST_FAILURE' + ... + # Subtest: large timeout async test is ok + ok 3 - large timeout async test is ok + --- + duration_ms: * + ... + # Subtest: large timeout callback test is ok + ok 4 - large timeout callback test is ok + --- + duration_ms: * + ... + 1..4 +not ok 54 - timeouts + --- + duration_ms: * + failureType: 'subtestsFailed' + error: '2 subtests failed' + code: 'ERR_TEST_FAILURE' + ... +# Subtest: invalid subtest fail +not ok 55 - invalid subtest fail --- duration_ms: * failureType: 'parentAlreadyFinished' From 586572a8bf112da013a0eb444c6a994d6b4f7a5b Mon Sep 17 00:00:00 2001 From: Moshe Atlow Date: Wed, 13 Jul 2022 08:17:50 +0300 Subject: [PATCH 13/13] fix rebase --- lib/internal/test_runner/test.js | 2 +- test/message/test_runner_describe_it.out | 12 +++++------- 2 files changed, 6 insertions(+), 8 deletions(-) diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 6134cff42806c9..2ecebde20ce34b 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -509,7 +509,7 @@ class Suite extends Test { await testTimeout(ArrayPrototypeReduce(subtests, async (prev, subtest) => { await prev; await subtest.run(); - }, PromiseResolve()); + }, PromiseResolve()), this.timeout); this.pass(); this.postRun(); } diff --git a/test/message/test_runner_describe_it.out b/test/message/test_runner_describe_it.out index 9eeecf3c3bdbd6..278d542fca2c7d 100644 --- a/test/message/test_runner_describe_it.out +++ b/test/message/test_runner_describe_it.out @@ -551,8 +551,6 @@ not ok 56 - describe async throw fails * * ... -# Subtest: invalid subtest fail -not ok 57 - invalid subtest fail # Subtest: timeouts # Subtest: timed out async test not ok 1 - timed out async test @@ -581,7 +579,7 @@ not ok 57 - invalid subtest fail duration_ms: * ... 1..4 -not ok 54 - timeouts +not ok 57 - timeouts --- duration_ms: * failureType: 'subtestsFailed' @@ -589,7 +587,7 @@ not ok 54 - timeouts code: 'ERR_TEST_FAILURE' ... # Subtest: invalid subtest fail -not ok 55 - invalid subtest fail +not ok 58 - invalid subtest fail --- duration_ms: * failureType: 'parentAlreadyFinished' @@ -598,16 +596,16 @@ not ok 55 - invalid subtest fail stack: |- * ... -1..57 +1..58 # Warning: Test "unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "async unhandled rejection - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from async unhandled rejection fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "immediate throw - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from immediate throw fail" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "immediate reject - passes but warns" generated asynchronous activity after the test ended. This activity created the error "Error: rejected from immediate reject fail" and would have caused the test to fail, but instead triggered an unhandledRejection event. # Warning: Test "callback called twice in different ticks" generated asynchronous activity after the test ended. This activity created the error "Error [ERR_TEST_FAILURE]: callback invoked multiple times" and would have caused the test to fail, but instead triggered an uncaughtException event. # Warning: Test "callback async throw after done" generated asynchronous activity after the test ended. This activity created the error "Error: thrown from callback async throw after done" and would have caused the test to fail, but instead triggered an uncaughtException event. -# tests 57 +# tests 58 # pass 23 -# fail 20 +# fail 21 # cancelled 0 # skipped 9 # todo 5