-
Notifications
You must be signed in to change notification settings - Fork 30.4k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
workers: fix invalid exit code in parent upon uncaught exception #21713
Changes from 2 commits
483cdec
180abec
004c004
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -445,6 +445,10 @@ function setupChild(evalScript) { | |
|
||
function fatalException(error) { | ||
debug(`[${threadId}] gets fatal exception`); | ||
|
||
// set exit code for local worker's 'exit' listeners | ||
process.exitCode = 1; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Copying the comment because Github currently hides it: I think behaviour should match the main thread’s behaviour as closely as possible – in there, we have that process.on("uncaughtException", () => console.log(process.exitCode));
throw new Error(); prints I really think we just need to set There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That makes sense, but we should probably consider setting the code before calling handler (in both here and for process), to avoid current situation where it emits code 1 but There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @lundibundi I think that makes sense, yes. But since it affects all Node.js code, not just Workers, we probably should do it in a separate (and maybe even semver-major) PR. The code you’d probably want to look at – if you want to tackle it yourself – is in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @addaleax Yeah, while looking into this one I looked though those functions (I actually first thought that exit code should be set on c++ side because we are listening on an exit callback from it). |
||
|
||
let caught = false; | ||
try { | ||
caught = originalFatalException.call(this, error); | ||
|
@@ -467,6 +471,10 @@ function setupChild(evalScript) { | |
else | ||
port.postMessage({ type: messageTypes.COULD_NOT_SERIALIZE_ERROR }); | ||
clearAsyncIdStack(); | ||
} else if (process.exitCode === 1) { | ||
// user caught an exception and didn't change the exit code -> assume it | ||
// was handled | ||
process.exitCode = 0; | ||
} | ||
} | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
// Flags: --experimental-worker | ||
'use strict'; | ||
const common = require('../common'); | ||
|
||
// This test checks that Worker has correct exit codes on parent side | ||
// in multiple situations. | ||
|
||
const assert = require('assert'); | ||
const worker = require('worker_threads'); | ||
const { Worker, isMainThread, parentPort } = worker; | ||
|
||
if (isMainThread) { | ||
parent(); | ||
} else { | ||
if (!parentPort) { | ||
console.error('Parent port must not be null'); | ||
process.exit(100); | ||
return; | ||
} | ||
parentPort.once('message', (msg) => { | ||
switch (msg) { | ||
case 'child1': | ||
return child1(); | ||
case 'child2': | ||
return child2(); | ||
case 'child3': | ||
return child3(); | ||
case 'child4': | ||
return child4(); | ||
case 'child5': | ||
return child5(); | ||
case 'child6': | ||
return child6(); | ||
case 'child7': | ||
return child7(); | ||
default: | ||
throw new Error('invalid'); | ||
} | ||
}); | ||
} | ||
|
||
function child1() { | ||
process.exitCode = 42; | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 42); | ||
}); | ||
} | ||
|
||
function child2() { | ||
process.exitCode = 99; | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 42); | ||
}); | ||
process.exit(42); | ||
} | ||
|
||
function child3() { | ||
process.exitCode = 99; | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 0); | ||
}); | ||
process.exit(0); | ||
} | ||
|
||
function child4() { | ||
process.exitCode = 99; | ||
process.on('exit', (code) => { | ||
// cannot use assert because it will be uncaughtException -> 1 exit code | ||
// that will render this test useless | ||
if (code !== 1) { | ||
console.error('wrong code! expected 1 for uncaughtException'); | ||
process.exit(99); | ||
} | ||
}); | ||
throw new Error('ok'); | ||
} | ||
|
||
function child5() { | ||
process.exitCode = 95; | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 95); | ||
process.exitCode = 99; | ||
}); | ||
} | ||
|
||
function child6() { | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 0); | ||
}); | ||
process.on('uncaughtException', common.mustCall(() => { | ||
// handle | ||
})); | ||
throw new Error('ok'); | ||
} | ||
|
||
function child7() { | ||
process.on('exit', (code) => { | ||
assert.strictEqual(code, 97); | ||
}); | ||
process.on('uncaughtException', common.mustCall(() => { | ||
process.exitCode = 97; | ||
})); | ||
throw new Error('ok'); | ||
} | ||
|
||
function parent() { | ||
const test = (arg, exit, error = null) => { | ||
const w = new Worker(__filename); | ||
w.on('exit', common.mustCall((code) => { | ||
assert.strictEqual( | ||
code, exit, | ||
`wrong exit for ${arg}\nexpected:${exit} but got:${code}`); | ||
console.log(`ok - ${arg} exited with ${exit}`); | ||
})); | ||
if (error) { | ||
w.on('error', common.mustCall((err) => { | ||
assert(error.test(err)); | ||
})); | ||
} | ||
w.postMessage(arg); | ||
}; | ||
|
||
test('child1', 42); | ||
test('child2', 42); | ||
test('child3', 0); | ||
test('child4', 1, /^Error: ok$/); | ||
test('child5', 99); | ||
test('child6', 0); | ||
test('child7', 97); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm afraid process.exitCode may be 1 even if the exception is caught.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree – I think you want to move this into the
if (!caught)
block below?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My bad, I assumed that this was unrecoverable (I thought 'uncaughtException' didn't count as recovery). Will push soon. I also added test similar to
test-process-exit-code
for workers.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think behaviour should match the main thread’s behaviour as closely as possible – in there, we have that
process.exitCode
is not set, as far as I can tell:prints
undefined
.