From 5ffc90a06b0595a2b8af0e4771698fb3bc1f2e32 Mon Sep 17 00:00:00 2001 From: Geoffrey Booth Date: Mon, 19 Dec 2022 23:36:45 -0800 Subject: [PATCH] esm: move hooks handling into separate class PR-URL: https://github.com/nodejs/node/pull/45869 Reviewed-By: Antoine du Hamel Reviewed-By: Jacob Smith Reviewed-By: Rich Trott Reviewed-By: Matteo Collina --- lib/internal/modules/esm/hooks.js | 651 ++++++++++++++++++++++++++++ lib/internal/modules/esm/load.js | 25 +- lib/internal/modules/esm/loader.js | 660 +++-------------------------- lib/internal/process/esm_loader.js | 4 +- 4 files changed, 741 insertions(+), 599 deletions(-) create mode 100644 lib/internal/modules/esm/hooks.js diff --git a/lib/internal/modules/esm/hooks.js b/lib/internal/modules/esm/hooks.js new file mode 100644 index 00000000000000..8d826f40420445 --- /dev/null +++ b/lib/internal/modules/esm/hooks.js @@ -0,0 +1,651 @@ +'use strict'; + +const { + ArrayPrototypeJoin, + ArrayPrototypePush, + FunctionPrototypeCall, + ObjectAssign, + ObjectDefineProperty, + ObjectSetPrototypeOf, + SafeSet, + StringPrototypeSlice, + StringPrototypeToUpperCase, + globalThis, +} = primordials; + +const { + ERR_LOADER_CHAIN_INCOMPLETE, + ERR_INTERNAL_ASSERTION, + ERR_INVALID_ARG_TYPE, + ERR_INVALID_ARG_VALUE, + ERR_INVALID_RETURN_PROPERTY_VALUE, + ERR_INVALID_RETURN_VALUE, +} = require('internal/errors').codes; +const { isURL, URL } = require('internal/url'); +const { + isAnyArrayBuffer, + isArrayBufferView, +} = require('internal/util/types'); +const { + validateObject, + validateString, +} = require('internal/validators'); + +const { + defaultResolve, +} = require('internal/modules/esm/resolve'); +const { + getDefaultConditions, +} = require('internal/modules/esm/utils'); + + +/** + * @typedef {object} KeyedHook + * @property {Function} fn The hook function. + * @property {URL['href']} url The URL of the module. + */ + +// [2] `validate...()`s throw the wrong error + + +class Hooks { + #hooks = { + /** + * Prior to ESM loading. These are called once before any modules are started. + * @private + * @property {KeyedHook[]} globalPreload Last-in-first-out list of preload hooks. + */ + globalPreload: [], + + /** + * Phase 1 of 2 in ESM loading. + * The output of the `resolve` chain of hooks is passed into the `load` chain of hooks. + * @private + * @property {KeyedHook[]} resolve Last-in-first-out collection of resolve hooks. + */ + resolve: [ + { + fn: defaultResolve, + url: 'node:internal/modules/esm/resolve', + }, + ], + + /** + * Phase 2 of 2 in ESM loading. + * @private + * @property {KeyedHook[]} load Last-in-first-out collection of loader hooks. + */ + load: [ + { + fn: require('internal/modules/esm/load').defaultLoad, + url: 'node:internal/modules/esm/load', + }, + ], + }; + + // Enable an optimization in ESMLoader.getModuleJob + hasCustomLoadHooks = false; + + // Cache URLs we've already validated to avoid repeated validation + #validatedUrls = new SafeSet(); + + #importMetaInitializer = require('internal/modules/esm/initialize_import_meta').initializeImportMeta; + + constructor(userLoaders) { + this.#addCustomLoaders(userLoaders); + } + + /** + * Collect custom/user-defined module loader hook(s). + * After all hooks have been collected, the global preload hook(s) must be initialized. + * @param {import('./loader.js).KeyedExports} customLoaders Exports from user-defined loaders + * (as returned by `ESMLoader.import()`). + */ + #addCustomLoaders( + customLoaders = [], + ) { + for (let i = 0; i < customLoaders.length; i++) { + const { + exports, + url, + } = customLoaders[i]; + const { + globalPreload, + resolve, + load, + } = pluckHooks(exports); + + if (globalPreload) { + ArrayPrototypePush( + this.#hooks.globalPreload, + { + fn: globalPreload, + url, + }, + ); + } + if (resolve) { + ArrayPrototypePush( + this.#hooks.resolve, + { + fn: resolve, + url, + }, + ); + } + if (load) { + this.hasCustomLoadHooks = true; + ArrayPrototypePush( + this.#hooks.load, + { + fn: load, + url, + }, + ); + } + } + } + + /** + * Initialize `globalPreload` hooks. + */ + preload() { + for (let i = this.#hooks.globalPreload.length - 1; i >= 0; i--) { + const { MessageChannel } = require('internal/worker/io'); + const channel = new MessageChannel(); + const { + port1: insidePreload, + port2: insideLoader, + } = channel; + + insidePreload.unref(); + insideLoader.unref(); + + const { + fn: preload, + url: specifier, + } = this.#hooks.globalPreload[i]; + + const preloaded = preload({ + port: insideLoader, + }); + + if (preloaded == null) { return; } + + const hookErrIdentifier = `${specifier} globalPreload`; + + if (typeof preloaded !== 'string') { // [2] + throw new ERR_INVALID_RETURN_VALUE( + 'a string', + hookErrIdentifier, + preload, + ); + } + const { compileFunction } = require('vm'); + const preloadInit = compileFunction( + preloaded, + ['getBuiltin', 'port', 'setImportMetaCallback'], + { + filename: '', + }, + ); + const { BuiltinModule } = require('internal/bootstrap/loaders'); + // We only allow replacing the importMetaInitializer during preload; + // after preload is finished, we disable the ability to replace it. + // + // This exposes accidentally setting the initializer too late by throwing an error. + let finished = false; + let replacedImportMetaInitializer = false; + let next = this.#importMetaInitializer; + try { + // Calls the compiled preload source text gotten from the hook + // Since the parameters are named we use positional parameters + // see compileFunction above to cross reference the names + FunctionPrototypeCall( + preloadInit, + globalThis, + // Param getBuiltin + (builtinName) => { + if (BuiltinModule.canBeRequiredByUsers(builtinName) && + BuiltinModule.canBeRequiredWithoutScheme(builtinName)) { + return require(builtinName); + } + throw new ERR_INVALID_ARG_VALUE('builtinName', builtinName); + }, + // Param port + insidePreload, + // Param setImportMetaCallback + (fn) => { + if (finished || typeof fn !== 'function') { + throw new ERR_INVALID_ARG_TYPE('fn', fn); + } + replacedImportMetaInitializer = true; + const parent = next; + next = (meta, context) => { + return fn(meta, context, parent); + }; + }); + } finally { + finished = true; + if (replacedImportMetaInitializer) { + this.#importMetaInitializer = next; + } + } + } + } + + importMetaInitialize(meta, context) { + this.#importMetaInitializer(meta, context); + } + + /** + * Resolve the location of the module. + * + * Internally, this behaves like a backwards iterator, wherein the stack of + * hooks starts at the top and each call to `nextResolve()` moves down 1 step + * until it reaches the bottom or short-circuits. + * @param {string} originalSpecifier The specified URL path of the module to + * be resolved. + * @param {string} [parentURL] The URL path of the module's parent. + * @param {ImportAssertions} [importAssertions] Assertions from the import + * statement or expression. + * @returns {Promise<{ format: string, url: URL['href'] }>} + */ + async resolve( + originalSpecifier, + parentURL, + importAssertions = { __proto__: null }, + ) { + const isMain = parentURL === undefined; + + if ( + !isMain && + typeof parentURL !== 'string' && + !isURL(parentURL) + ) { + throw new ERR_INVALID_ARG_TYPE( + 'parentURL', + ['string', 'URL'], + parentURL, + ); + } + const chain = this.#hooks.resolve; + const context = { + conditions: getDefaultConditions(), + importAssertions, + parentURL, + }; + const meta = { + chainFinished: null, + context, + hookErrIdentifier: '', + hookIndex: chain.length - 1, + hookName: 'resolve', + shortCircuited: false, + }; + + const validateArgs = (hookErrIdentifier, suppliedSpecifier, ctx) => { + validateString( + suppliedSpecifier, + `${hookErrIdentifier} specifier`, + ); // non-strings can be coerced to a URL string + + if (ctx) validateObject(ctx, `${hookErrIdentifier} context`); + }; + const validateOutput = (hookErrIdentifier, output) => { + if (typeof output !== 'object' || output === null) { // [2] + throw new ERR_INVALID_RETURN_VALUE( + 'an object', + hookErrIdentifier, + output, + ); + } + }; + + const nextResolve = nextHookFactory(chain, meta, { validateArgs, validateOutput }); + + const resolution = await nextResolve(originalSpecifier, context); + const { hookErrIdentifier } = meta; // Retrieve the value after all settled + + validateOutput(hookErrIdentifier, resolution); + + if (resolution?.shortCircuit === true) { meta.shortCircuited = true; } + + if (!meta.chainFinished && !meta.shortCircuited) { + throw new ERR_LOADER_CHAIN_INCOMPLETE(hookErrIdentifier); + } + + const { + format, + url, + } = resolution; + + if ( + format != null && + typeof format !== 'string' // [2] + ) { + throw new ERR_INVALID_RETURN_PROPERTY_VALUE( + 'a string', + hookErrIdentifier, + 'format', + format, + ); + } + + if (typeof url !== 'string') { + // non-strings can be coerced to a URL string + // validateString() throws a less-specific error + throw new ERR_INVALID_RETURN_PROPERTY_VALUE( + 'a URL string', + hookErrIdentifier, + 'url', + url, + ); + } + + // Avoid expensive URL instantiation for known-good URLs + if (!this.#validatedUrls.has(url)) { + try { + new URL(url); + this.#validatedUrls.add(url); + } catch { + throw new ERR_INVALID_RETURN_PROPERTY_VALUE( + 'a URL string', + hookErrIdentifier, + 'url', + url, + ); + } + } + + return { + __proto__: null, + format, + url, + }; + } + + /** + * Provide source that is understood by one of Node's translators. + * + * Internally, this behaves like a backwards iterator, wherein the stack of + * hooks starts at the top and each call to `nextLoad()` moves down 1 step + * until it reaches the bottom or short-circuits. + * @param {URL['href']} url The URL/path of the module to be loaded + * @param {object} context Metadata about the module + * @returns {Promise<{ format: ModuleFormat, source: ModuleSource }>} + */ + async load(url, context = {}) { + const chain = this.#hooks.load; + const meta = { + chainFinished: null, + context, + hookErrIdentifier: '', + hookIndex: chain.length - 1, + hookName: 'load', + shortCircuited: false, + }; + + const validateArgs = (hookErrIdentifier, nextUrl, ctx) => { + if (typeof nextUrl !== 'string') { + // Non-strings can be coerced to a URL string + // validateString() throws a less-specific error + throw new ERR_INVALID_ARG_TYPE( + `${hookErrIdentifier} url`, + 'a URL string', + nextUrl, + ); + } + + // Avoid expensive URL instantiation for known-good URLs + if (!this.#validatedUrls.has(nextUrl)) { + try { + new URL(nextUrl); + this.#validatedUrls.add(nextUrl); + } catch { + throw new ERR_INVALID_ARG_VALUE( + `${hookErrIdentifier} url`, + nextUrl, + 'should be a URL string', + ); + } + } + + if (ctx) { validateObject(ctx, `${hookErrIdentifier} context`); } + }; + const validateOutput = (hookErrIdentifier, output) => { + if (typeof output !== 'object' || output === null) { // [2] + throw new ERR_INVALID_RETURN_VALUE( + 'an object', + hookErrIdentifier, + output, + ); + } + }; + + const nextLoad = nextHookFactory(chain, meta, { validateArgs, validateOutput }); + + const loaded = await nextLoad(url, context); + const { hookErrIdentifier } = meta; // Retrieve the value after all settled + + validateOutput(hookErrIdentifier, loaded); + + if (loaded?.shortCircuit === true) { meta.shortCircuited = true; } + + if (!meta.chainFinished && !meta.shortCircuited) { + throw new ERR_LOADER_CHAIN_INCOMPLETE(hookErrIdentifier); + } + + const { + format, + source, + } = loaded; + let responseURL = loaded.responseURL; + + if (responseURL === undefined) { + responseURL = url; + } + + let responseURLObj; + if (typeof responseURL === 'string') { + try { + responseURLObj = new URL(responseURL); + } catch { + // responseURLObj not defined will throw in next branch. + } + } + + if (responseURLObj?.href !== responseURL) { + throw new ERR_INVALID_RETURN_PROPERTY_VALUE( + 'undefined or a fully resolved URL string', + hookErrIdentifier, + 'responseURL', + responseURL, + ); + } + + if (format == null) { + require('internal/modules/esm/load').throwUnknownModuleFormat(url, format); + } + + if (typeof format !== 'string') { // [2] + throw new ERR_INVALID_RETURN_PROPERTY_VALUE( + 'a string', + hookErrIdentifier, + 'format', + format, + ); + } + + if ( + source != null && + typeof source !== 'string' && + !isAnyArrayBuffer(source) && + !isArrayBufferView(source) + ) { + throw ERR_INVALID_RETURN_PROPERTY_VALUE( + 'a string, an ArrayBuffer, or a TypedArray', + hookErrIdentifier, + 'source', + source, + ); + } + + return { + __proto__: null, + format, + responseURL, + source, + }; + } +} + +ObjectSetPrototypeOf(Hooks.prototype, null); + + +/** + * A utility function to pluck the hooks from a user-defined loader. + * @param {import('./loader.js).ModuleExports} exports + * @returns {import('./loader.js).ExportedHooks} + */ +function pluckHooks({ + globalPreload, + resolve, + load, + // obsolete hooks: + dynamicInstantiate, + getFormat, + getGlobalPreloadCode, + getSource, + transformSource, +}) { + const obsoleteHooks = []; + const acceptedHooks = { __proto__: null }; + + if (getGlobalPreloadCode) { + globalPreload ??= getGlobalPreloadCode; + + process.emitWarning( + 'Loader hook "getGlobalPreloadCode" has been renamed to "globalPreload"', + ); + } + if (dynamicInstantiate) { + ArrayPrototypePush(obsoleteHooks, 'dynamicInstantiate'); + } + if (getFormat) { + ArrayPrototypePush(obsoleteHooks, 'getFormat'); + } + if (getSource) { + ArrayPrototypePush(obsoleteHooks, 'getSource'); + } + if (transformSource) { + ArrayPrototypePush(obsoleteHooks, 'transformSource'); + } + + if (obsoleteHooks.length) { + process.emitWarning( + `Obsolete loader hook(s) supplied and will be ignored: ${ + ArrayPrototypeJoin(obsoleteHooks, ', ') + }`, + 'DeprecationWarning', + ); + } + + if (globalPreload) { + acceptedHooks.globalPreload = globalPreload; + } + if (resolve) { + acceptedHooks.resolve = resolve; + } + if (load) { + acceptedHooks.load = load; + } + + return acceptedHooks; +} + + +/** + * A utility function to iterate through a hook chain, track advancement in the + * chain, and generate and supply the `next` argument to the custom + * hook. + * @param {KeyedHook[]} chain The whole hook chain. + * @param {object} meta Properties that change as the current hook advances + * along the chain. + * @param {boolean} meta.chainFinished Whether the end of the chain has been + * reached AND invoked. + * @param {string} meta.hookErrIdentifier A user-facing identifier to help + * pinpoint where an error occurred. Ex "file:///foo.mjs 'resolve'". + * @param {number} meta.hookIndex A non-negative integer tracking the current + * position in the hook chain. + * @param {string} meta.hookName The kind of hook the chain is (ex 'resolve') + * @param {boolean} meta.shortCircuited Whether a hook signaled a short-circuit. + * @param {(hookErrIdentifier, hookArgs) => void} validate A wrapper function + * containing all validation of a custom loader hook's intermediary output. Any + * validation within MUST throw. + * @returns {function next(...hookArgs)} The next hook in the chain. + */ +function nextHookFactory(chain, meta, { validateArgs, validateOutput }) { + // First, prepare the current + const { hookName } = meta; + const { + fn: hook, + url: hookFilePath, + } = chain[meta.hookIndex]; + + // ex 'nextResolve' + const nextHookName = `next${ + StringPrototypeToUpperCase(hookName[0]) + + StringPrototypeSlice(hookName, 1) + }`; + + // When hookIndex is 0, it's reached the default, which does not call next() + // so feed it a noop that blows up if called, so the problem is obvious. + const generatedHookIndex = meta.hookIndex; + let nextNextHook; + if (meta.hookIndex > 0) { + // Now, prepare the next: decrement the pointer so the next call to the + // factory generates the next link in the chain. + meta.hookIndex--; + + nextNextHook = nextHookFactory(chain, meta, { validateArgs, validateOutput }); + } else { + // eslint-disable-next-line func-name-matching + nextNextHook = function chainAdvancedTooFar() { + throw new ERR_INTERNAL_ASSERTION( + `ESM custom loader '${hookName}' advanced beyond the end of the chain.`, + ); + }; + } + + return ObjectDefineProperty( + async (arg0 = undefined, context) => { + // Update only when hook is invoked to avoid fingering the wrong filePath + meta.hookErrIdentifier = `${hookFilePath} '${hookName}'`; + + validateArgs(`${meta.hookErrIdentifier} hook's ${nextHookName}()`, arg0, context); + + const outputErrIdentifier = `${chain[generatedHookIndex].url} '${hookName}' hook's ${nextHookName}()`; + + // Set when next is actually called, not just generated. + if (generatedHookIndex === 0) { meta.chainFinished = true; } + + if (context) { // `context` has already been validated, so no fancy check needed. + ObjectAssign(meta.context, context); + } + + const output = await hook(arg0, meta.context, nextNextHook); + + validateOutput(outputErrIdentifier, output); + + if (output?.shortCircuit === true) { meta.shortCircuited = true; } + + return output; + }, + 'name', + { __proto__: null, value: nextHookName }, + ); +} + + +exports.Hooks = Hooks; diff --git a/lib/internal/modules/esm/load.js b/lib/internal/modules/esm/load.js index d71a81a4edc811..29135cd08103f2 100644 --- a/lib/internal/modules/esm/load.js +++ b/lib/internal/modules/esm/load.js @@ -5,6 +5,7 @@ const { RegExpPrototypeExec, decodeURIComponent, } = primordials; +const { kEmptyObject } = require('internal/util'); const { defaultGetFormat } = require('internal/modules/esm/get_format'); const { validateAssertions } = require('internal/modules/esm/assert'); @@ -22,6 +23,7 @@ const { Buffer: { from: BufferFrom } } = require('buffer'); const { URL } = require('internal/url'); const { ERR_INVALID_URL, + ERR_UNKNOWN_MODULE_FORMAT, ERR_UNSUPPORTED_ESM_URL_SCHEME, } = require('internal/errors').codes; @@ -74,7 +76,7 @@ async function getSource(url, context) { * @param {object} context * @returns {object} */ -async function defaultLoad(url, context) { +async function defaultLoad(url, context = kEmptyObject) { let responseURL = url; const { importAssertions } = context; let { @@ -107,6 +109,7 @@ async function defaultLoad(url, context) { }; } + /** * throws an error if the protocol is not one of the protocols * that can be loaded in the default loader @@ -137,6 +140,26 @@ function throwIfUnsupportedURLScheme(parsed, experimentalNetworkImports) { } } +/** + * For a falsy `format` returned from `load`, throw an error. + * This could happen from either a custom user loader _or_ from the default loader, because the default loader tries to + * determine formats for data URLs. + * @param {string} url The resolved URL of the module + * @param {null | undefined | false | 0 | -0 | 0n | ''} format Falsy format returned from `load` + */ +function throwUnknownModuleFormat(url, format) { + const dataUrl = RegExpPrototypeExec( + /^data:([^/]+\/[^;,]+)(?:[^,]*?)(;base64)?,/, + url, + ); + + throw new ERR_UNKNOWN_MODULE_FORMAT( + dataUrl ? dataUrl[1] : format, + url); +} + + module.exports = { defaultLoad, + throwUnknownModuleFormat, }; diff --git a/lib/internal/modules/esm/loader.js b/lib/internal/modules/esm/loader.js index 87f11a6cb2edeb..54fff8831e5838 100644 --- a/lib/internal/modules/esm/loader.js +++ b/lib/internal/modules/esm/loader.js @@ -6,58 +6,33 @@ require('internal/modules/cjs/loader'); const { Array, ArrayIsArray, - ArrayPrototypeJoin, - ArrayPrototypePush, FunctionPrototypeCall, - ObjectAssign, ObjectCreate, - ObjectDefineProperty, ObjectSetPrototypeOf, - RegExpPrototypeExec, SafePromiseAllReturnArrayLike, SafeWeakMap, - StringPrototypeSlice, - StringPrototypeToUpperCase, - globalThis, } = primordials; const { - ERR_LOADER_CHAIN_INCOMPLETE, - ERR_INTERNAL_ASSERTION, - ERR_INVALID_ARG_TYPE, - ERR_INVALID_ARG_VALUE, - ERR_INVALID_RETURN_PROPERTY_VALUE, - ERR_INVALID_RETURN_VALUE, ERR_UNKNOWN_MODULE_FORMAT, } = require('internal/errors').codes; -const { pathToFileURL, isURL, URL } = require('internal/url'); +const { getOptionValue } = require('internal/options'); +const { pathToFileURL } = require('internal/url'); const { emitExperimentalWarning } = require('internal/util'); + const { - isAnyArrayBuffer, - isArrayBufferView, -} = require('internal/util/types'); -const { - validateObject, - validateString, -} = require('internal/validators'); + getDefaultConditions, +} = require('internal/modules/esm/utils'); + function newModuleMap() { const ModuleMap = require('internal/modules/esm/module_map'); return new ModuleMap(); } -const { - defaultResolve, -} = require('internal/modules/esm/resolve'); - -const { - getDefaultConditions, -} = require('internal/modules/esm/utils'); - function getTranslators() { const { translators } = require('internal/modules/esm/translators'); return translators; } -const { getOptionValue } = require('internal/options'); /** * @typedef {object} ExportedHooks @@ -76,12 +51,6 @@ const { getOptionValue } = require('internal/options'); * @property {URL['href']} url The URL of the module. */ -/** - * @typedef {object} KeyedHook - * @property {Function} fn The hook function. - * @property {URL['href']} url The URL of the module. - */ - /** * @typedef {'builtin'|'commonjs'|'json'|'module'|'wasm'} ModuleFormat */ @@ -90,92 +59,8 @@ const { getOptionValue } = require('internal/options'); * @typedef {ArrayBuffer|TypedArray|string} ModuleSource */ -// [2] `validate...()`s throw the wrong error - let emittedSpecifierResolutionWarning = false; -/** - * A utility function to iterate through a hook chain, track advancement in the - * chain, and generate and supply the `next` argument to the custom - * hook. - * @param {KeyedHook[]} chain The whole hook chain. - * @param {object} meta Properties that change as the current hook advances - * along the chain. - * @param {boolean} meta.chainFinished Whether the end of the chain has been - * reached AND invoked. - * @param {string} meta.hookErrIdentifier A user-facing identifier to help - * pinpoint where an error occurred. Ex "file:///foo.mjs 'resolve'". - * @param {number} meta.hookIndex A non-negative integer tracking the current - * position in the hook chain. - * @param {string} meta.hookName The kind of hook the chain is (ex 'resolve') - * @param {boolean} meta.shortCircuited Whether a hook signaled a short-circuit. - * @param {(hookErrIdentifier, hookArgs) => void} validate A wrapper function - * containing all validation of a custom loader hook's intermediary output. Any - * validation within MUST throw. - * @returns {function next(...hookArgs)} The next hook in the chain. - */ -function nextHookFactory(chain, meta, { validateArgs, validateOutput }) { - // First, prepare the current - const { hookName } = meta; - const { - fn: hook, - url: hookFilePath, - } = chain[meta.hookIndex]; - - // ex 'nextResolve' - const nextHookName = `next${ - StringPrototypeToUpperCase(hookName[0]) + - StringPrototypeSlice(hookName, 1) - }`; - - // When hookIndex is 0, it's reached the default, which does not call next() - // so feed it a noop that blows up if called, so the problem is obvious. - const generatedHookIndex = meta.hookIndex; - let nextNextHook; - if (meta.hookIndex > 0) { - // Now, prepare the next: decrement the pointer so the next call to the - // factory generates the next link in the chain. - meta.hookIndex--; - - nextNextHook = nextHookFactory(chain, meta, { validateArgs, validateOutput }); - } else { - // eslint-disable-next-line func-name-matching - nextNextHook = function chainAdvancedTooFar() { - throw new ERR_INTERNAL_ASSERTION( - `ESM custom loader '${hookName}' advanced beyond the end of the chain.`, - ); - }; - } - - return ObjectDefineProperty( - async (arg0 = undefined, context) => { - // Update only when hook is invoked to avoid fingering the wrong filePath - meta.hookErrIdentifier = `${hookFilePath} '${hookName}'`; - - validateArgs(`${meta.hookErrIdentifier} hook's ${nextHookName}()`, arg0, context); - - const outputErrIdentifier = `${chain[generatedHookIndex].url} '${hookName}' hook's ${nextHookName}()`; - - // Set when next is actually called, not just generated. - if (generatedHookIndex === 0) { meta.chainFinished = true; } - - if (context) { // `context` has already been validated, so no fancy check needed. - ObjectAssign(meta.context, context); - } - - const output = await hook(arg0, meta.context, nextNextHook); - - validateOutput(outputErrIdentifier, output); - - if (output?.shortCircuit === true) { meta.shortCircuited = true; } - return output; - - }, - 'name', - { __proto__: null, value: nextHookName }, - ); -} - /** * An ESMLoader instance is used as the main entry point for loading ES modules. * Currently, this is a singleton -- there is only one used for loading @@ -183,40 +68,15 @@ function nextHookFactory(chain, meta, { validateArgs, validateOutput }) { */ class ESMLoader { - #hooks = { - /** - * Prior to ESM loading. These are called once before any modules are started. - * @private - * @property {KeyedHook[]} globalPreload Last-in-first-out list of preload hooks. - */ - globalPreload: [], - - /** - * Phase 2 of 2 in ESM loading (phase 1 is below). - * @private - * @property {KeyedHook[]} load Last-in-first-out collection of loader hooks. - */ - load: [ - { - fn: require('internal/modules/esm/load').defaultLoad, - url: 'node:internal/modules/esm/load', - }, - ], - - /** - * Phase 1 of 2 in ESM loading. - * @private - * @property {KeyedHook[]} resolve Last-in-first-out collection of resolve hooks. - */ - resolve: [ - { - fn: defaultResolve, - url: 'node:internal/modules/esm/resolve', - }, - ], - }; - - #importMetaInitializer = require('internal/modules/esm/initialize_import_meta').initializeImportMeta; + #hooks; + #defaultResolve; + #defaultLoad; + #importMetaInitializer; + + /** + * The conditions for resolving packages if `--conditions` is not used. + */ + #defaultConditions = getDefaultConditions(); /** * Map of already-loaded CJS modules to use @@ -257,118 +117,13 @@ class ESMLoader { } } - /** - * - * @param {ModuleExports} exports - * @returns {ExportedHooks} - */ - static pluckHooks({ - globalPreload, - resolve, - load, - // obsolete hooks: - dynamicInstantiate, - getFormat, - getGlobalPreloadCode, - getSource, - transformSource, - }) { - const obsoleteHooks = []; - const acceptedHooks = ObjectCreate(null); - - if (getGlobalPreloadCode) { - globalPreload ??= getGlobalPreloadCode; - - process.emitWarning( - 'Loader hook "getGlobalPreloadCode" has been renamed to "globalPreload"', - ); - } - if (dynamicInstantiate) ArrayPrototypePush( - obsoleteHooks, - 'dynamicInstantiate', - ); - if (getFormat) ArrayPrototypePush( - obsoleteHooks, - 'getFormat', - ); - if (getSource) ArrayPrototypePush( - obsoleteHooks, - 'getSource', - ); - if (transformSource) ArrayPrototypePush( - obsoleteHooks, - 'transformSource', - ); - - if (obsoleteHooks.length) process.emitWarning( - `Obsolete loader hook(s) supplied and will be ignored: ${ - ArrayPrototypeJoin(obsoleteHooks, ', ') - }`, - 'DeprecationWarning', - ); - - if (globalPreload) { - acceptedHooks.globalPreload = globalPreload; - } - if (resolve) { - acceptedHooks.resolve = resolve; - } - if (load) { - acceptedHooks.load = load; - } - - return acceptedHooks; + addCustomLoaders(userLoaders) { + const { Hooks } = require('internal/modules/esm/hooks'); + this.#hooks = new Hooks(userLoaders); } - /** - * Collect custom/user-defined hook(s). After all hooks have been collected, - * the global preload hook(s) must be called. - * @param {KeyedExports} customLoaders - * A list of exports from user-defined loaders (as returned by - * ESMLoader.import()). - */ - addCustomLoaders( - customLoaders = [], - ) { - for (let i = 0; i < customLoaders.length; i++) { - const { - exports, - url, - } = customLoaders[i]; - const { - globalPreload, - resolve, - load, - } = ESMLoader.pluckHooks(exports); - - if (globalPreload) { - ArrayPrototypePush( - this.#hooks.globalPreload, - { - fn: globalPreload, - url, - }, - ); - } - if (resolve) { - ArrayPrototypePush( - this.#hooks.resolve, - { - fn: resolve, - url, - }, - ); - } - if (load) { - ArrayPrototypePush( - this.#hooks.load, - { - fn: load, - url, - }, - ); - } - } + preload() { + this.#hooks?.preload(); } async eval( @@ -414,10 +169,9 @@ class ESMLoader { async getModuleJob(specifier, parentURL, importAssertions) { let importAssertionsForResolve; - // By default, `this.#hooks.load` contains just the Node default load hook - if (this.#hooks.load.length !== 1) { - // We can skip cloning if there are no user-provided loaders because - // the Node.js default resolve hook does not use import assertions. + // We can skip cloning if there are no user-provided loaders because + // the Node.js default resolve hook does not use import assertions. + if (this.#hooks?.hasCustomLoadHooks) { importAssertionsForResolve = { __proto__: null, ...importAssertions, @@ -545,356 +299,70 @@ class ESMLoader { return namespaces; } - /** - * Provide source that is understood by one of Node's translators. - * - * Internally, this behaves like a backwards iterator, wherein the stack of - * hooks starts at the top and each call to `nextLoad()` moves down 1 step - * until it reaches the bottom or short-circuits. - * @param {URL['href']} url The URL/path of the module to be loaded - * @param {object} context Metadata about the module - * @returns {{ format: ModuleFormat, source: ModuleSource }} - */ - async load(url, context = {}) { - const chain = this.#hooks.load; - const meta = { - chainFinished: null, - context, - hookErrIdentifier: '', - hookIndex: chain.length - 1, - hookName: 'load', - shortCircuited: false, - }; - - const validateArgs = (hookErrIdentifier, nextUrl, ctx) => { - if (typeof nextUrl !== 'string') { - // non-strings can be coerced to a url string - // validateString() throws a less-specific error - throw new ERR_INVALID_ARG_TYPE( - `${hookErrIdentifier} url`, - 'a url string', - nextUrl, - ); - } - - // Try to avoid expensive URL instantiation for known-good urls - if (!this.moduleMap.has(nextUrl)) { - try { - new URL(nextUrl); - } catch { - throw new ERR_INVALID_ARG_VALUE( - `${hookErrIdentifier} url`, - nextUrl, - 'should be a url string', - ); - } - } - - if (ctx) validateObject(ctx, `${hookErrIdentifier} context`); - }; - const validateOutput = (hookErrIdentifier, output) => { - if (typeof output !== 'object' || output === null) { // [2] - throw new ERR_INVALID_RETURN_VALUE( - 'an object', - hookErrIdentifier, - output, - ); - } - }; - - const nextLoad = nextHookFactory(chain, meta, { validateArgs, validateOutput }); - - const loaded = await nextLoad(url, context); - const { hookErrIdentifier } = meta; // Retrieve the value after all settled - - validateOutput(hookErrIdentifier, loaded); - - if (loaded?.shortCircuit === true) { meta.shortCircuited = true; } - - if (!meta.chainFinished && !meta.shortCircuited) { - throw new ERR_LOADER_CHAIN_INCOMPLETE(hookErrIdentifier); - } - - const { - format, - source, - } = loaded; - let responseURL = loaded.responseURL; - - if (responseURL === undefined) { - responseURL = url; - } - - let responseURLObj; - if (typeof responseURL === 'string') { - try { - responseURLObj = new URL(responseURL); - } catch { - // responseURLObj not defined will throw in next branch. - } - } - - if (responseURLObj?.href !== responseURL) { - throw new ERR_INVALID_RETURN_PROPERTY_VALUE( - 'undefined or a fully resolved URL string', - hookErrIdentifier, - 'responseURL', - responseURL, - ); - } - - if (format == null) { - const dataUrl = RegExpPrototypeExec( - /^data:([^/]+\/[^;,]+)(?:[^,]*?)(;base64)?,/, - url, - ); - - throw new ERR_UNKNOWN_MODULE_FORMAT( - dataUrl ? dataUrl[1] : format, - url); - } - - if (typeof format !== 'string') { // [2] - throw new ERR_INVALID_RETURN_PROPERTY_VALUE( - 'a string', - hookErrIdentifier, - 'format', - format, - ); - } - - if ( - source != null && - typeof source !== 'string' && - !isAnyArrayBuffer(source) && - !isArrayBufferView(source) - ) { - throw ERR_INVALID_RETURN_PROPERTY_VALUE( - 'a string, an ArrayBuffer, or a TypedArray', - hookErrIdentifier, - 'source', - source, - ); - } - - return { - __proto__: null, - format, - responseURL, - source, - }; - } - - preload() { - for (let i = this.#hooks.globalPreload.length - 1; i >= 0; i--) { - const { MessageChannel } = require('internal/worker/io'); - const channel = new MessageChannel(); - const { - port1: insidePreload, - port2: insideLoader, - } = channel; - - insidePreload.unref(); - insideLoader.unref(); - - const { - fn: preload, - url: specifier, - } = this.#hooks.globalPreload[i]; - - const preloaded = preload({ - port: insideLoader, - }); - - if (preloaded == null) { return; } - - const hookErrIdentifier = `${specifier} globalPreload`; - - if (typeof preloaded !== 'string') { // [2] - throw new ERR_INVALID_RETURN_VALUE( - 'a string', - hookErrIdentifier, - preload, - ); - } - const { compileFunction } = require('vm'); - const preloadInit = compileFunction( - preloaded, - ['getBuiltin', 'port', 'setImportMetaCallback'], - { - filename: '', - }, - ); - const { BuiltinModule } = require('internal/bootstrap/loaders'); - // We only allow replacing the importMetaInitializer during preload, - // after preload is finished, we disable the ability to replace it - // - // This exposes accidentally setting the initializer too late by - // throwing an error. - let finished = false; - let replacedImportMetaInitializer = false; - let next = this.#importMetaInitializer; - try { - // Calls the compiled preload source text gotten from the hook - // Since the parameters are named we use positional parameters - // see compileFunction above to cross reference the names - FunctionPrototypeCall( - preloadInit, - globalThis, - // Param getBuiltin - (builtinName) => { - if (BuiltinModule.canBeRequiredByUsers(builtinName) && - BuiltinModule.canBeRequiredWithoutScheme(builtinName)) { - return require(builtinName); - } - throw new ERR_INVALID_ARG_VALUE('builtinName', builtinName); - }, - // Param port - insidePreload, - // Param setImportMetaCallback - (fn) => { - if (finished || typeof fn !== 'function') { - throw new ERR_INVALID_ARG_TYPE('fn', fn); - } - replacedImportMetaInitializer = true; - const parent = next; - next = (meta, context) => { - return fn(meta, context, parent); - }; - }); - } finally { - finished = true; - if (replacedImportMetaInitializer) { - this.#importMetaInitializer = next; - } - } - } - } - - importMetaInitialize(meta, context) { - this.#importMetaInitializer(meta, context); - } - /** * Resolve the location of the module. - * - * Internally, this behaves like a backwards iterator, wherein the stack of - * hooks starts at the top and each call to `nextResolve()` moves down 1 step - * until it reaches the bottom or short-circuits. * @param {string} originalSpecifier The specified URL path of the module to * be resolved. * @param {string} [parentURL] The URL path of the module's parent. * @param {ImportAssertions} importAssertions Assertions from the import * statement or expression. - * @returns {{ format: string, url: URL['href'] }} + * @returns {Promise<{ format: string, url: URL['href'] }>} */ - async resolve(originalSpecifier, parentURL, importAssertions) { - const isMain = parentURL === undefined; - - if ( - !isMain && - typeof parentURL !== 'string' && - !isURL(parentURL) - ) { - throw new ERR_INVALID_ARG_TYPE( - 'parentURL', - ['string', 'URL'], - parentURL, - ); + async resolve( + originalSpecifier, + parentURL, + importAssertions = ObjectCreate(null), + ) { + if (this.#hooks) { + return this.#hooks.resolve(originalSpecifier, parentURL, importAssertions); + } + if (!this.#defaultResolve) { + this.#defaultResolve = require('internal/modules/esm/resolve').defaultResolve; } - const chain = this.#hooks.resolve; const context = { - conditions: getDefaultConditions(), + __proto__: null, + conditions: this.#defaultConditions, importAssertions, parentURL, }; - const meta = { - chainFinished: null, - context, - hookErrIdentifier: '', - hookIndex: chain.length - 1, - hookName: 'resolve', - shortCircuited: false, - }; + return this.#defaultResolve(originalSpecifier, context); - const validateArgs = (hookErrIdentifier, suppliedSpecifier, ctx) => { - validateString( - suppliedSpecifier, - `${hookErrIdentifier} specifier`, - ); // non-strings can be coerced to a url string + } - if (ctx) validateObject(ctx, `${hookErrIdentifier} context`); - }; - const validateOutput = (hookErrIdentifier, output) => { - if (typeof output !== 'object' || output === null) { // [2] - throw new ERR_INVALID_RETURN_VALUE( - 'an object', - hookErrIdentifier, - output, - ); + /** + * Provide source that is understood by one of Node's translators. + * @param {URL['href']} url The URL/path of the module to be loaded + * @param {object} [context] Metadata about the module + * @returns {Promise<{ format: ModuleFormat, source: ModuleSource }>} + */ + async load(url, context) { + let loadResult; + if (this.#hooks) { + loadResult = await this.#hooks.load(url, context); + } else { + if (!this.#defaultLoad) { + this.#defaultLoad = require('internal/modules/esm/load').defaultLoad; } - }; - - const nextResolve = nextHookFactory(chain, meta, { validateArgs, validateOutput }); - - const resolution = await nextResolve(originalSpecifier, context); - const { hookErrIdentifier } = meta; // Retrieve the value after all settled - - validateOutput(hookErrIdentifier, resolution); - - if (resolution?.shortCircuit === true) { meta.shortCircuited = true; } - - if (!meta.chainFinished && !meta.shortCircuited) { - throw new ERR_LOADER_CHAIN_INCOMPLETE(hookErrIdentifier); + loadResult = await this.#defaultLoad(url, context); } - const { - format, - url, - } = resolution; - - if ( - format != null && - typeof format !== 'string' // [2] - ) { - throw new ERR_INVALID_RETURN_PROPERTY_VALUE( - 'a string', - hookErrIdentifier, - 'format', - format, - ); + const { format } = loadResult; + if (format == null) { + require('internal/modules/esm/load').throwUnknownModuleFormat(url, format); } - if (typeof url !== 'string') { - // non-strings can be coerced to a url string - // validateString() throws a less-specific error - throw new ERR_INVALID_RETURN_PROPERTY_VALUE( - 'a url string', - hookErrIdentifier, - 'url', - url, - ); - } + return loadResult; + } - // Try to avoid expensive URL instantiation for known-good urls - if (!this.moduleMap.has(url)) { - try { - new URL(url); - } catch { - throw new ERR_INVALID_RETURN_PROPERTY_VALUE( - 'a url string', - hookErrIdentifier, - 'url', - url, - ); + importMetaInitialize(meta, context) { + if (this.#hooks) { + this.#hooks.importMetaInitialize(meta, context); + } else { + if (!this.#importMetaInitializer) { + this.#importMetaInitializer = require('internal/modules/esm/initialize_import_meta').initializeImportMeta; } + this.#importMetaInitializer(meta, context); } - - return { - __proto__: null, - format, - url, - }; } } diff --git a/lib/internal/process/esm_loader.js b/lib/internal/process/esm_loader.js index 75808335b90786..9a04e094e001c4 100644 --- a/lib/internal/process/esm_loader.js +++ b/lib/internal/process/esm_loader.js @@ -16,7 +16,7 @@ const esmLoader = new ESMLoader(); exports.esmLoader = esmLoader; // Module.runMain() causes loadESM() to re-run (which it should do); however, this should NOT cause -// ESM to be re-initialised; doing so causes duplicate custom loaders to be added to the public +// ESM to be re-initialized; doing so causes duplicate custom loaders to be added to the public // esmLoader. let isESMInitialized = false; @@ -80,7 +80,7 @@ function loadModulesInIsolation(parentURL, specifiers, loaders = []) { internalEsmLoader.addCustomLoaders(loaders); internalEsmLoader.preload(); - // Importation must be handled by internal loader to avoid poluting userland + // Importation must be handled by internal loader to avoid polluting userland return internalEsmLoader.import( specifiers, parentURL,