-
-
Notifications
You must be signed in to change notification settings - Fork 3k
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
feat: add option to use posix exit code upon fatal signal #4989
base: main
Are you sure you want to change the base?
Conversation
This PR hasn't had any recent activity, and I'm labeling it |
We should probably disable the stale action, pending picking up reviewing PRs. |
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.
Nice and clean implementation, thanks! ✨
Requesting changes on a few things. But let me know, please, if I'm off base here 🙂.
Note that after this lands, we'll want to file a followup issue about making this the default behavior in some future version of Mocha. |
Co-authored-by: Josh Goldberg ✨ <[email protected]>
Address PR review comments mocha org
Address PR review comments mocha org
@JoshuaKGoldberg Thanks for the review! Hopefully that latest update addresses your questions, but let me know if there's anything else you'd like to see. Looking froward to not using our own fork of mocha! 😆 |
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.
Progress! Thanks for adding in the tests. I think we'll need to capture some more cases?
Co-authored-by: Josh Goldberg ✨ <[email protected]>
Omit win32 from signals test suite
@JoshuaKGoldberg just circling back to see if there's anything you need from me to unblock this enhancement? lmk 🙏 |
Thanks for the ping! Nothing yet. The other maintainers have been swamped (so much happening in the ecosystem!) but I'm still holding onto hope someone will be able to review. I've personally time boxed this one to within June. |
Update package.json
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.
As far as I can see this only changes the flow in this flow:
Line 80 in 2f3fedc
if (mochaArgs['node-option'] || Object.keys(nodeArgs).length || hasInspect) { |
Not in the other flow:
Lines 139 to 141 in 2f3fedc
} else { | |
debug('running Mocha in-process'); | |
require('../lib/cli/cli').main([], mochaArgs); |
Is this really the right place for this fix?
Eg: The non-standard error codes are set here:
Lines 26 to 45 in 2f3fedc
const exitMochaLater = code => { | |
process.on('exit', () => { | |
process.exitCode = Math.min(code, 255); | |
}); | |
}; | |
/** | |
* Exits Mocha when Mocha itself has finished execution, regardless of | |
* what the tests or code under test is doing. | |
* @param {number} code - Exit code; typically # of failures | |
* @ignore | |
* @private | |
*/ | |
const exitMocha = code => { | |
const clampedCode = Math.min(code, 255); | |
let draining = 0; | |
// Eagerly set the process's exit code in case stream.write doesn't | |
// execute its callback before the process terminates. | |
process.exitCode = clampedCode; |
And in the old-school case where one forces the process to be killed once all tests are done it happens within that:
Line 52 in 2f3fedc
process.exit(clampedCode); |
I overall find this PR to be a mix of two changes:
- Stop mocha from abusing the exit code as an error count reporter
- Add signal specific exit codes
The latter of the two I would want to read up on more, but the first of them should be an easier fix.
Node.js should already be setting the signal exit codes, I wonder if we can piggy back on that instead? https://nodejs.org/api/process.html#exit-codes |
@voxpelli Thanks for the review. That's right, the option only applies "when mocha is run as a child process". The reasoning was to try and be as unobtrusive as possible and address specifically the problem of fatal errors (OOM et al) getting swallowed silently in a CI/CD pipeline where mocha is spawned by something like nyc. It sounds like there's general agreement that mocha shouldn't use the exit code to report the number of errors though, so I can update this PR to support the option in both cases. As for exit codes other than 0, 1 and 128+signal - I guess we'd need to know what the exit code scheme should be and could be something for another PR.... |
@73rhodes Is there a need to handle 128+signal codes manually or is that already handled by node.js itself? Reason why I ask is since its typically not something I have seen handled manually by other CLI tools built with node.js |
…rocess modes of running mocha
Support posix-exit-codes for child and in-process modes
@voxpelli ... I just updated the PR with a commit (this PR) that adds the signal-handling behavior for both the child-process and in-process modes of running mocha, as requested. As far as piggybacking on node's signal handling, I'm just working with these inputs:
and
where, based on what I've seen and implemented in the tests, we need to check the |
Just circling back here to see if there are any specific changes being requested here to move forward. We would prefer to use the original mocha package but proper signal handling is critical for us. Please let me know if there's anything I can do to unblock this, thanks! |
For me I’m still at:
And I’m also torn on shipping this as an option rather than shipping it as a breaking change in the next major – I would like Mocha to gain fewer options and less complex code base, not the opposite |
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.
One thing that making this a breaking change instead would do is that instead of still having exitMochaLater
, exitMocha
, proc.on('exit'
be getting sent the number errors, that could be cleaned up.
And the current setup of a process.on('exit'
being set in a proc.on('exit'
and then doing a process.kill
on itself – its all very messy and hard to follow and rather than adding more if-statements and more paths to it I would vote for trying to make it less messy.
@JoshuaKGoldberg shared with me how eg. vitest
handles some of this, and its much much cleaner and easier to follow: https://github.com/vitest-dev/vitest/blob/f9a628438a5462436b59dd9bdeffddada19a9e81/packages/vitest/src/node/reporters/renderers/windowedRenderer.ts#L191
const numericSignal = | ||
typeof signal === 'string' ? os.constants.signals[signal] : signal; |
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.
According to the documentation the signal
argument of the exit
event on child_process.spawn()
is always a string – when is it ever something else?
if (mochaArgs['posix-exit-codes'] === true) { | ||
process.exit(SIGNAL_OFFSET + numericSignal); | ||
} else { | ||
process.kill(process.pid, signal); | ||
} |
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.
If the process.kill(process.pid, signal)
is important because it will trigger handles like the process.on('SIGINT'
, then that's what always should be used here.
Else the process.exit()
variant should always be used.
Though the logic of Mocha here is kind of weird – the signal
comes from the child process, and the SIGNAL_OFFSET + numericSignal
should be sent if the process itself, not one of its childs, gets sent the signal
– so I would say that process.exit(SIGNAL_OFFSET + numericSignal)
here is wrong and that the original process.kill(process.pid, signal)
is weird (as its essentially bubbling up the process signal? Is that common practice?)
} else if (code !== 0 && mochaArgs['posix-exit-codes'] === true) { | ||
process.exit(EXIT_FAILURE); | ||
} else { | ||
process.exit(code); |
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.
Could probably make it this:
} else if (code !== 0 && mochaArgs['posix-exit-codes'] === true) { | |
process.exit(EXIT_FAILURE); | |
} else { | |
process.exit(code); | |
} else { | |
process.exit(Math.min(code, mochaArgs['posix-exit-codes'] ? 1 : 255)); |
That reuses the Math.min(code, 255)
logic that's in other places.
Not sure the EXIT_FAILURE
helps much here, the direct 1
makes it possible to use the Math.min
like this
const usePosixExitCodes = process.argv.includes('--posix-exit-codes'); | ||
const exitCode = | ||
usePosixExitCodes && code > 0 ? EXIT_FAILURE : Math.min(code, 255); | ||
process.exitCode = exitCode; |
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.
const usePosixExitCodes = process.argv.includes('--posix-exit-codes'); | |
const exitCode = | |
usePosixExitCodes && code > 0 ? EXIT_FAILURE : Math.min(code, 255); | |
process.exitCode = exitCode; | |
process.exitCode = Math.min(code, process.argv.includes('--posix-exit-codes') ? 1 : 255); |
const usePosixExitCodes = process.argv.includes('--posix-exit-codes'); | ||
const clampedCode = | ||
usePosixExitCodes && code > 0 ? EXIT_FAILURE : Math.min(code, 255); |
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.
const usePosixExitCodes = process.argv.includes('--posix-exit-codes'); | |
const clampedCode = | |
usePosixExitCodes && code > 0 ? EXIT_FAILURE : Math.min(code, 255); | |
const clampedCode = Math.min(code, process.argv.includes('--posix-exit-codes') ? 1 : 255); |
Moving from a confusing setup like: proc.on('exit', (code, signal) => {
process.on('exit', () => {
if (signal) {
process.kill(process.pid, signal);
} else {
process.exit(code);
}
});
}); To something like: proc.on('exit', (code, signal) => {
process.on('exit', () => {
if (signal) {
const numericSignal =
typeof signal === 'string' ? os.constants.signals[signal] : signal;
if (mochaArgs['posix-exit-codes'] === true) {
process.exit(SIGNAL_OFFSET + numericSignal);
} else {
process.kill(process.pid, signal);
}
} else if (code !== 0 && mochaArgs['posix-exit-codes'] === true) {
process.exit(EXIT_FAILURE);
} else {
process.exit(code);
}
});
}); It doesn't really make it easier to consume 🤔 |
And for reference: The original introduction of that listener in a listener happened 12 years ago in #571 with the PR description:
And might have been somewhat invalidated by #3827 which makes it so that mocha does |
VoxPelle pointed out some good structural points
Requirements
Description of the Change
Mocha uses the number of failed tests as an exit code, which is unconventional and has led to a number of issues when trying to integrate mocha into ci/cd pipelines.
To resolve these issues, this PR introduces a
--posix-exit-codes
boolean command-line option. If this option is specified, a fatal signal (eg.SIGABRT
, et al) will cause the process to consistently exit with a standard posix exit code (128 + the numeric ID of the signal) when mocha is run as a child process (which is the case when passing node options). This helps to solve issues for toolchains that expect standard posix exit codes, for example by preventing out-of-memory crashes from being silently ignored.The GNU libc manual page provides additional context on why using the number of errors as an exit status is problematic:
Providing the option to exit with standard posix shell exit codes avoids these problems and solves a number of downstream issues listed below.
Alternate Designs
The alternatives considered were not in the scope of the mocha project.
Why should this be in core?
This option is implemented in the core repository where signal handling occurs.
Benefits
This PR provides a solution for #3559 and various downstream issues reported by mocha consumers; it preserves non-zero exit codes when mocha is spawned as a child process via various CI/CD or reporting tools and prevents silently swallowing out-of-memory errors.
Possible Drawbacks
Introduces another option. Requires docs. Leaves non-standard exit code behavior to remain as the default.
Applicable issues
fixes #3559
#3893
#2445
#2438
istanbuljs/nyc#798
cypress-io/cypress#24695