Skip to content

Commit 1f4f9be

Browse files
JakobJingleheimeraduh95
authored andcommitted
module: fix async resolution error within the sync findPackageJSON
PR-URL: #56382 Backport-PR-URL: #56494 Reviewed-By: Jordan Harband <[email protected]> Reviewed-By: Geoffrey Booth <[email protected]>
1 parent bbedffa commit 1f4f9be

File tree

3 files changed

+58
-8
lines changed

3 files changed

+58
-8
lines changed

doc/api/module.md

+9-4
Original file line numberDiff line numberDiff line change
@@ -232,13 +232,17 @@ added: REPLACEME
232232
* `base` {string|URL} The absolute location (`file:` URL string or FS path) of the
233233
containing module. For CJS, use `__filename` (not `__dirname`!); for ESM, use
234234
`import.meta.url`. You do not need to pass it if `specifier` is an `absolute specifier`.
235-
* Returns: {string|undefined} A path if the `package.json` is found. When `startLocation`
235+
* Returns: {string|undefined} A path if the `package.json` is found. When `specifier`
236236
is a package, the package's root `package.json`; when a relative or unresolved, the closest
237-
`package.json` to the `startLocation`.
237+
`package.json` to the `specifier`.
238238
239-
> **Caveat**: Do not use this to try to determine module format. There are many things effecting
239+
> **Caveat**: Do not use this to try to determine module format. There are many things affecting
240240
> that determination; the `type` field of package.json is the _least_ definitive (ex file extension
241-
> superceeds it, and a loader hook superceeds that).
241+
> supersedes it, and a loader hook supersedes that).
242+
243+
> **Caveat**: This currently leverages only the built-in default resolver; if
244+
> [`resolve` customization hooks][resolve hook] are registered, they will not affect the resolution.
245+
> This may change in the future.
242246
243247
```text
244248
/path/to/project
@@ -1579,6 +1583,7 @@ returned object contains the following keys:
15791583
[module wrapper]: modules.md#the-module-wrapper
15801584
[prefix-only modules]: modules.md#built-in-modules-with-mandatory-node-prefix
15811585
[realm]: https://tc39.es/ecma262/#realm
1586+
[resolve hook]: #resolvespecifier-context-nextresolve
15821587
[source map include directives]: https://sourcemaps.info/spec.html#h.lmz475t4mvbx
15831588
[transferable objects]: worker_threads.md#portpostmessagevalue-transferlist
15841589
[transform TypeScript features]: typescript.md#typescript-features

lib/internal/modules/package_json_reader.js

+9-4
Original file line numberDiff line numberDiff line change
@@ -268,8 +268,8 @@ function getPackageJSONURL(specifier, base) {
268268
throw new ERR_MODULE_NOT_FOUND(packageName, fileURLToPath(base), null);
269269
}
270270

271-
const pjsonImportAttributes = { __proto__: null, type: 'json' };
272-
let cascadedLoader;
271+
/** @type {import('./esm/resolve.js').defaultResolve} */
272+
let defaultResolve;
273273
/**
274274
* @param {URL['href'] | string | URL} specifier The location for which to get the "root" package.json
275275
* @param {URL['href'] | string | URL} [base] The location of the current module (ex file://tmp/foo.js).
@@ -297,10 +297,15 @@ function findPackageJSON(specifier, base = 'data:') {
297297
}
298298

299299
let resolvedTarget;
300-
cascadedLoader ??= require('internal/modules/esm/loader').getOrInitializeCascadedLoader();
300+
defaultResolve ??= require('internal/modules/esm/resolve').defaultResolve;
301301

302302
try {
303-
resolvedTarget = cascadedLoader.resolve(specifier, `${parentURL}`, pjsonImportAttributes).url;
303+
// TODO(@JakobJingleheimer): Detect whether findPackageJSON is being used within a loader
304+
// (possibly piggyback on `allowImportMetaResolve`)
305+
// - When inside, use the default resolve
306+
// - (I think it's impossible to use the chain because of re-entry & a deadlock from atomics).
307+
// - When outside, use cascadedLoader.resolveSync (not implemented yet, but the pieces exist).
308+
resolvedTarget = defaultResolve(specifier, { parentURL: `${parentURL}` }).url;
304309
} catch (err) {
305310
if (err.code === 'ERR_UNSUPPORTED_DIR_IMPORT') {
306311
resolvedTarget = err.url;

test/parallel/test-find-package-json.js

+40
Original file line numberDiff line numberDiff line change
@@ -149,4 +149,44 @@ describe('findPackageJSON', () => { // Throws when no arguments are provided
149149
});
150150
}));
151151
});
152+
153+
it('should work within a loader', async () => {
154+
const specifierBase = './packages/root-types-field';
155+
const target = fixtures.fileURL(specifierBase, 'index.js');
156+
const foundPjsonPath = path.toNamespacedPath(fixtures.path(specifierBase, 'package.json'));
157+
const { code, stderr, stdout } = await common.spawnPromisified(process.execPath, [
158+
'--no-warnings',
159+
'--loader',
160+
[
161+
'data:text/javascript,',
162+
'import fs from "node:fs";',
163+
'import module from "node:module";',
164+
encodeURIComponent(`fs.writeSync(1, module.findPackageJSON(${JSON.stringify(target)}));`),
165+
'export const resolve = async (s, c, n) => n(s);',
166+
].join(''),
167+
'--eval',
168+
'import "node:os";', // Can be anything that triggers the resolve hook chain
169+
]);
170+
171+
assert.strictEqual(stderr, '');
172+
assert.ok(stdout.includes(foundPjsonPath), stdout);
173+
assert.strictEqual(code, 0);
174+
});
175+
176+
it('should work with an async resolve hook registered', async () => {
177+
const specifierBase = './packages/root-types-field';
178+
const target = fixtures.fileURL(specifierBase, 'index.js');
179+
const foundPjsonPath = path.toNamespacedPath(fixtures.path(specifierBase, 'package.json'));
180+
const { code, stderr, stdout } = await common.spawnPromisified(process.execPath, [
181+
'--no-warnings',
182+
'--loader',
183+
'data:text/javascript,export const resolve = async (s, c, n) => n(s);',
184+
'--print',
185+
`require("node:module").findPackageJSON(${JSON.stringify(target)})`,
186+
]);
187+
188+
assert.strictEqual(stderr, '');
189+
assert.ok(stdout.includes(foundPjsonPath), stdout);
190+
assert.strictEqual(code, 0);
191+
});
152192
});

0 commit comments

Comments
 (0)