mocha: š Bug: Support `setImmediate` between hooks on Promises
Prerequisites
- Checked that your issue isnāt already filed by cross referencing issues with the
common mistake
label - Checked next-gen ES issues and syntax problems by using the same environment and/or transpiler configuration without Mocha to ensure it isnāt just a feature that actually isnāt supported in the environment in question or a bug in your code.
- āSmoke testedā the code to be tested by running it outside the real test suite to get a better sense of whether the problem is in the code under test, your usage of Mocha, or Mocha itself
- Ensured that there is no discrepancy between the locally and globally installed versions of Mocha. You can find them with:
node node_modules/.bin/mocha --version
(Local) andmocha --version
(Global). We recommend avoiding the use of globally installed Mocha.
Description
Iām opening this issue as a result of an investigation into https://github.com/domenic/chai-as-promised/issues/173. However, thereās a lot of unrelated noise in that thread, so Iāll demonstrate the issue here independently of chai-as-promised
.
Hooks typically wait until the next tick of the event loop before proceeding, per these lines. As a result, if a Promise
with no rejection handler is declared in a hook prior to the test in which the rejection handler is added to the Promise
, then a warning is typically generated about an unhandled promise rejection, followed by another warning about asynchronous handling of the promise rejection.
I used the word ātypicallyā twice above because what I described doesnāt apply to the final beforeEach
prior to the test; unlike the other hooks, the final beforeEach
proceeds immediately to the test, without waiting for the next tick of the event loop.
Steps to Reproduce
Example 1:
let promise;
describe("mySuite", function () {
before(function () {
promise = Promise.reject(new Error());
});
it("myTest", function () {
promise.catch(function () {});
});
});
Example 2:
let promise;
describe("mySuite", function () {
beforeEach(function () { // This is the only different line
promise = Promise.reject(new Error());
});
it("myTest", function () {
promise.catch(function () {});
});
});
Expected behavior: Although the tests are meaningless, ideally Iād expect neither of the examples to generate any unhandled promise rejection warnings, or at least for both examples to result in the same behavior.
Actual behavior: Example 1 generates an unhandled promise rejection and Example 2 doesnāt.
Example 1 output:
mocha
mySuite (node:14219) UnhandledPromiseRejectionWarning: Unhandled promise rejection (rejection id: 1): Error (node:14219) [DEP0018] DeprecationWarning: Unhandled promise rejections are deprecated. In the future, promise rejections that are not handled will terminate the Node.js process with a non-zero exit code. ā myTest (node:14219) PromiseRejectionHandledWarning: Promise rejection was handled asynchronously (rejection id: 1)
1 passing (11ms)
Example 2 output:
mocha
mySuite ā myTest
1 passing (7ms)
Reproduces how often: 100%
Versions
Node v8.5.0 Mocha v3.5.3
Additional Information
Changing this line to favor process.nextTick
over global.setImmediate
fixes the issue so that neither example generates a warning, and the Mocha test suite still passes. But I donāt know what other consequences there are.
I think that in most situations this issue isnāt a problem. Most promises being tested in the wild have (or should have) rejection handlers already registered, so no warning would be generated in those cases, despite the next tick between hooks. I just started digging in order to understand what was happening with https://github.com/domenic/chai-as-promised/issues/173. The example in that issue is completely contrived, as are the examples I provided here.
About this issue
- Original URL
- State: open
- Created 7 years ago
- Reactions: 2
- Comments: 15 (8 by maintainers)
My gut instinct here is that if the behavior occurring within a tick is whatās being tested then it belongs inside the test. E.g. instead of this:
ā¦do this instead (although it looks a little awkward):
A much stronger case, in my opinion, is https://github.com/domenic/chai-as-promised/issues/173#issuecomment-330409841 ā which essentially means that there is no way to set up a rejected promise before the test (once Node starts treating unhandled rejections the same way it treats unhandled exceptions). Is there some workaround ā some way to define what the promise will be/do but not eagerly run it? Maybe something like this?
Thereās a downside that one has to know about the problem and this solution. Might be other downsides Iām missing at this point. This also exposes an inherent conflict between āif I create a new promise for each test then every test must handle the rejection even if itās irrelevantā and āif I reuse a promise then the first test handling its rejection would allow the rest to ignore it whether they should or notā*; but I am pretty sure that issue is going to be present regardless of how the promise is created or whether and where itās stored.
*(EDITTED TO ADD: you could bypass the need for the first test to be the one that handles the rejection if you immediately
.catch(function() {})
the promise to suppress the error ā without saving the caught version so you can still examine the rejection in the original ā in fact, the real danger Iām getting at is that even if you use the first-test-catches approach instead of that obvious hack it has the same effect on the rest of the tests as the hack would.)Consistency around the hook behavior is needed, but more importantly users just need to know what to expect. It doesnāt work quite right atm, and thereās an open discussion (though dormant) about how various failure conditions in the hooks should impact the tests.
Iām 99% sure the intent of
setImmediate(fn)
was āexecute this beforesetTimeout(fn)
would.āSo yeah, bug. Thanks š