Skip to content

Commit 8c95cb0

Browse files
committed
test_runner: add initial code coverage support
This commit adds code coverage functionality to the node:test module. When node:test is used in conjunction with the new --test-coverage CLI flag, a coverage report is created when the test runner finishes. The coverage summary is forwarded to any test runner reporters so that the display can be customized as desired. This new functionality is compatible with the existing NODE_V8_COVERAGE environment variable as well. There are still several limitations, which will be addressed in subsequent pull requests: - Coverage is only reported for a single process. It is possible to merge coverage reports together. Once this is done, the --test flag will be supported as well. - Source maps are not currently supported. - Excluding specific files or directories from the coverage report is not currently supported. Node core modules and node_modules/ are excluded though. PR-URL: #46017 Reviewed-By: Moshe Atlow <[email protected]> Reviewed-By: Geoffrey Booth <[email protected]>
1 parent f6e402e commit 8c95cb0

File tree

16 files changed

+796
-33
lines changed

16 files changed

+796
-33
lines changed

doc/api/cli.md

+12
Original file line numberDiff line numberDiff line change
@@ -1233,6 +1233,17 @@ Starts the Node.js command line test runner. This flag cannot be combined with
12331233
See the documentation on [running tests from the command line][]
12341234
for more details.
12351235

1236+
### `--test-coverage`
1237+
1238+
<!-- YAML
1239+
added: REPLACEME
1240+
-->
1241+
1242+
When used in conjunction with the `node:test` module, a code coverage report is
1243+
generated as part of the test runner output. If no tests are run, a coverage
1244+
report is not generated. See the documentation on
1245+
[collecting code coverage from tests][] for more details.
1246+
12361247
### `--test-name-pattern`
12371248

12381249
<!-- YAML
@@ -2354,6 +2365,7 @@ done
23542365
[`unhandledRejection`]: process.md#event-unhandledrejection
23552366
[`v8.startupSnapshot` API]: v8.md#startup-snapshot-api
23562367
[`worker_threads.threadId`]: worker_threads.md#workerthreadid
2368+
[collecting code coverage from tests]: test.md#collecting-code-coverage
23572369
[conditional exports]: packages.md#conditional-exports
23582370
[context-aware]: addons.md#context-aware-addons
23592371
[debugger]: debugger.md

doc/api/test.md

+87
Original file line numberDiff line numberDiff line change
@@ -370,6 +370,54 @@ Otherwise, the test is considered to be a failure. Test files must be
370370
executable by Node.js, but are not required to use the `node:test` module
371371
internally.
372372

373+
## Collecting code coverage
374+
375+
When Node.js is started with the [`--test-coverage`][] command-line flag, code
376+
coverage is collected and statistics are reported once all tests have completed.
377+
If the [`NODE_V8_COVERAGE`][] environment variable is used to specify a
378+
code coverage directory, the generated V8 coverage files are written to that
379+
directory. Node.js core modules and files within `node_modules/` directories
380+
are not included in the coverage report. If coverage is enabled, the coverage
381+
report is sent to any [test reporters][] via the `'test:coverage'` event.
382+
383+
Coverage can be disabled on a series of lines using the following
384+
comment syntax:
385+
386+
```js
387+
/* node:coverage disable */
388+
if (anAlwaysFalseCondition) {
389+
// Code in this branch will never be executed, but the lines are ignored for
390+
// coverage purposes. All lines following the 'disable' comment are ignored
391+
// until a corresponding 'enable' comment is encountered.
392+
console.log('this is never executed');
393+
}
394+
/* node:coverage enable */
395+
```
396+
397+
Coverage can also be disabled for a specified number of lines. After the
398+
specified number of lines, coverage will be automatically reenabled. If the
399+
number of lines is not explicitly provided, a single line is ignored.
400+
401+
```js
402+
/* node:coverage ignore next */
403+
if (anAlwaysFalseCondition) { console.log('this is never executed'); }
404+
405+
/* node:coverage ignore next 3 */
406+
if (anAlwaysFalseCondition) {
407+
console.log('this is never executed');
408+
}
409+
```
410+
411+
The test runner's code coverage functionality has the following limitations,
412+
which will be addressed in a future Node.js release:
413+
414+
* Although coverage data is collected for child processes, this information is
415+
not included in the coverage report. Because the command line test runner uses
416+
child processes to execute test files, it cannot be used with `--test-coverage`.
417+
* Source maps are not supported.
418+
* Excluding specific files or directories from the coverage report is not
419+
supported.
420+
373421
## Mocking
374422

375423
The `node:test` module supports mocking during testing via a top-level `mock`
@@ -1249,6 +1297,42 @@ A successful call to [`run()`][] method will return a new {TestsStream}
12491297
object, streaming a series of events representing the execution of the tests.
12501298
`TestsStream` will emit events, in the order of the tests definition
12511299

1300+
### Event: `'test:coverage'`
1301+
1302+
* `data` {Object}
1303+
* `summary` {Object} An object containing the coverage report.
1304+
* `files` {Array} An array of coverage reports for individual files. Each
1305+
report is an object with the following schema:
1306+
* `path` {string} The absolute path of the file.
1307+
* `totalLineCount` {number} The total number of lines.
1308+
* `totalBranchCount` {number} The total number of branches.
1309+
* `totalFunctionCount` {number} The total number of functions.
1310+
* `coveredLineCount` {number} The number of covered lines.
1311+
* `coveredBranchCount` {number} The number of covered branches.
1312+
* `coveredFunctionCount` {number} The number of covered functions.
1313+
* `coveredLinePercent` {number} The percentage of lines covered.
1314+
* `coveredBranchPercent` {number} The percentage of branches covered.
1315+
* `coveredFunctionPercent` {number} The percentage of functions covered.
1316+
* `uncoveredLineNumbers` {Array} An array of integers representing line
1317+
numbers that are uncovered.
1318+
* `totals` {Object} An object containing a summary of coverage for all
1319+
files.
1320+
* `totalLineCount` {number} The total number of lines.
1321+
* `totalBranchCount` {number} The total number of branches.
1322+
* `totalFunctionCount` {number} The total number of functions.
1323+
* `coveredLineCount` {number} The number of covered lines.
1324+
* `coveredBranchCount` {number} The number of covered branches.
1325+
* `coveredFunctionCount` {number} The number of covered functions.
1326+
* `coveredLinePercent` {number} The percentage of lines covered.
1327+
* `coveredBranchPercent` {number} The percentage of branches covered.
1328+
* `coveredFunctionPercent` {number} The percentage of functions covered.
1329+
* `workingDirectory` {string} The working directory when code coverage
1330+
began. This is useful for displaying relative path names in case the tests
1331+
changed the working directory of the Node.js process.
1332+
* `nesting` {number} The nesting level of the test.
1333+
1334+
Emitted when code coverage is enabled and all tests have completed.
1335+
12521336
### Event: `'test:diagnostic'`
12531337

12541338
* `data` {Object}
@@ -1631,6 +1715,7 @@ added:
16311715

16321716
[TAP]: https://testanything.org/
16331717
[`--import`]: cli.md#--importmodule
1718+
[`--test-coverage`]: cli.md#--test-coverage
16341719
[`--test-name-pattern`]: cli.md#--test-name-pattern
16351720
[`--test-only`]: cli.md#--test-only
16361721
[`--test-reporter-destination`]: cli.md#--test-reporter-destination
@@ -1639,6 +1724,7 @@ added:
16391724
[`MockFunctionContext`]: #class-mockfunctioncontext
16401725
[`MockTracker.method`]: #mockmethodobject-methodname-implementation-options
16411726
[`MockTracker`]: #class-mocktracker
1727+
[`NODE_V8_COVERAGE`]: cli.md#node_v8_coveragedir
16421728
[`SuiteContext`]: #class-suitecontext
16431729
[`TestContext`]: #class-testcontext
16441730
[`context.diagnostic`]: #contextdiagnosticmessage
@@ -1649,4 +1735,5 @@ added:
16491735
[describe options]: #describename-options-fn
16501736
[it options]: #testname-options-fn
16511737
[stream.compose]: stream.md#streamcomposestreams
1738+
[test reporters]: #test-reporters
16521739
[test runner execution model]: #test-runner-execution-model

doc/node.1

+3
Original file line numberDiff line numberDiff line change
@@ -391,6 +391,9 @@ Specify the minimum allocation from the OpenSSL secure heap. The default is 2. T
391391
.It Fl -test
392392
Starts the Node.js command line test runner.
393393
.
394+
.It Fl -test-coverage
395+
Enable code coverage in the test runner.
396+
.
394397
.It Fl -test-name-pattern
395398
A regular expression that configures the test runner to only execute tests
396399
whose name matches the provided pattern.

lib/internal/process/pre_execution.js

+13-30
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ const {
2020
exposeInterface,
2121
exposeLazyInterfaces,
2222
defineReplaceableLazyAttribute,
23+
setupCoverageHooks,
2324
} = require('internal/util');
2425

2526
const {
@@ -66,15 +67,7 @@ function prepareExecution(options) {
6667
setupFetch();
6768
setupWebCrypto();
6869
setupCustomEvent();
69-
70-
// Resolve the coverage directory to an absolute path, and
71-
// overwrite process.env so that the original path gets passed
72-
// to child processes even when they switch cwd.
73-
if (process.env.NODE_V8_COVERAGE) {
74-
process.env.NODE_V8_COVERAGE =
75-
setupCoverageHooks(process.env.NODE_V8_COVERAGE);
76-
}
77-
70+
setupCodeCoverage();
7871
setupDebugEnv();
7972
// Process initial diagnostic reporting configuration, if present.
8073
initializeReport();
@@ -304,6 +297,17 @@ function setupWebCrypto() {
304297
}
305298
}
306299

300+
function setupCodeCoverage() {
301+
// Resolve the coverage directory to an absolute path, and
302+
// overwrite process.env so that the original path gets passed
303+
// to child processes even when they switch cwd. Don't do anything if the
304+
// --test-coverage flag is present, as the test runner will handle coverage.
305+
if (process.env.NODE_V8_COVERAGE && !getOptionValue('--test-coverage')) {
306+
process.env.NODE_V8_COVERAGE =
307+
setupCoverageHooks(process.env.NODE_V8_COVERAGE);
308+
}
309+
}
310+
307311
// TODO(daeyeon): move this to internal/bootstrap/browser when the CLI flag is
308312
// removed.
309313
function setupCustomEvent() {
@@ -315,27 +319,6 @@ function setupCustomEvent() {
315319
exposeInterface(globalThis, 'CustomEvent', CustomEvent);
316320
}
317321

318-
// Setup User-facing NODE_V8_COVERAGE environment variable that writes
319-
// ScriptCoverage to a specified file.
320-
function setupCoverageHooks(dir) {
321-
const cwd = require('internal/process/execution').tryGetCwd();
322-
const { resolve } = require('path');
323-
const coverageDirectory = resolve(cwd, dir);
324-
const { sourceMapCacheToObject } =
325-
require('internal/source_map/source_map_cache');
326-
327-
if (process.features.inspector) {
328-
internalBinding('profiler').setCoverageDirectory(coverageDirectory);
329-
internalBinding('profiler').setSourceMapCacheGetter(sourceMapCacheToObject);
330-
} else {
331-
process.emitWarning('The inspector is disabled, ' +
332-
'coverage could not be collected',
333-
'Warning');
334-
return '';
335-
}
336-
return coverageDirectory;
337-
}
338-
339322
function setupStacktracePrinterOnSigint() {
340323
if (!getOptionValue('--trace-sigint')) {
341324
return;

0 commit comments

Comments
 (0)