diff --git a/doc/api/cli.md b/doc/api/cli.md index 01efb97e31be23..4028e0d374b179 100644 --- a/doc/api/cli.md +++ b/doc/api/cli.md @@ -2336,7 +2336,7 @@ changes: --> Configures the test runner to only execute top level tests that have the `only` -option set. +option set. This flag is not necessary when test isolation is disabled. ### `--test-reporter` diff --git a/doc/api/test.md b/doc/api/test.md index 19819202d1d2e1..82cba9842aad34 100644 --- a/doc/api/test.md +++ b/doc/api/test.md @@ -229,10 +229,10 @@ const { describe, it } = require('node:test'); ## `only` tests -If Node.js is started with the [`--test-only`][] command-line option, it is -possible to skip all tests except for a selected subset by passing -the `only` option to the tests that should run. When a test with the `only` -option is set, all subtests are also run. +If Node.js is started with the [`--test-only`][] command-line option, or test +isolation is disabled, it is possible to skip all tests except for a selected +subset by passing the `only` option to the tests that should run. When a test +with the `only` option is set, all subtests are also run. If a suite has the `only` option set, all tests within the suite are run, unless it has descendants with the `only` option set, in which case only those tests are run. diff --git a/lib/internal/test_runner/harness.js b/lib/internal/test_runner/harness.js index 34ebbf50c658c1..50ecb290bfa65a 100644 --- a/lib/internal/test_runner/harness.js +++ b/lib/internal/test_runner/harness.js @@ -36,6 +36,10 @@ testResources.set(reporterScope.asyncId(), reporterScope); function createTestTree(rootTestOptions, globalOptions) { const buildPhaseDeferred = createDeferredPromise(); + const isFilteringByName = globalOptions.testNamePatterns || + globalOptions.testSkipPatterns; + const isFilteringByOnly = globalOptions.only || (globalOptions.isTestRunner && + globalOptions.isolation === 'none'); const harness = { __proto__: null, buildPromise: buildPhaseDeferred.promise, @@ -62,6 +66,8 @@ function createTestTree(rootTestOptions, globalOptions) { shouldColorizeTestFiles: shouldColorizeTestFiles(globalOptions.destinations), teardown: null, snapshotManager: null, + isFilteringByName, + isFilteringByOnly, async waitForBuildPhase() { if (harness.buildSuites.length > 0) { await SafePromiseAllReturnVoid(harness.buildSuites); diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 26a65eda67dd69..bfb0712f633b1b 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -395,8 +395,10 @@ class Test extends AsyncResource { this.parent = parent; this.testNumber = 0; this.outputSubtestCount = 0; - this.filteredSubtestCount = 0; + this.diagnostics = []; this.filtered = false; + this.filteredByName = false; + this.hasOnlyTests = false; if (parent === null) { this.root = this; @@ -413,26 +415,48 @@ class Test extends AsyncResource { } else { const nesting = parent.parent === null ? parent.nesting : parent.nesting + 1; + const { config, isFilteringByName, isFilteringByOnly } = parent.root.harness; this.root = parent.root; this.harness = null; - this.config = this.root.harness.config; + this.config = config; this.concurrency = parent.concurrency; this.nesting = nesting; - this.only = only ?? (parent.only && !parent.runOnlySubtests); + this.only = only; this.reporter = parent.reporter; this.runOnlySubtests = false; this.childNumber = parent.subtests.length + 1; this.timeout = parent.timeout; this.entryFile = parent.entryFile; - if (this.willBeFiltered()) { - this.filtered = true; - this.parent.filteredSubtestCount++; + if (isFilteringByName) { + this.filteredByName = this.willBeFilteredByName(); + if (!this.filteredByName) { + for (let t = this.parent; t !== null && t.filteredByName; t = t.parent) { + t.filteredByName = false; + } + } } - if (this.config.only && only === false) { - fn = noop; + if (isFilteringByOnly) { + if (this.only) { + // If filtering impacts the tests within a suite, then the suite only + // runs those tests. If filtering does not impact the tests within a + // suite, then all tests are run. + this.parent.runOnlySubtests = true; + + if (this.parent === this.root || this.parent.startTime === null) { + for (let t = this.parent; t !== null && !t.hasOnlyTests; t = t.parent) { + t.hasOnlyTests = true; + } + } + } else if (this.only === false) { + fn = noop; + } + } else if (only || this.parent.runOnlySubtests) { + const warning = + "'only' and 'runOnly' require the --test-only command-line option."; + this.diagnostic(warning); } } @@ -491,7 +515,6 @@ class Test extends AsyncResource { this.endTime = null; this.passed = false; this.error = null; - this.diagnostics = []; this.message = typeof skip === 'string' ? skip : typeof todo === 'string' ? todo : null; this.activeSubtests = 0; @@ -509,12 +532,6 @@ class Test extends AsyncResource { ownAfterEachCount: 0, }; - if (!this.config.only && (only || this.parent?.runOnlySubtests)) { - const warning = - "'only' and 'runOnly' require the --test-only command-line option."; - this.diagnostic(warning); - } - if (loc === undefined) { this.loc = undefined; } else { @@ -542,9 +559,27 @@ class Test extends AsyncResource { } } - willBeFiltered() { - if (this.config.only && !this.only) return true; + applyFilters() { + if (this.error) { + // Never filter out errors. + return; + } + if (this.filteredByName) { + this.filtered = true; + return; + } + + if (this.root.harness.isFilteringByOnly && !this.only && !this.hasOnlyTests) { + if (this.parent.runOnlySubtests || + this.parent.hasOnlyTests || + this.only === false) { + this.filtered = true; + } + } + } + + willBeFilteredByName() { const { testNamePatterns, testSkipPatterns } = this.config; if (testNamePatterns && !testMatchesPattern(this, testNamePatterns)) { @@ -757,12 +792,10 @@ class Test extends AsyncResource { ArrayPrototypePush(this.diagnostics, message); } - get shouldFilter() { - return this.filtered && this.parent?.filteredSubtestCount > 0; - } - start() { - if (this.shouldFilter) { + this.applyFilters(); + + if (this.filtered) { noopTestStream ??= new TestsStream(); this.reporter = noopTestStream; this.run = this.filteredRun; @@ -970,7 +1003,7 @@ class Test extends AsyncResource { this.mock?.reset(); if (this.parent !== null) { - if (!this.shouldFilter) { + if (!this.filtered) { const report = this.getReportDetails(); report.details.passed = this.passed; this.testNumber ||= ++this.parent.outputSubtestCount; @@ -1159,7 +1192,7 @@ class TestHook extends Test { getRunArgs() { return this.#args; } - willBeFiltered() { + willBeFilteredByName() { return false; } postRun() { @@ -1192,7 +1225,6 @@ class Suite extends Test { this.fn = options.fn || this.fn; this.skipped = false; } - this.runOnlySubtests = this.config.only; try { const { ctx, args } = this.getRunArgs(); @@ -1216,21 +1248,6 @@ class Suite extends Test { postBuild() { this.buildPhaseFinished = true; - if (this.filtered && - (this.filteredSubtestCount !== this.subtests.length || this.error)) { - // A suite can transition from filtered to unfiltered based on the - // tests that it contains - in case of children matching patterns. - this.filtered = false; - this.parent.filteredSubtestCount--; - } else if ( - this.config.only && - this.config.testNamePatterns == null && - this.config.testSkipPatterns == null && - this.filteredSubtestCount === this.subtests.length - ) { - // If no subtests are marked as "only", run them all - this.filteredSubtestCount = 0; - } } getRunArgs() { diff --git a/test/parallel/test-runner-no-isolation-filtering.js b/test/parallel/test-runner-no-isolation-filtering.js index 21db267d5323c6..b99b4783ddcc1c 100644 --- a/test/parallel/test-runner-no-isolation-filtering.js +++ b/test/parallel/test-runner-no-isolation-filtering.js @@ -45,10 +45,8 @@ test('works with --test-name-pattern', () => { assert.strictEqual(child.status, 0); assert.strictEqual(child.signal, null); - assert.match(stdout, /# tests 1/); + assert.match(stdout, /# tests 0/); assert.match(stdout, /# suites 0/); - assert.match(stdout, /# pass 1/); - assert.match(stdout, /ok 1 - test one/); }); test('works with --test-skip-pattern', () => { diff --git a/test/parallel/test-runner-no-isolation-hooks.mjs b/test/parallel/test-runner-no-isolation-hooks.mjs index 2c1570d2d75514..ce7b1388e92425 100644 --- a/test/parallel/test-runner-no-isolation-hooks.mjs +++ b/test/parallel/test-runner-no-isolation-hooks.mjs @@ -30,15 +30,6 @@ const order = [ 'afterEach one: suite one - test', 'afterEach two: suite one - test', - 'beforeEach(): global', - 'beforeEach one: test one', - 'beforeEach two: test one', - 'test one', - - 'afterEach(): global', - 'afterEach one: test one', - 'afterEach two: test one', - 'before suite two: suite two', 'beforeEach(): global', 'beforeEach one: suite two - test',