From 1fad0ca9c437404c417b8c5a7131634cc52d462a Mon Sep 17 00:00:00 2001 From: Colin Ihrig Date: Sun, 19 May 2024 01:09:09 -0400 Subject: [PATCH] test_runner: support module mocking This commit adds experimental module mocking to the test runner. PR-URL: https://github.com/nodejs/node/pull/52848 Fixes: https://github.com/nodejs/node/issues/51164 Reviewed-By: Benjamin Gruenbaum Reviewed-By: Moshe Atlow Reviewed-By: Rafael Gonzaga Reviewed-By: Geoffrey Booth --- doc/api/cli.md | 10 + doc/api/test.md | 81 +++ doc/node.1 | 3 + lib/internal/test_runner/mock/mock.js | 330 ++++++++- lib/internal/util.js | 23 +- lib/test/mock_loader.js | 227 +++++++ src/node_options.cc | 3 + src/node_options.h | 1 + test/fixtures/module-mocking/basic-cjs.js | 5 + .../module-mocking/basic-esm-mock.mjs | 1 + test/fixtures/module-mocking/basic-esm.mjs | 1 + test/fixtures/test-runner/mock-nm.js | 21 + test/parallel/test-runner-module-mocking.js | 640 ++++++++++++++++++ tools/doc/type-parser.mjs | 2 + 14 files changed, 1336 insertions(+), 12 deletions(-) create mode 100644 lib/test/mock_loader.js create mode 100644 test/fixtures/module-mocking/basic-cjs.js create mode 100644 test/fixtures/module-mocking/basic-esm-mock.mjs create mode 100644 test/fixtures/module-mocking/basic-esm.mjs create mode 100644 test/fixtures/test-runner/mock-nm.js create mode 100644 test/parallel/test-runner-module-mocking.js diff --git a/doc/api/cli.md b/doc/api/cli.md index f686ffade8f421..8f1ddc95d0b64b 100644 --- a/doc/api/cli.md +++ b/doc/api/cli.md @@ -981,6 +981,16 @@ generated as part of the test runner output. If no tests are run, a coverage report is not generated. See the documentation on [collecting code coverage from tests][] for more details. +### `--experimental-test-module-mocks` + + + +> Stability: 1.0 - Early development + +Enable module mocking in the test runner. + ### `--experimental-vm-modules` + +> Stability: 1.0 - Early development + +The `MockModuleContext` class is used to manipulate the behavior of module mocks +created via the [`MockTracker`][] APIs. + +### `ctx.restore()` + + + +Resets the implementation of the mock module. + ## Class: `MockTracker` + +> Stability: 1.0 - Early development + +* `specifier` {string} A string identifying the module to mock. +* `options` {Object} Optional configuration options for the mock module. The + following properties are supported: + * `cache` {boolean} If `false`, each call to `require()` or `import()` + generates a new mock module. If `true`, subsequent calls will return the same + module mock, and the mock module is inserted into the CommonJS cache. + **Default:** false. + * `defaultExport` {any} An optional value used as the mocked module's default + export. If this value is not provided, ESM mocks do not include a default + export. If the mock is a CommonJS or builtin module, this setting is used as + the value of `module.exports`. If this value is not provided, CJS and builtin + mocks use an empty object as the value of `module.exports`. + * `namedExports` {Object} An optional object whose keys and values are used to + create the named exports of the mock module. If the mock is a CommonJS or + builtin module, these values are copied onto `module.exports`. Therefore, if a + mock is created with both named exports and a non-object default export, the + mock will throw an exception when used as a CJS or builtin module. +* Returns: {MockModuleContext} An object that can be used to manipulate the mock. + +This function is used to mock the exports of ECMAScript modules, CommonJS +modules, and Node.js builtin modules. Any references to the original module +prior to mocking are not impacted. The following example demonstrates how a mock +is created for a module. + +```js +test('mocks a builtin module in both module systems', async (t) => { + // Create a mock of 'node:readline' with a named export named 'fn', which + // does not exist in the original 'node:readline' module. + const mock = t.mock.module('node:readline', { + namedExports: { fn() { return 42; } }, + }); + + let esmImpl = await import('node:readline'); + let cjsImpl = require('node:readline'); + + // cursorTo() is an export of the original 'node:readline' module. + assert.strictEqual(esmImpl.cursorTo, undefined); + assert.strictEqual(cjsImpl.cursorTo, undefined); + assert.strictEqual(esmImpl.fn(), 42); + assert.strictEqual(cjsImpl.fn(), 42); + + mock.restore(); + + // The mock is restored, so the original builtin module is returned. + esmImpl = await import('node:readline'); + cjsImpl = require('node:readline'); + + assert.strictEqual(typeof esmImpl.cursorTo, 'function'); + assert.strictEqual(typeof cjsImpl.cursorTo, 'function'); + assert.strictEqual(esmImpl.fn, undefined); + assert.strictEqual(cjsImpl.fn, undefined); +}); +``` + ### `mock.reset()`