Skip to content

Commit 561c374

Browse files
committed
test_runner: added coverage support with watch mode
1 parent c25878d commit 561c374

File tree

5 files changed

+109
-12
lines changed

5 files changed

+109
-12
lines changed

lib/internal/test_runner/harness.js

+1
Original file line numberDiff line numberDiff line change
@@ -179,6 +179,7 @@ function setup(root) {
179179
__proto__: null,
180180
bootstrapComplete: false,
181181
coverage: FunctionPrototypeBind(collectCoverage, null, root, coverage),
182+
watchMode: globalOptions.watch,
182183
counters: {
183184
__proto__: null,
184185
all: 0,

lib/internal/test_runner/runner.js

+14-11
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,7 @@ const {
6565
kTestTimeoutFailure,
6666
Test,
6767
} = require('internal/test_runner/test');
68+
let addAbortListener;
6869

6970
const {
7071
convertStringToRegExp,
@@ -79,15 +80,13 @@ const {
7980
exitCodes: { kGenericUserError },
8081
} = internalBinding('errors');
8182

82-
const kFilterArgs = ['--test', '--experimental-test-coverage', '--watch'];
83+
const kFilterArgs = new SafeSet(['--test', '--experimental-test-coverage', '--watch']);
8384
const kFilterArgValues = ['--test-reporter', '--test-reporter-destination'];
8485
const kDiagnosticsFilterArgs = ['tests', 'suites', 'pass', 'fail', 'cancelled', 'skipped', 'todo', 'duration_ms'];
8586

8687
const kCanceledTests = new SafeSet()
8788
.add(kCancelledByParent).add(kAborted).add(kTestTimeoutFailure);
8889

89-
let kResistStopPropagation;
90-
9190
function createTestFileList() {
9291
const cwd = process.cwd();
9392
const hasUserSuppliedPattern = process.argv.length > 1;
@@ -108,7 +107,7 @@ function createTestFileList() {
108107
}
109108

110109
function filterExecArgv(arg, i, arr) {
111-
return !ArrayPrototypeIncludes(kFilterArgs, arg) &&
110+
return !kFilterArgs.has(arg) &&
112111
!ArrayPrototypeSome(kFilterArgValues, (p) => arg === p || (i > 0 && arr[i - 1] === p) || StringPrototypeStartsWith(arg, `${p}=`));
113112
}
114113

@@ -425,12 +424,12 @@ function watchFiles(testFiles, opts) {
425424
}));
426425
});
427426
if (opts.signal) {
428-
kResistStopPropagation ??= require('internal/event_target').kResistStopPropagation;
429-
opts.signal.addEventListener(
430-
'abort',
431-
() => opts.root.postRun(),
432-
{ __proto__: null, once: true, [kResistStopPropagation]: true },
433-
);
427+
addAbortListener ??= require('events').addAbortListener;
428+
addAbortListener(opts.signal, () => {
429+
opts.root.postRun();
430+
opts.root.reporter.end();
431+
});
432+
434433
}
435434

436435
return filesWatcher;
@@ -503,8 +502,12 @@ function run(options = kEmptyObject) {
503502
let filesWatcher;
504503
const opts = { __proto__: null, root, signal, inspectPort, testNamePatterns, only };
505504
if (watch) {
505+
root.harness.watchMode = true;
506506
filesWatcher = watchFiles(testFiles, opts);
507-
postRun = undefined;
507+
postRun = () => {
508+
kFilterArgs.delete('--experimental-test-coverage');
509+
root.postRun();
510+
};
508511
}
509512
const runFiles = () => {
510513
root.harness.bootstrapComplete = true;

lib/internal/test_runner/test.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -758,7 +758,9 @@ class Test extends AsyncResource {
758758
reporter.coverage(nesting, loc, coverage);
759759
}
760760

761-
reporter.end();
761+
if (harness.watchMode === false) {
762+
reporter.end();
763+
}
762764
}
763765
}
764766

lib/internal/test_runner/utils.js

+2
Original file line numberDiff line numberDiff line change
@@ -193,6 +193,7 @@ function parseCommandLine() {
193193

194194
const isTestRunner = getOptionValue('--test');
195195
const coverage = getOptionValue('--experimental-test-coverage');
196+
const watch = getOptionValue('--watch');
196197
const isChildProcess = process.env.NODE_TEST_CONTEXT === 'child';
197198
const isChildProcessV8 = process.env.NODE_TEST_CONTEXT === 'child-v8';
198199
let destinations;
@@ -244,6 +245,7 @@ function parseCommandLine() {
244245
__proto__: null,
245246
isTestRunner,
246247
coverage,
248+
watch,
247249
testOnlyFlag,
248250
testNamePatterns,
249251
reporters,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
// Flags: --expose-internals
2+
import * as common from '../common/index.mjs';
3+
import { test } from 'node:test';
4+
import { spawn } from 'node:child_process';
5+
import { writeFileSync } from 'node:fs';
6+
import util from 'internal/util';
7+
import tmpdir from '../common/tmpdir.js';
8+
import assert from 'node:assert';
9+
10+
const skipIfNoInspector = {
11+
skip: !process.features.inspector ? 'inspector disabled' : false
12+
};
13+
14+
if (common.isIBMi)
15+
common.skip('IBMi does not support `fs.watch()`');
16+
17+
function getCoverageFixtureReport() {
18+
const report = [
19+
'# start of coverage report',
20+
'# ---------------------------------------------------------------',
21+
'# file | line % | branch % | funcs % | uncovered lines',
22+
'# ---------------------------------------------------------------',
23+
'# dependency.js | 100.00 | 100.00 | 100.00 | ',
24+
'# dependency.mjs | 100.00 | 100.00 | 100.00 | ',
25+
'# test.js | 100.00 | 100.00 | 100.00 | ',
26+
'# ---------------------------------------------------------------',
27+
'# all files | 100.00 | 100.00 | 100.00 |',
28+
'# ---------------------------------------------------------------',
29+
'# end of coverage report',
30+
].join('\n');
31+
32+
if (common.isWindows) {
33+
return report.replaceAll('/', '\\');
34+
}
35+
36+
return report;
37+
}
38+
39+
const fixtureContent = {
40+
'dependency.js': 'module.exports = {};',
41+
'dependency.mjs': 'export const a = 1;',
42+
'test.js': `
43+
const test = require('node:test');
44+
require('./dependency.js');
45+
import('./dependency.mjs');
46+
import('data:text/javascript,');
47+
test('test has ran');`,
48+
};
49+
50+
tmpdir.refresh();
51+
52+
const fixturePaths = Object.keys(fixtureContent)
53+
.reduce((acc, file) => ({ ...acc, [file]: tmpdir.resolve(file) }), {});
54+
Object.entries(fixtureContent)
55+
.forEach(([file, content]) => writeFileSync(fixturePaths[file], content));
56+
57+
async function testWatch({ fileToUpdate, file }) {
58+
const ran1 = util.createDeferredPromise();
59+
const ran2 = util.createDeferredPromise();
60+
const args = [ '--test', '--watch', '--experimental-test-coverage',
61+
'--test-reporter', 'tap', file ? fixturePaths[file] : undefined];
62+
const child = spawn(process.execPath,
63+
args.filter(Boolean),
64+
{ encoding: 'utf8', stdio: 'pipe', cwd: tmpdir.path });
65+
let stdout = '';
66+
67+
child.stdout.on('data', (data) => {
68+
stdout += data.toString();
69+
const testRuns = stdout.match(/ - test has ran/g);
70+
if (testRuns?.length >= 1) ran1.resolve();
71+
if (testRuns?.length >= 2) ran2.resolve();
72+
});
73+
74+
await ran1.promise;
75+
const content = fixtureContent[fileToUpdate];
76+
const path = fixturePaths[fileToUpdate];
77+
const interval = setInterval(() => writeFileSync(path, content), common.platformTimeout(1000));
78+
await ran2.promise;
79+
clearInterval(interval);
80+
child.kill();
81+
82+
return stdout;
83+
}
84+
85+
test('should report coverage report with watch mode', skipIfNoInspector, async () => {
86+
const stdout = await testWatch({ file: 'test.js', fileToUpdate: 'test.js' });
87+
const expectedCoverageReport = getCoverageFixtureReport();
88+
assert(stdout.includes(expectedCoverageReport));
89+
});

0 commit comments

Comments
 (0)