diff --git a/doc/api/cli.md b/doc/api/cli.md index 6b1860ecc81980..871526fcdbec61 100644 --- a/doc/api/cli.md +++ b/doc/api/cli.md @@ -523,8 +523,19 @@ application reference the transpiled code, not the original source position. `--enable-source-maps` enables caching of Source Maps and makes a best effort to report stack traces relative to the original source file. -Overriding `Error.prepareStackTrace` prevents `--enable-source-maps` from -modifying the stack trace. +Overriding `Error.prepareStackTrace` may prevent `--enable-source-maps` from +modifying the stack trace. Call and return the results of the original +`Error.prepareStackTrace` in the overriding function to modify the stack trace +with source maps. + +```js +const originalPrepareStackTrace = Error.prepareStackTrace; +Error.prepareStackTrace = (error, trace) => { + // Modify error and trace and format stack trace with + // original Error.prepareStackTrace. + return originalPrepareStackTrace(error, trace); +}; +``` Note, enabling source maps can introduce latency to your application when `Error.stack` is accessed. If you access `Error.stack` frequently diff --git a/lib/.eslintrc.yaml b/lib/.eslintrc.yaml index 3eb3b2089b0596..93ad0a986786b7 100644 --- a/lib/.eslintrc.yaml +++ b/lib/.eslintrc.yaml @@ -23,7 +23,7 @@ rules: message: Use an error exported by the internal/errors module. - selector: CallExpression[callee.object.name='Error'][callee.property.name='captureStackTrace'] message: Please use `require('internal/errors').hideStackFrames()` instead. - - selector: AssignmentExpression:matches([left.name='prepareStackTrace'], [left.property.name='prepareStackTrace']) + - selector: AssignmentExpression:matches([left.object.name='Error']):matches([left.name='prepareStackTrace'], [left.property.name='prepareStackTrace']) message: Use 'overrideStackTrace' from 'lib/internal/errors.js' instead of 'Error.prepareStackTrace'. - selector: ThrowStatement > NewExpression[callee.name=/^ERR_[A-Z_]+$/] > ObjectExpression:first-child:not(:has([key.name='message']):has([key.name='code']):has([key.name='syscall'])) message: The context passed into SystemError constructor must have .code, .syscall and .message. diff --git a/lib/internal/bootstrap/realm.js b/lib/internal/bootstrap/realm.js index 6034af9a36003c..c5379894fc8b35 100644 --- a/lib/internal/bootstrap/realm.js +++ b/lib/internal/bootstrap/realm.js @@ -449,7 +449,8 @@ function setupPrepareStackTrace() { setPrepareStackTraceCallback, } = internalBinding('errors'); const { - prepareStackTrace, + prepareStackTraceCallback, + ErrorPrepareStackTrace, fatalExceptionStackEnhancers: { beforeInspector, afterInspector, @@ -457,9 +458,17 @@ function setupPrepareStackTrace() { } = requireBuiltin('internal/errors'); // Tell our PrepareStackTraceCallback passed to the V8 API // to call prepareStackTrace(). - setPrepareStackTraceCallback(prepareStackTrace); + setPrepareStackTraceCallback(prepareStackTraceCallback); // Set the function used to enhance the error stack for printing setEnhanceStackForFatalException(beforeInspector, afterInspector); + // Setup the default Error.prepareStackTrace. + ObjectDefineProperty(Error, 'prepareStackTrace', { + __proto__: null, + writable: true, + enumerable: false, + configurable: true, + value: ErrorPrepareStackTrace, + }); } // Store the internal loaders in C++. diff --git a/lib/internal/errors.js b/lib/internal/errors.js index 39e512762a470c..984ad7d6538981 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -85,21 +85,14 @@ const kTypes = [ const MainContextError = Error; const overrideStackTrace = new SafeWeakMap(); -const kNoOverride = Symbol('kNoOverride'); - -const prepareStackTrace = (globalThis, error, trace) => { - // API for node internals to override error stack formatting - // without interfering with userland code. - if (overrideStackTrace.has(error)) { - const f = overrideStackTrace.get(error); - overrideStackTrace.delete(error); - return f(error, trace); - } - - const globalOverride = - maybeOverridePrepareStackTrace(globalThis, error, trace); - if (globalOverride !== kNoOverride) return globalOverride; +let internalPrepareStackTrace = defaultPrepareStackTrace; +/** + * The default implementation of `Error.prepareStackTrace` with simple + * concatenation of stack frames. + * Read more about `Error.prepareStackTrace` at https://v8.dev/docs/stack-trace-api#customizing-stack-traces. + */ +function defaultPrepareStackTrace(error, trace) { // Normal error formatting: // // Error: Message @@ -115,9 +108,35 @@ const prepareStackTrace = (globalThis, error, trace) => { return errorString; } return `${errorString}\n at ${ArrayPrototypeJoin(trace, '\n at ')}`; -}; +} + +function setInternalPrepareStackTrace(callback) { + internalPrepareStackTrace = callback; +} + +/** + * Every realm has its own prepareStackTraceCallback. When `error.stack` is + * accessed, if the error is created in a shadow realm, the shadow realm's + * prepareStackTraceCallback is invoked. Otherwise, the principal realm's + * prepareStackTraceCallback is invoked. Note that accessing `error.stack` + * of error objects created in a VM Context will always invoke the + * prepareStackTraceCallback of the principal realm. + * @param {object} globalThis The global object of the realm that the error was + * created in. When the error object is created in a VM Context, this is the + * global object of that VM Context. + * @param {object} error The error object. + * @param {CallSite[]} trace An array of CallSite objects, read more at https://v8.dev/docs/stack-trace-api#customizing-stack-traces. + * @returns {string} + */ +function prepareStackTraceCallback(globalThis, error, trace) { + // API for node internals to override error stack formatting + // without interfering with userland code. + if (overrideStackTrace.has(error)) { + const f = overrideStackTrace.get(error); + overrideStackTrace.delete(error); + return f(error, trace); + } -const maybeOverridePrepareStackTrace = (globalThis, error, trace) => { // Polyfill of V8's Error.prepareStackTrace API. // https://crbug.com/v8/7848 // `globalThis` is the global that contains the constructor which @@ -132,8 +151,17 @@ const maybeOverridePrepareStackTrace = (globalThis, error, trace) => { return MainContextError.prepareStackTrace(error, trace); } - return kNoOverride; -}; + // If the Error.prepareStackTrace was not a function, fallback to the + // internal implementation. + return internalPrepareStackTrace(error, trace); +} + +/** + * The default Error.prepareStackTrace implementation. + */ +function ErrorPrepareStackTrace(error, trace) { + return internalPrepareStackTrace(error, trace); +} const aggregateTwoErrors = (innerError, outerError) => { if (innerError && outerError && innerError !== outerError) { @@ -1055,10 +1083,11 @@ module.exports = { isStackOverflowError, kEnhanceStackBeforeInspector, kIsNodeError, - kNoOverride, - maybeOverridePrepareStackTrace, + defaultPrepareStackTrace, + setInternalPrepareStackTrace, overrideStackTrace, - prepareStackTrace, + prepareStackTraceCallback, + ErrorPrepareStackTrace, setArrowMessage, SystemError, uvErrmapGet, diff --git a/lib/internal/source_map/prepare_stack_trace.js b/lib/internal/source_map/prepare_stack_trace.js index 860564f4a34ab5..568e86f7faaf74 100644 --- a/lib/internal/source_map/prepare_stack_trace.js +++ b/lib/internal/source_map/prepare_stack_trace.js @@ -19,9 +19,6 @@ const { getStringWidth } = require('internal/util/inspect'); const { readFileSync } = require('fs'); const { findSourceMap } = require('internal/source_map/source_map_cache'); const { - kNoOverride, - overrideStackTrace, - maybeOverridePrepareStackTrace, kIsNodeError, } = require('internal/errors'); const { fileURLToPath } = require('internal/url'); @@ -29,20 +26,7 @@ const { setGetSourceMapErrorSource } = internalBinding('errors'); // Create a prettified stacktrace, inserting context from source maps // if possible. -const prepareStackTrace = (globalThis, error, trace) => { - // API for node internals to override error stack formatting - // without interfering with userland code. - // TODO(bcoe): add support for source-maps to repl. - if (overrideStackTrace.has(error)) { - const f = overrideStackTrace.get(error); - overrideStackTrace.delete(error); - return f(error, trace); - } - - const globalOverride = - maybeOverridePrepareStackTrace(globalThis, error, trace); - if (globalOverride !== kNoOverride) return globalOverride; - +function prepareStackTraceWithSourceMaps(error, trace) { let errorString; if (kIsNodeError in error) { errorString = `${error.name} [${error.code}]: ${error.message}`; @@ -57,7 +41,7 @@ const prepareStackTrace = (globalThis, error, trace) => { let lastSourceMap; let lastFileName; const preparedTrace = ArrayPrototypeJoin(ArrayPrototypeMap(trace, (t, i) => { - const str = i !== 0 ? '\n at ' : ''; + const str = '\n at '; try { // A stack trace will often have several call sites in a row within the // same file, cache the source map and file content accordingly: @@ -106,8 +90,8 @@ const prepareStackTrace = (globalThis, error, trace) => { } return `${str}${t}`; }), ''); - return `${errorString}\n at ${preparedTrace}`; -}; + return `${errorString}${preparedTrace}`; +} // Transpilers may have removed the original symbol name used in the stack // trace, if possible restore it from the names field of the source map: @@ -210,5 +194,5 @@ function getSourceMapErrorSource(fileName, lineNumber, columnNumber) { setGetSourceMapErrorSource(getSourceMapErrorSource); module.exports = { - prepareStackTrace, + prepareStackTraceWithSourceMaps, }; diff --git a/lib/internal/source_map/source_map_cache.js b/lib/internal/source_map/source_map_cache.js index 2813da21dfdc63..8a01ce154052a8 100644 --- a/lib/internal/source_map/source_map_cache.js +++ b/lib/internal/source_map/source_map_cache.js @@ -19,8 +19,10 @@ let debug = require('internal/util/debuglog').debuglog('source_map', (fn) => { const { validateBoolean } = require('internal/validators'); const { setSourceMapsEnabled: setSourceMapsNative, - setPrepareStackTraceCallback, } = internalBinding('errors'); +const { + setInternalPrepareStackTrace, +} = require('internal/errors'); const { getLazy } = require('internal/util'); // Since the CJS module cache is mutable, which leads to memory leaks when @@ -55,15 +57,15 @@ function setSourceMapsEnabled(val) { setSourceMapsNative(val); if (val) { const { - prepareStackTrace, + prepareStackTraceWithSourceMaps, } = require('internal/source_map/prepare_stack_trace'); - setPrepareStackTraceCallback(prepareStackTrace); + setInternalPrepareStackTrace(prepareStackTraceWithSourceMaps); } else if (sourceMapsEnabled !== undefined) { // Reset prepare stack trace callback only when disabling source maps. const { - prepareStackTrace, + defaultPrepareStackTrace, } = require('internal/errors'); - setPrepareStackTraceCallback(prepareStackTrace); + setInternalPrepareStackTrace(defaultPrepareStackTrace); } sourceMapsEnabled = val; diff --git a/lib/repl.js b/lib/repl.js index 3029c94b1e1ac0..3effdc7bb82d41 100644 --- a/lib/repl.js +++ b/lib/repl.js @@ -45,7 +45,7 @@ const { ArrayPrototypeAt, ArrayPrototypeFilter, - ArrayPrototypeFindIndex, + ArrayPrototypeFindLastIndex, ArrayPrototypeForEach, ArrayPrototypeIncludes, ArrayPrototypeJoin, @@ -53,15 +53,13 @@ const { ArrayPrototypePop, ArrayPrototypePush, ArrayPrototypePushApply, - ArrayPrototypeReverse, ArrayPrototypeShift, ArrayPrototypeSlice, ArrayPrototypeSome, ArrayPrototypeSort, - ArrayPrototypeSplice, ArrayPrototypeUnshift, Boolean, - Error, + Error: MainContextError, FunctionPrototypeBind, JSONStringify, MathMaxApply, @@ -153,6 +151,7 @@ const { }, isErrorStackTraceLimitWritable, overrideStackTrace, + ErrorPrepareStackTrace, } = require('internal/errors'); const { sendInspectorCommand } = require('internal/util/inspector'); const { getOptionValue } = require('internal/options'); @@ -630,10 +629,10 @@ function REPLServer(prompt, if (self.breakEvalOnSigint) { const interrupt = new Promise((resolve, reject) => { sigintListener = () => { - const tmp = Error.stackTraceLimit; - if (isErrorStackTraceLimitWritable()) Error.stackTraceLimit = 0; + const tmp = MainContextError.stackTraceLimit; + if (isErrorStackTraceLimitWritable()) MainContextError.stackTraceLimit = 0; const err = new ERR_SCRIPT_EXECUTION_INTERRUPTED(); - if (isErrorStackTraceLimitWritable()) Error.stackTraceLimit = tmp; + if (isErrorStackTraceLimitWritable()) MainContextError.stackTraceLimit = tmp; reject(err); }; prioritizedSigintQueue.add(sigintListener); @@ -679,23 +678,22 @@ function REPLServer(prompt, if (typeof stackFrames === 'object') { // Search from the bottom of the call stack to // find the first frame with a null function name - const idx = ArrayPrototypeFindIndex( - ArrayPrototypeReverse(stackFrames), + const idx = ArrayPrototypeFindLastIndex( + stackFrames, (frame) => frame.getFunctionName() === null, ); // If found, get rid of it and everything below it - frames = ArrayPrototypeSplice(stackFrames, idx + 1); + frames = ArrayPrototypeSlice(stackFrames, 0, idx); } else { frames = stackFrames; } // FIXME(devsnek): this is inconsistent with the checks // that the real prepareStackTrace dispatch uses in // lib/internal/errors.js. - if (typeof Error.prepareStackTrace === 'function') { - return Error.prepareStackTrace(error, frames); + if (typeof MainContextError.prepareStackTrace === 'function') { + return MainContextError.prepareStackTrace(error, frames); } - ArrayPrototypePush(frames, error); - return ArrayPrototypeJoin(ArrayPrototypeReverse(frames), '\n at '); + return ErrorPrepareStackTrace(error, frames); }); decorateErrorStack(e); diff --git a/test/fixtures/source-map/output/source_map_prepare_stack_trace.js b/test/fixtures/source-map/output/source_map_prepare_stack_trace.js new file mode 100644 index 00000000000000..d49bad116b479f --- /dev/null +++ b/test/fixtures/source-map/output/source_map_prepare_stack_trace.js @@ -0,0 +1,34 @@ +// Flags: --enable-source-maps + +'use strict'; +require('../../../common'); +const assert = require('assert'); +Error.stackTraceLimit = 5; + +assert.strictEqual(typeof Error.prepareStackTrace, 'function'); +const defaultPrepareStackTrace = Error.prepareStackTrace; +Error.prepareStackTrace = (error, trace) => { + trace = trace.filter(it => { + return it.getFunctionName() !== 'functionC'; + }); + return defaultPrepareStackTrace(error, trace); +}; + +try { + require('../enclosing-call-site-min.js'); +} catch (e) { + console.log(e); +} + +delete require.cache[require + .resolve('../enclosing-call-site-min.js')]; + +// Disable +process.setSourceMapsEnabled(false); +assert.strictEqual(process.sourceMapsEnabled, false); + +try { + require('../enclosing-call-site-min.js'); +} catch (e) { + console.log(e); +} diff --git a/test/fixtures/source-map/output/source_map_prepare_stack_trace.snapshot b/test/fixtures/source-map/output/source_map_prepare_stack_trace.snapshot new file mode 100644 index 00000000000000..9e9740a26fc34e --- /dev/null +++ b/test/fixtures/source-map/output/source_map_prepare_stack_trace.snapshot @@ -0,0 +1,10 @@ +Error: an error! + at functionD (*enclosing-call-site.js:16:17) + at functionB (*enclosing-call-site.js:6:3) + at functionA (*enclosing-call-site.js:2:3) + at Object. (*enclosing-call-site.js:24:3) +Error: an error! + at functionD (*enclosing-call-site-min.js:1:156) + at functionB (*enclosing-call-site-min.js:1:60) + at functionA (*enclosing-call-site-min.js:1:26) + at Object. (*enclosing-call-site-min.js:1:199) diff --git a/test/parallel/test-error-prepare-stack-trace.js b/test/parallel/test-error-prepare-stack-trace.js index 28ecdd25f50135..0d375a6db38675 100644 --- a/test/parallel/test-error-prepare-stack-trace.js +++ b/test/parallel/test-error-prepare-stack-trace.js @@ -1,10 +1,13 @@ -// Flags: --enable-source-maps 'use strict'; require('../common'); +const { spawnSyncAndExitWithoutError } = require('../common/child_process'); const assert = require('assert'); -// Error.prepareStackTrace() can be overridden with source maps enabled. +// Verify that the default Error.prepareStackTrace is present. +assert.strictEqual(typeof Error.prepareStackTrace, 'function'); + +// Error.prepareStackTrace() can be overridden. { let prepareCalled = false; Error.prepareStackTrace = (_error, trace) => { @@ -17,3 +20,12 @@ const assert = require('assert'); } assert(prepareCalled); } + +if (process.argv[2] !== 'child') { + // Verify that the above test still passes when source-maps support is + // enabled. + spawnSyncAndExitWithoutError( + process.execPath, + ['--enable-source-maps', __filename, 'child'], + {}); +} diff --git a/test/parallel/test-node-output-sourcemaps.mjs b/test/parallel/test-node-output-sourcemaps.mjs index b01f30765c7de8..21060c3b342cf1 100644 --- a/test/parallel/test-node-output-sourcemaps.mjs +++ b/test/parallel/test-node-output-sourcemaps.mjs @@ -31,6 +31,7 @@ describe('sourcemaps output', { concurrency: true }, () => { { name: 'source-map/output/source_map_enclosing_function.js' }, { name: 'source-map/output/source_map_eval.js' }, { name: 'source-map/output/source_map_no_source_file.js' }, + { name: 'source-map/output/source_map_prepare_stack_trace.js' }, { name: 'source-map/output/source_map_reference_error_tabs.js' }, { name: 'source-map/output/source_map_sourcemapping_url_string.js' }, { name: 'source-map/output/source_map_throw_catch.js' }, diff --git a/test/parallel/test-repl-pretty-custom-stack.js b/test/parallel/test-repl-pretty-custom-stack.js index a10cd032a688c4..f5697c2362e261 100644 --- a/test/parallel/test-repl-pretty-custom-stack.js +++ b/test/parallel/test-repl-pretty-custom-stack.js @@ -42,8 +42,9 @@ const origPrepareStackTrace = Error.prepareStackTrace; Error.prepareStackTrace = (err, stack) => { if (err instanceof SyntaxError) return err.toString(); - stack.push(err); - return stack.reverse().join('--->\n'); + // Insert the error at the beginning of the stack + stack.unshift(err); + return stack.join('--->\n'); }; process.on('uncaughtException', (e) => { @@ -78,3 +79,10 @@ const tests = [ ]; tests.forEach(run); + +// Verify that the stack can be generated when Error.prepareStackTrace is deleted. +delete Error.prepareStackTrace; +run({ + command: 'throw new TypeError(\'Whoops!\')', + expected: 'Uncaught TypeError: Whoops!\n' +});