mirror of
https://github.com/nodejs/node.git
synced 2025-05-06 16:46:56 +00:00

This makes sure that errors that shut down the application are inspected with `util.inspect()`. That makes sure that all extra properties on the error will be visible and also that the stack trace is highlighted (Node.js internal frames will be grey and node modules are underlined). PR-URL: https://github.com/nodejs/node/pull/27243 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Michaël Zasso <targos@protonmail.com> Reviewed-By: Rich Trott <rtrott@gmail.com>
211 lines
6.6 KiB
JavaScript
211 lines
6.6 KiB
JavaScript
'use strict';
|
|
|
|
const { JSON } = primordials;
|
|
|
|
const path = require('path');
|
|
|
|
const {
|
|
codes: {
|
|
ERR_INVALID_ARG_TYPE,
|
|
ERR_UNCAUGHT_EXCEPTION_CAPTURE_ALREADY_SET
|
|
}
|
|
} = require('internal/errors');
|
|
|
|
const {
|
|
executionAsyncId,
|
|
clearDefaultTriggerAsyncId,
|
|
clearAsyncIdStack,
|
|
hasAsyncIdStack,
|
|
afterHooksExist,
|
|
emitAfter
|
|
} = require('internal/async_hooks');
|
|
|
|
// shouldAbortOnUncaughtToggle is a typed array for faster
|
|
// communication with JS.
|
|
const { shouldAbortOnUncaughtToggle } = internalBinding('util');
|
|
|
|
function tryGetCwd() {
|
|
try {
|
|
return process.cwd();
|
|
} catch {
|
|
// getcwd(3) can fail if the current working directory has been deleted.
|
|
// Fall back to the directory name of the (absolute) executable path.
|
|
// It's not really correct but what are the alternatives?
|
|
return path.dirname(process.execPath);
|
|
}
|
|
}
|
|
|
|
function evalModule(source, printResult) {
|
|
const { decorateErrorStack } = require('internal/util');
|
|
const asyncESM = require('internal/process/esm_loader');
|
|
const { kStdout, kStderr, print } = require('internal/util/print');
|
|
asyncESM.loaderPromise.then(async (loader) => {
|
|
const { result } = await loader.eval(source);
|
|
if (printResult) { print(kStdout, result); }
|
|
})
|
|
.catch((e) => {
|
|
decorateErrorStack(e);
|
|
print(kStderr, e);
|
|
process.exit(1);
|
|
});
|
|
}
|
|
|
|
function evalScript(name, body, breakFirstLine, printResult) {
|
|
const CJSModule = require('internal/modules/cjs/loader');
|
|
const { kVmBreakFirstLineSymbol } = require('internal/util');
|
|
|
|
const cwd = tryGetCwd();
|
|
const origModule = global.module; // Set e.g. when called from the REPL.
|
|
|
|
const module = new CJSModule(name);
|
|
module.filename = path.join(cwd, name);
|
|
module.paths = CJSModule._nodeModulePaths(cwd);
|
|
global.kVmBreakFirstLineSymbol = kVmBreakFirstLineSymbol;
|
|
const script = `
|
|
global.__filename = ${JSON.stringify(name)};
|
|
global.exports = exports;
|
|
global.module = module;
|
|
global.__dirname = __dirname;
|
|
global.require = require;
|
|
const { kVmBreakFirstLineSymbol } = global;
|
|
delete global.kVmBreakFirstLineSymbol;
|
|
return require("vm").runInThisContext(
|
|
${JSON.stringify(body)}, {
|
|
filename: ${JSON.stringify(name)},
|
|
displayErrors: true,
|
|
[kVmBreakFirstLineSymbol]: ${!!breakFirstLine}
|
|
});\n`;
|
|
const result = module._compile(script, `${name}-wrapper`);
|
|
if (printResult) {
|
|
const { kStdout, print } = require('internal/util/print');
|
|
print(kStdout, result);
|
|
}
|
|
|
|
if (origModule !== undefined)
|
|
global.module = origModule;
|
|
}
|
|
|
|
const exceptionHandlerState = { captureFn: null };
|
|
|
|
function setUncaughtExceptionCaptureCallback(fn) {
|
|
if (fn === null) {
|
|
exceptionHandlerState.captureFn = fn;
|
|
shouldAbortOnUncaughtToggle[0] = 1;
|
|
return;
|
|
}
|
|
if (typeof fn !== 'function') {
|
|
throw new ERR_INVALID_ARG_TYPE('fn', ['Function', 'null'], fn);
|
|
}
|
|
if (exceptionHandlerState.captureFn !== null) {
|
|
throw new ERR_UNCAUGHT_EXCEPTION_CAPTURE_ALREADY_SET();
|
|
}
|
|
exceptionHandlerState.captureFn = fn;
|
|
shouldAbortOnUncaughtToggle[0] = 0;
|
|
}
|
|
|
|
function hasUncaughtExceptionCaptureCallback() {
|
|
return exceptionHandlerState.captureFn !== null;
|
|
}
|
|
|
|
function noop() {}
|
|
|
|
// XXX(joyeecheung): for some reason this cannot be defined at the top-level
|
|
// and exported to be written to process._fatalException, it has to be
|
|
// returned as an *anonymous function* wrapped inside a factory function,
|
|
// otherwise it breaks the test-timers.setInterval async hooks test -
|
|
// this may indicate that node::FatalException should fix up the callback scope
|
|
// before calling into process._fatalException, or this function should
|
|
// take extra care of the async hooks before it schedules a setImmediate.
|
|
function createFatalException() {
|
|
return (er, fromPromise) => {
|
|
// It's possible that defaultTriggerAsyncId was set for a constructor
|
|
// call that threw and was never cleared. So clear it now.
|
|
clearDefaultTriggerAsyncId();
|
|
|
|
// If diagnostic reporting is enabled, call into its handler to see
|
|
// whether it is interested in handling the situation.
|
|
// Ignore if the error is scoped inside a domain.
|
|
// use == in the checks as we want to allow for null and undefined
|
|
if (er == null || er.domain == null) {
|
|
try {
|
|
const report = internalBinding('report');
|
|
if (report != null && report.shouldReportOnUncaughtException()) {
|
|
report.writeReport(er ? er.message : 'Exception',
|
|
'Exception',
|
|
null,
|
|
er ? er.stack : undefined);
|
|
}
|
|
} catch {} // Ignore the exception. Diagnostic reporting is unavailable.
|
|
}
|
|
|
|
const type = fromPromise ? 'unhandledRejection' : 'uncaughtException';
|
|
if (exceptionHandlerState.captureFn !== null) {
|
|
exceptionHandlerState.captureFn(er);
|
|
} else if (!process.emit('uncaughtException', er, type)) {
|
|
// If someone handled it, then great. Otherwise, die in C++ land
|
|
// since that means that we'll exit the process, emit the 'exit' event.
|
|
const { inspect } = require('internal/util/inspect');
|
|
const colors = internalBinding('util').guessHandleType(2) === 'TTY' &&
|
|
require('internal/tty').hasColors() ||
|
|
inspect.defaultOptions.colors;
|
|
try {
|
|
if (!process._exiting) {
|
|
process._exiting = true;
|
|
process.exitCode = 1;
|
|
process.emit('exit', 1);
|
|
}
|
|
} catch {
|
|
// Nothing to be done about it at this point.
|
|
}
|
|
try {
|
|
const { kExpandStackSymbol } = require('internal/util');
|
|
if (typeof er[kExpandStackSymbol] === 'function')
|
|
er[kExpandStackSymbol]();
|
|
er.stack = inspect(er, { colors });
|
|
} catch {
|
|
// Nothing to be done about it at this point.
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// If we handled an error, then make sure any ticks get processed
|
|
// by ensuring that the next Immediate cycle isn't empty.
|
|
require('timers').setImmediate(noop);
|
|
|
|
// Emit the after() hooks now that the exception has been handled.
|
|
if (afterHooksExist()) {
|
|
do {
|
|
emitAfter(executionAsyncId());
|
|
} while (hasAsyncIdStack());
|
|
// Or completely empty the id stack.
|
|
} else {
|
|
clearAsyncIdStack();
|
|
}
|
|
|
|
return true;
|
|
};
|
|
}
|
|
|
|
function readStdin(callback) {
|
|
process.stdin.setEncoding('utf8');
|
|
|
|
let code = '';
|
|
process.stdin.on('data', (d) => {
|
|
code += d;
|
|
});
|
|
|
|
process.stdin.on('end', () => {
|
|
callback(code);
|
|
});
|
|
}
|
|
|
|
module.exports = {
|
|
readStdin,
|
|
tryGetCwd,
|
|
evalModule,
|
|
evalScript,
|
|
fatalException: createFatalException(),
|
|
setUncaughtExceptionCaptureCallback,
|
|
hasUncaughtExceptionCaptureCallback
|
|
};
|