mirror of
https://github.com/nodejs/node.git
synced 2025-04-28 13:40:37 +00:00

Files in `node_modules` are not authored by the user directly and the original sources are less relevant to the user. Skipping source maps in `node_modules` improves the general performance. Add `module.setSourceMapsSupport(enabled, options)` to skip source maps in `node_modules` if it is needed. This moves all source maps related API to `node:module` and this a step to promote the source maps API to stable. PR-URL: https://github.com/nodejs/node/pull/56639 Reviewed-By: Vinícius Lourenço Claro Cardoso <contact@viniciusl.com.br> Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com> Reviewed-By: Matteo Collina <matteo.collina@gmail.com> Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
405 lines
15 KiB
JavaScript
405 lines
15 KiB
JavaScript
'use strict';
|
|
|
|
const {
|
|
Array,
|
|
ArrayPrototypeJoin,
|
|
ArrayPrototypeSome,
|
|
FunctionPrototype,
|
|
ObjectSetPrototypeOf,
|
|
PromisePrototypeThen,
|
|
PromiseResolve,
|
|
RegExpPrototypeExec,
|
|
RegExpPrototypeSymbolReplace,
|
|
SafePromiseAllReturnArrayLike,
|
|
SafePromiseAllReturnVoid,
|
|
SafeSet,
|
|
StringPrototypeIncludes,
|
|
StringPrototypeSplit,
|
|
StringPrototypeStartsWith,
|
|
globalThis,
|
|
} = primordials;
|
|
let debug = require('internal/util/debuglog').debuglog('esm', (fn) => {
|
|
debug = fn;
|
|
});
|
|
|
|
const { ModuleWrap, kInstantiated } = internalBinding('module_wrap');
|
|
const {
|
|
privateSymbols: {
|
|
entry_point_module_private_symbol,
|
|
},
|
|
} = internalBinding('util');
|
|
const { decorateErrorStack, kEmptyObject } = require('internal/util');
|
|
const {
|
|
getSourceMapsSupport,
|
|
} = require('internal/source_map/source_map_cache');
|
|
const assert = require('internal/assert');
|
|
const resolvedPromise = PromiseResolve();
|
|
const {
|
|
setHasStartedUserESMExecution,
|
|
} = require('internal/modules/helpers');
|
|
const { getOptionValue } = require('internal/options');
|
|
const noop = FunctionPrototype;
|
|
const {
|
|
ERR_REQUIRE_ASYNC_MODULE,
|
|
} = require('internal/errors').codes;
|
|
let hasPausedEntry = false;
|
|
|
|
const CJSGlobalLike = [
|
|
'require',
|
|
'module',
|
|
'exports',
|
|
'__filename',
|
|
'__dirname',
|
|
];
|
|
const isCommonJSGlobalLikeNotDefinedError = (errorMessage) =>
|
|
ArrayPrototypeSome(
|
|
CJSGlobalLike,
|
|
(globalLike) => errorMessage === `${globalLike} is not defined`,
|
|
);
|
|
|
|
class ModuleJobBase {
|
|
constructor(url, importAttributes, isMain, inspectBrk) {
|
|
this.importAttributes = importAttributes;
|
|
this.isMain = isMain;
|
|
this.inspectBrk = inspectBrk;
|
|
|
|
this.url = url;
|
|
}
|
|
}
|
|
|
|
/* A ModuleJob tracks the loading of a single Module, and the ModuleJobs of
|
|
* its dependencies, over time. */
|
|
class ModuleJob extends ModuleJobBase {
|
|
#loader = null;
|
|
|
|
/**
|
|
* @param {ModuleLoader} loader The ESM loader.
|
|
* @param {string} url URL of the module to be wrapped in ModuleJob.
|
|
* @param {ImportAttributes} importAttributes Import attributes from the import statement.
|
|
* @param {ModuleWrap|Promise<ModuleWrap>} moduleOrModulePromise Translated ModuleWrap for the module.
|
|
* @param {boolean} isMain Whether the module is the entry point.
|
|
* @param {boolean} inspectBrk Whether this module should be evaluated with the
|
|
* first line paused in the debugger (because --inspect-brk is passed).
|
|
* @param {boolean} isForRequireInImportedCJS Whether this is created for require() in imported CJS.
|
|
*/
|
|
constructor(loader, url, importAttributes = { __proto__: null },
|
|
moduleOrModulePromise, isMain, inspectBrk, isForRequireInImportedCJS = false) {
|
|
super(url, importAttributes, isMain, inspectBrk);
|
|
this.#loader = loader;
|
|
|
|
// Expose the promise to the ModuleWrap directly for linking below.
|
|
if (isForRequireInImportedCJS) {
|
|
this.module = moduleOrModulePromise;
|
|
assert(this.module instanceof ModuleWrap);
|
|
this.modulePromise = PromiseResolve(this.module);
|
|
} else {
|
|
this.modulePromise = moduleOrModulePromise;
|
|
}
|
|
|
|
// Promise for the list of all dependencyJobs.
|
|
this.linked = this._link();
|
|
// This promise is awaited later anyway, so silence
|
|
// 'unhandled rejection' warnings.
|
|
PromisePrototypeThen(this.linked, undefined, noop);
|
|
|
|
// instantiated == deep dependency jobs wrappers are instantiated,
|
|
// and module wrapper is instantiated.
|
|
this.instantiated = undefined;
|
|
}
|
|
|
|
/**
|
|
* Iterates the module requests and links with the loader.
|
|
* @returns {Promise<ModuleJob[]>} Dependency module jobs.
|
|
*/
|
|
async _link() {
|
|
this.module = await this.modulePromise;
|
|
assert(this.module instanceof ModuleWrap);
|
|
|
|
const moduleRequests = this.module.getModuleRequests();
|
|
// Explicitly keeping track of dependency jobs is needed in order
|
|
// to flatten out the dependency graph below in `_instantiate()`,
|
|
// so that circular dependencies can't cause a deadlock by two of
|
|
// these `link` callbacks depending on each other.
|
|
// Create an ArrayLike to avoid calling into userspace with `.then`
|
|
// when returned from the async function.
|
|
const dependencyJobs = Array(moduleRequests.length);
|
|
ObjectSetPrototypeOf(dependencyJobs, null);
|
|
|
|
// Specifiers should be aligned with the moduleRequests array in order.
|
|
const specifiers = Array(moduleRequests.length);
|
|
const modulePromises = Array(moduleRequests.length);
|
|
// Iterate with index to avoid calling into userspace with `Symbol.iterator`.
|
|
for (let idx = 0; idx < moduleRequests.length; idx++) {
|
|
const { specifier, attributes } = moduleRequests[idx];
|
|
// TODO(joyeecheung): resolve all requests first, then load them in another
|
|
// loop so that hooks can pre-fetch sources off-thread.
|
|
const dependencyJobPromise = this.#loader.getModuleJobForImport(
|
|
specifier, this.url, attributes,
|
|
);
|
|
const modulePromise = PromisePrototypeThen(dependencyJobPromise, (job) => {
|
|
debug(`async link() ${this.url} -> ${specifier}`, job);
|
|
dependencyJobs[idx] = job;
|
|
return job.modulePromise;
|
|
});
|
|
modulePromises[idx] = modulePromise;
|
|
specifiers[idx] = specifier;
|
|
}
|
|
|
|
const modules = await SafePromiseAllReturnArrayLike(modulePromises);
|
|
this.module.link(specifiers, modules);
|
|
|
|
return dependencyJobs;
|
|
}
|
|
|
|
instantiate() {
|
|
if (this.instantiated === undefined) {
|
|
this.instantiated = this._instantiate();
|
|
}
|
|
return this.instantiated;
|
|
}
|
|
|
|
async _instantiate() {
|
|
const jobsInGraph = new SafeSet();
|
|
const addJobsToDependencyGraph = async (moduleJob) => {
|
|
debug(`async addJobsToDependencyGraph() ${this.url}`, moduleJob);
|
|
|
|
if (jobsInGraph.has(moduleJob)) {
|
|
return;
|
|
}
|
|
jobsInGraph.add(moduleJob);
|
|
const dependencyJobs = await moduleJob.linked;
|
|
return SafePromiseAllReturnVoid(dependencyJobs, addJobsToDependencyGraph);
|
|
};
|
|
await addJobsToDependencyGraph(this);
|
|
|
|
try {
|
|
if (!hasPausedEntry && this.inspectBrk) {
|
|
hasPausedEntry = true;
|
|
const initWrapper = internalBinding('inspector').callAndPauseOnStart;
|
|
initWrapper(this.module.instantiate, this.module);
|
|
} else {
|
|
this.module.instantiate();
|
|
}
|
|
} catch (e) {
|
|
decorateErrorStack(e);
|
|
// TODO(@bcoe): Add source map support to exception that occurs as result
|
|
// of missing named export. This is currently not possible because
|
|
// stack trace originates in module_job, not the file itself. A hidden
|
|
// symbol with filename could be set in node_errors.cc to facilitate this.
|
|
if (!getSourceMapsSupport().enabled &&
|
|
StringPrototypeIncludes(e.message,
|
|
' does not provide an export named')) {
|
|
const splitStack = StringPrototypeSplit(e.stack, '\n');
|
|
const parentFileUrl = RegExpPrototypeSymbolReplace(
|
|
/:\d+$/,
|
|
splitStack[0],
|
|
'',
|
|
);
|
|
const { 1: childSpecifier, 2: name } = RegExpPrototypeExec(
|
|
/module '(.*)' does not provide an export named '(.+)'/,
|
|
e.message);
|
|
const { url: childFileURL } = await this.#loader.resolve(
|
|
childSpecifier,
|
|
parentFileUrl,
|
|
kEmptyObject,
|
|
);
|
|
let format;
|
|
try {
|
|
// This might throw for non-CommonJS modules because we aren't passing
|
|
// in the import attributes and some formats require them; but we only
|
|
// care about CommonJS for the purposes of this error message.
|
|
({ format } =
|
|
await this.#loader.load(childFileURL));
|
|
} catch {
|
|
// Continue regardless of error.
|
|
}
|
|
|
|
if (format === 'commonjs') {
|
|
const importStatement = splitStack[1];
|
|
// TODO(@ctavan): The original error stack only provides the single
|
|
// line which causes the error. For multi-line import statements we
|
|
// cannot generate an equivalent object destructuring assignment by
|
|
// just parsing the error stack.
|
|
const oneLineNamedImports = RegExpPrototypeExec(/{.*}/, importStatement);
|
|
const destructuringAssignment = oneLineNamedImports &&
|
|
RegExpPrototypeSymbolReplace(/\s+as\s+/g, oneLineNamedImports, ': ');
|
|
e.message = `Named export '${name}' not found. The requested module` +
|
|
` '${childSpecifier}' is a CommonJS module, which may not support` +
|
|
' all module.exports as named exports.\nCommonJS modules can ' +
|
|
'always be imported via the default export, for example using:' +
|
|
`\n\nimport pkg from '${childSpecifier}';\n${
|
|
destructuringAssignment ?
|
|
`const ${destructuringAssignment} = pkg;\n` : ''}`;
|
|
const newStack = StringPrototypeSplit(e.stack, '\n');
|
|
newStack[3] = `SyntaxError: ${e.message}`;
|
|
e.stack = ArrayPrototypeJoin(newStack, '\n');
|
|
}
|
|
}
|
|
throw e;
|
|
}
|
|
|
|
for (const dependencyJob of jobsInGraph) {
|
|
// Calling `this.module.instantiate()` instantiates not only the
|
|
// ModuleWrap in this module, but all modules in the graph.
|
|
dependencyJob.instantiated = resolvedPromise;
|
|
}
|
|
}
|
|
|
|
runSync() {
|
|
assert(this.module instanceof ModuleWrap);
|
|
if (this.instantiated !== undefined) {
|
|
return { __proto__: null, module: this.module };
|
|
}
|
|
|
|
this.module.instantiate();
|
|
this.instantiated = PromiseResolve();
|
|
const timeout = -1;
|
|
const breakOnSigint = false;
|
|
setHasStartedUserESMExecution();
|
|
this.module.evaluate(timeout, breakOnSigint);
|
|
return { __proto__: null, module: this.module };
|
|
}
|
|
|
|
async run(isEntryPoint = false) {
|
|
await this.instantiate();
|
|
if (isEntryPoint) {
|
|
globalThis[entry_point_module_private_symbol] = this.module;
|
|
}
|
|
const timeout = -1;
|
|
const breakOnSigint = false;
|
|
setHasStartedUserESMExecution();
|
|
try {
|
|
await this.module.evaluate(timeout, breakOnSigint);
|
|
} catch (e) {
|
|
if (e?.name === 'ReferenceError' &&
|
|
isCommonJSGlobalLikeNotDefinedError(e.message)) {
|
|
e.message += ' in ES module scope';
|
|
|
|
if (StringPrototypeStartsWith(e.message, 'require ')) {
|
|
e.message += ', you can use import instead';
|
|
}
|
|
|
|
const packageConfig =
|
|
StringPrototypeStartsWith(this.module.url, 'file://') &&
|
|
RegExpPrototypeExec(/\.js(\?[^#]*)?(#.*)?$/, this.module.url) !== null &&
|
|
require('internal/modules/package_json_reader')
|
|
.getPackageScopeConfig(this.module.url);
|
|
if (packageConfig.type === 'module') {
|
|
e.message +=
|
|
'\nThis file is being treated as an ES module because it has a ' +
|
|
`'.js' file extension and '${packageConfig.pjsonPath}' contains ` +
|
|
'"type": "module". To treat it as a CommonJS script, rename it ' +
|
|
'to use the \'.cjs\' file extension.';
|
|
}
|
|
}
|
|
throw e;
|
|
}
|
|
return { __proto__: null, module: this.module };
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This is a fully synchronous job and does not spawn additional threads in any way.
|
|
* All the steps are ensured to be synchronous and it throws on instantiating
|
|
* an asynchronous graph. It also disallows CJS <-> ESM cycles.
|
|
*
|
|
* This is used for ES modules loaded via require(esm). Modules loaded by require() in
|
|
* imported CJS are handled by ModuleJob with the isForRequireInImportedCJS set to true instead.
|
|
* The two currently have different caching behaviors.
|
|
* TODO(joyeecheung): consolidate this with the isForRequireInImportedCJS variant of ModuleJob.
|
|
*/
|
|
class ModuleJobSync extends ModuleJobBase {
|
|
#loader = null;
|
|
|
|
/**
|
|
* @param {ModuleLoader} loader The ESM loader.
|
|
* @param {string} url URL of the module to be wrapped in ModuleJob.
|
|
* @param {ImportAttributes} importAttributes Import attributes from the import statement.
|
|
* @param {ModuleWrap} moduleWrap Translated ModuleWrap for the module.
|
|
* @param {boolean} isMain Whether the module is the entry point.
|
|
* @param {boolean} inspectBrk Whether this module should be evaluated with the
|
|
* first line paused in the debugger (because --inspect-brk is passed).
|
|
*/
|
|
constructor(loader, url, importAttributes, moduleWrap, isMain, inspectBrk) {
|
|
super(url, importAttributes, isMain, inspectBrk, true);
|
|
|
|
this.#loader = loader;
|
|
this.module = moduleWrap;
|
|
|
|
assert(this.module instanceof ModuleWrap);
|
|
// Store itself into the cache first before linking in case there are circular
|
|
// references in the linking.
|
|
loader.loadCache.set(url, importAttributes.type, this);
|
|
|
|
try {
|
|
const moduleRequests = this.module.getModuleRequests();
|
|
// Specifiers should be aligned with the moduleRequests array in order.
|
|
const specifiers = Array(moduleRequests.length);
|
|
const modules = Array(moduleRequests.length);
|
|
const jobs = Array(moduleRequests.length);
|
|
for (let i = 0; i < moduleRequests.length; ++i) {
|
|
const { specifier, attributes } = moduleRequests[i];
|
|
const job = this.#loader.getModuleJobForRequire(specifier, url, attributes);
|
|
specifiers[i] = specifier;
|
|
modules[i] = job.module;
|
|
jobs[i] = job;
|
|
}
|
|
this.module.link(specifiers, modules);
|
|
this.linked = jobs;
|
|
} finally {
|
|
// Restore it - if it succeeds, we'll reset in the caller; Otherwise it's
|
|
// not cached and if the error is caught, subsequent attempt would still fail.
|
|
loader.loadCache.delete(url, importAttributes.type);
|
|
}
|
|
}
|
|
|
|
get modulePromise() {
|
|
return PromiseResolve(this.module);
|
|
}
|
|
|
|
async run() {
|
|
// This path is hit by a require'd module that is imported again.
|
|
const status = this.module.getStatus();
|
|
if (status > kInstantiated) {
|
|
if (this.evaluationPromise) {
|
|
await this.evaluationPromise;
|
|
}
|
|
return { __proto__: null, module: this.module };
|
|
} else if (status === kInstantiated) {
|
|
// The evaluation may have been canceled because instantiateSync() detected TLA first.
|
|
// But when it is imported again, it's fine to re-evaluate it asynchronously.
|
|
const timeout = -1;
|
|
const breakOnSigint = false;
|
|
this.evaluationPromise = this.module.evaluate(timeout, breakOnSigint);
|
|
await this.evaluationPromise;
|
|
this.evaluationPromise = undefined;
|
|
return { __proto__: null, module: this.module };
|
|
}
|
|
|
|
assert.fail('Unexpected status of a module that is imported again after being required. ' +
|
|
`Status = ${status}`);
|
|
}
|
|
|
|
runSync() {
|
|
// TODO(joyeecheung): add the error decoration logic from the async instantiate.
|
|
this.module.async = this.module.instantiateSync();
|
|
// If --experimental-print-required-tla is true, proceeds to evaluation even
|
|
// if it's async because we want to search for the TLA and help users locate
|
|
// them.
|
|
// TODO(joyeecheung): track the asynchroniticy using v8::Module::HasTopLevelAwait()
|
|
// and we'll be able to throw right after compilation of the modules, using acron
|
|
// to find and print the TLA.
|
|
if (this.module.async && !getOptionValue('--experimental-print-required-tla')) {
|
|
throw new ERR_REQUIRE_ASYNC_MODULE();
|
|
}
|
|
setHasStartedUserESMExecution();
|
|
const namespace = this.module.evaluateSync();
|
|
return { __proto__: null, module: this.module, namespace };
|
|
}
|
|
}
|
|
|
|
ObjectSetPrototypeOf(ModuleJobBase.prototype, null);
|
|
module.exports = {
|
|
ModuleJob, ModuleJobSync, ModuleJobBase,
|
|
};
|