Skip to content
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

uncaughtException after runner's end: improve exit code #4019

Merged
merged 1 commit into from
Feb 11, 2020

Conversation

juergba
Copy link
Contributor

@juergba juergba commented Sep 16, 2019

@juergba
Copy link
Contributor Author

juergba commented Sep 16, 2019

@medikoo I would highly appreciate your feedback to this draft PR, thank you!

@coveralls
Copy link

coveralls commented Sep 16, 2019

Coverage Status

Coverage remained the same at 92.857% when pulling 147f763 on juergba/allowUncaught into b431609 on master.

Copy link

@medikoo medikoo left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great thanks @juergba for initiative. Having that fixed in Mocha will be really helpfiul.

Concerning #3917 I commented (it fixes issue but only for Node.js v10+, not sure if it's intended)

Concerning #3938 it fixes issue but only when--allowUncaught is set. If I see the complete fix would be to do console.error('\n', err); process.exit(1); also when runnable.isFailed() || runnable.isPending() is met here:

return;

lib/cli/run-helpers.js Outdated Show resolved Hide resolved
@juergba
Copy link
Contributor Author

juergba commented Sep 17, 2019

Concerning #3938 it fixes issue but only when --allowUncaught is set.

TBH I don't really want to touch this recovery feature when --allowUncaught: false. I did some testing and the behavior is buggy and unpredictable. There are too many end events and the runner keeps going after the first one. I will have a deeper look at this topic though.
=> see #4025

For your needs is there anything opposing against using --allowUncaught: true and not using the recovery feature?

lib/runner.js Outdated Show resolved Hide resolved
@juergba juergba force-pushed the juergba/allowUncaught branch from 8dbfb73 to 5b63b8e Compare September 17, 2019 15:29
@juergba
Copy link
Contributor Author

juergba commented Sep 17, 2019

I have to investigate further about the dependencies of --allow-uncaught and --exit. While combining both options the outcome of my tests is suboptimal.

@juergba
Copy link
Contributor Author

juergba commented Sep 19, 2019

The more I'm testing this, the more weird behavior pops up.

describe('test', () => {
  it('test1', function() { });
  it('test2', function () {
    this.skip();
  });  
  it('test3', function () { });  
  it('test4', function () {
    this.skip();
  });
});

Running with node bin\mocha --allow-uncaught test.js:

test
    √ test1

There is no (uncaught) exception, but all this.skip() tests plus the test summary are truncated.
see => #4030

@juergba
Copy link
Contributor Author

juergba commented Sep 24, 2019

@medikoo above bug (this.skip() plus --allow-uncaught) is rather complex and I will need more time to solve this problem. It's mainly due to our implementation of (async) this.skip() by throwing an uncaught exception.

Node's docu states:

Attempting to resume normally after an uncaught exception can be similar to pulling out of the power cord when upgrading a computer — nine out of ten times nothing happens - but the 10th time, the system becomes corrupted.

Do you have some time left to comment this?

@medikoo
Copy link

medikoo commented Sep 25, 2019

Node's docu states

In my understanding, it's just to indicate that we cannot be sure of side effects/broken states that come with uncaught exceptions, and therefore we should not try to recover from them. Otherwise if we know what we're doing (e.g. recovering from plain throw new Error('Example')) it can be safe.

Still, I totally I agree, that designing this.skip() to work through uncaughtException channel is very controversial.
I assume it was decided back in a past, when async tests were not necessarily written with promises, and were settled via done callback.
Then idea was that any this.skip() between start of a test, and done callback invocation, should be caught and recognized, so uncaughtException seemed the only way to cover that.

Maybe it'll be good with a new major, to narrow async support just to promises, and that way errors handling could be simplified (?)

When sync tests and async tests resolved via promises are concerned, I believe that any uncaughtException should be treated as an error that should be reported (even one thrown by this.skip())) as it's a clear signal of orphaned async flow, which is an error on its own.

With how current implemention works, I would definitely not ignore uncaught exceptions that follows ones thrown via this.skip(). Fact that they're ignored now is source of #3938

@juergba
Copy link
Contributor Author

juergba commented Sep 25, 2019

@medikoo thank you.

Then idea was that any this.skip() between start of a test, and done callback invocation, should be caught and recognized, so uncaughtException seemed the only way to cover that.

it('test', function (done) {
      var self = this;
      setTimeout(function () {
        instruction1;
        self.skip();       // includes implicit 'done()' and throws a 'Pending' error
        instruction2;     // does not run
        instruction3;     // does not run
      }, 100);
    });

The idea was to abort an asyncly running callback and not to execute instruction2 and instruction3.

The user could easily get the same result by using if skip/else and we wouldn't need to throw this Pending error. We would just have to label the test as pending.

@medikoo
Copy link

medikoo commented Sep 25, 2019

I think fact that this.skip() throws is desirable, questionable is how it's caught :)

But in async tests reported via done, I believe the uncaughtException channel is the only way to go.

@juergba juergba force-pushed the juergba/allowUncaught branch from 5b63b8e to f6780d7 Compare September 28, 2019 07:21
@juergba juergba force-pushed the juergba/allowUncaught branch from f6780d7 to 9743cb5 Compare October 8, 2019 14:09
@juergba juergba force-pushed the juergba/allowUncaught branch from 9743cb5 to f90c624 Compare November 22, 2019 18:07
@juergba juergba force-pushed the juergba/allowUncaught branch from f90c624 to d45a389 Compare January 2, 2020 17:48
@juergba juergba force-pushed the juergba/allowUncaught branch from d45a389 to e24b4b1 Compare January 12, 2020 09:21
@juergba juergba force-pushed the juergba/allowUncaught branch from e24b4b1 to e540f84 Compare February 11, 2020 09:09
@juergba juergba force-pushed the juergba/allowUncaught branch from e540f84 to 147f763 Compare February 11, 2020 13:51
@juergba juergba changed the title Exit Code improvements with uncaught exception uncaughtException after runner's end: improve exit code Feb 11, 2020
@juergba juergba self-assigned this Feb 11, 2020
@juergba juergba added area: async related to asynchronous use of Mocha type: bug a defect, confirmed by a maintainer qa semver-patch implementation requires increase of "patch" version number; "bug fixes" labels Feb 11, 2020
@juergba juergba added this to the next milestone Feb 11, 2020
@juergba juergba marked this pull request as ready for review February 11, 2020 14:14
@juergba juergba merged commit 2ff1cb2 into master Feb 11, 2020
@juergba
Copy link
Contributor Author

juergba commented Feb 11, 2020

@medikoo thank you for your support!

@juergba juergba deleted the juergba/allowUncaught branch February 11, 2020 14:56
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area: async related to asynchronous use of Mocha semver-patch implementation requires increase of "patch" version number; "bug fixes" type: bug a defect, confirmed by a maintainer
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Crashes that happen after test completion are reported with success exit code
3 participants