Skip to content

Commit 1d686b7

Browse files
GeoffreyBoothRafaelGSS
authored andcommitted
module: move test reporter loading
Move the logic for handling --test-reporter out of the general module loader and into the test_runner subsystem. PR-URL: #45923 Reviewed-By: Moshe Atlow <[email protected]> Reviewed-By: Benjamin Gruenbaum <[email protected]> Reviewed-By: Colin Ihrig <[email protected]> Reviewed-By: Jacob Smith <[email protected]> Reviewed-By: Antoine du Hamel <[email protected]>
1 parent ce2b88d commit 1d686b7

File tree

10 files changed

+90
-61
lines changed

10 files changed

+90
-61
lines changed

doc/api/test.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -470,7 +470,7 @@ The following built-reporters are supported:
470470
The `spec` reporter outputs the test results in a human-readable format.
471471

472472
* `dot`
473-
The `dot` reporter outputs the test results in a comact format,
473+
The `dot` reporter outputs the test results in a compact format,
474474
where each passing test is represented by a `.`,
475475
and each failing test is represented by a `X`.
476476

@@ -591,6 +591,9 @@ module.exports = async function * customReporter(source) {
591591
};
592592
```
593593

594+
The value provided to `--test-reporter` should be a string like one used in an
595+
`import()` in JavaScript code, or a value provided for [`--import`][].
596+
594597
### Multiple reporters
595598

596599
The [`--test-reporter`][] flag can be specified multiple times to report test
@@ -1581,6 +1584,7 @@ added:
15811584
aborted.
15821585

15831586
[TAP]: https://testanything.org/
1587+
[`--import`]: cli.md#--importmodule
15841588
[`--test-name-pattern`]: cli.md#--test-name-pattern
15851589
[`--test-only`]: cli.md#--test-only
15861590
[`--test-reporter-destination`]: cli.md#--test-reporter-destination

lib/internal/modules/run_main.js

+25-1
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,11 @@
22

33
const {
44
ObjectCreate,
5+
StringPrototypeEndsWith,
56
} = primordials;
7+
68
const { getOptionValue } = require('internal/options');
79
const path = require('path');
8-
const { shouldUseESMLoader } = require('internal/modules/utils');
910

1011
function resolveMainPath(main) {
1112
// Note extension resolution for the main entry point can be deprecated in a
@@ -23,6 +24,29 @@ function resolveMainPath(main) {
2324
return mainPath;
2425
}
2526

27+
function shouldUseESMLoader(mainPath) {
28+
/**
29+
* @type {string[]} userLoaders A list of custom loaders registered by the user
30+
* (or an empty list when none have been registered).
31+
*/
32+
const userLoaders = getOptionValue('--experimental-loader');
33+
/**
34+
* @type {string[]} userImports A list of preloaded modules registered by the user
35+
* (or an empty list when none have been registered).
36+
*/
37+
const userImports = getOptionValue('--import');
38+
if (userLoaders.length > 0 || userImports.length > 0)
39+
return true;
40+
const { readPackageScope } = require('internal/modules/cjs/loader');
41+
// Determine the module format of the main
42+
if (mainPath && StringPrototypeEndsWith(mainPath, '.mjs'))
43+
return true;
44+
if (!mainPath || StringPrototypeEndsWith(mainPath, '.cjs'))
45+
return false;
46+
const pkg = readPackageScope(mainPath);
47+
return pkg && pkg.data.type === 'module';
48+
}
49+
2650
function runMainESM(mainPath) {
2751
const { loadESM } = require('internal/process/esm_loader');
2852
const { pathToFileURL } = require('internal/url');

lib/internal/modules/utils.js

-55
This file was deleted.

lib/internal/test_runner/utils.js

+13-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
'use strict';
22
const {
33
ArrayPrototypePush,
4+
ObjectCreate,
45
ObjectGetOwnPropertyDescriptor,
56
SafePromiseAllReturnArrayLike,
67
RegExp,
@@ -9,9 +10,9 @@ const {
910
} = primordials;
1011
const { basename } = require('path');
1112
const { createWriteStream } = require('fs');
13+
const { pathToFileURL } = require('internal/url');
1214
const { createDeferredPromise } = require('internal/util');
1315
const { getOptionValue } = require('internal/options');
14-
const { requireOrImport } = require('internal/modules/utils');
1516

1617
const {
1718
codes: {
@@ -103,7 +104,17 @@ const kDefaultDestination = 'stdout';
103104
async function getReportersMap(reporters, destinations) {
104105
return SafePromiseAllReturnArrayLike(reporters, async (name, i) => {
105106
const destination = kBuiltinDestinations.get(destinations[i]) ?? createWriteStream(destinations[i]);
106-
let reporter = await requireOrImport(kBuiltinReporters.get(name) ?? name);
107+
108+
// Load the test reporter passed to --test-reporter
109+
const reporterSpecifier = kBuiltinReporters.get(name) ?? name;
110+
let parentURL;
111+
try {
112+
parentURL = pathToFileURL(process.cwd() + '/').href;
113+
} catch {
114+
parentURL = 'file:///';
115+
}
116+
const { esmLoader } = require('internal/process/esm_loader');
117+
let reporter = await esmLoader.import(reporterSpecifier, parentURL, ObjectCreate(null));
107118

108119
if (reporter?.default) {
109120
reporter = reporter.default;

test/fixtures/test-runner/node_modules/reporter-cjs/index.js

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

test/fixtures/test-runner/node_modules/reporter-cjs/package.json

+4
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

test/fixtures/test-runner/node_modules/reporter-esm/index.mjs

+8
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

test/fixtures/test-runner/node_modules/reporter-esm/package.json

+4
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

test/parallel/test-bootstrap-modules.js

-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,6 @@ const expectedModules = new Set([
5353
'NativeModule internal/idna',
5454
'NativeModule internal/linkedlist',
5555
'NativeModule internal/modules/cjs/loader',
56-
'NativeModule internal/modules/utils',
5756
'NativeModule internal/modules/esm/utils',
5857
'NativeModule internal/modules/helpers',
5958
'NativeModule internal/modules/package_json_reader',

test/parallel/test-runner-reporters.js

+23-1
Original file line numberDiff line numberDiff line change
@@ -86,12 +86,34 @@ describe('node:test reporters', { concurrency: true }, () => {
8686
it(`should support a '${ext}' file as a custom reporter`, async () => {
8787
const filename = `custom.${ext}`;
8888
const child = spawnSync(process.execPath,
89-
['--test', '--test-reporter', fixtures.path('test-runner/custom_reporters/', filename),
89+
['--test', '--test-reporter', fixtures.fileURL('test-runner/custom_reporters/', filename),
9090
testFile]);
9191
assert.strictEqual(child.stderr.toString(), '');
9292
const stdout = child.stdout.toString();
9393
assert.match(stdout, /{"test:start":5,"test:pass":2,"test:fail":3,"test:plan":3,"test:diagnostic":\d+}$/);
9494
assert.strictEqual(stdout.slice(0, filename.length + 2), `${filename} {`);
9595
});
9696
});
97+
98+
it('should support a custom reporter from node_modules', async () => {
99+
const child = spawnSync(process.execPath,
100+
['--test', '--test-reporter', 'reporter-cjs', 'reporters.js'],
101+
{ cwd: fixtures.path('test-runner') });
102+
assert.strictEqual(child.stderr.toString(), '');
103+
assert.match(
104+
child.stdout.toString(),
105+
/^package: reporter-cjs{"test:start":5,"test:pass":2,"test:fail":3,"test:plan":3,"test:diagnostic":\d+}$/,
106+
);
107+
});
108+
109+
it('should support a custom ESM reporter from node_modules', async () => {
110+
const child = spawnSync(process.execPath,
111+
['--test', '--test-reporter', 'reporter-esm', 'reporters.js'],
112+
{ cwd: fixtures.path('test-runner') });
113+
assert.strictEqual(child.stderr.toString(), '');
114+
assert.match(
115+
child.stdout.toString(),
116+
/^package: reporter-esm{"test:start":5,"test:pass":2,"test:fail":3,"test:plan":3,"test:diagnostic":\d+}$/,
117+
);
118+
});
97119
});

0 commit comments

Comments
 (0)