javascriptnode.jsmocha.jschaichai-as-promised

chai-as-promised: multiple expect statements in a single test


I'm using chai-as-promised to test some promises. My issue is I'm not sure how to have multiple expect statements in a single test. In order for the expect().to.be.fulfilled to work properly, I need to return it, like this:

it('test', () => {
  return expect(promise).to.be.fulfilled
}

... or to use notify, like this:

it('test', (done) => {
  expect(promise).to.be.fulfilled.notify(done)
}

The issue comes when I have another thing I need to check, such as that a certain function gets called, like this:

it('test', (done) => {
  var promise = doSomething()
  expect(sinon_function_spy.callCount).to.equal(1)
  expect(promise).to.be.fulfilled.notify(done)
})

The problem here is that, because doSomething() is asynchronous, the call to sinon_function_spy may not have occurred yet when I call that expect, making this test flaky. If I use a then, like this:

it('test', (done) => {
  var promise = doSomething()
  promise.then(() => {
    expect(sinon_function_spy.callCount).to.equal(1)
  })
  expect(promise).to.be.fulfilled.notify(done)
})

Then the test technically passes and fails as expected, but it will fail because the promise gets rejected, due to the thrown exception in the then call. Similarly, if I have a case where the promise is expected to reject:

it('test', (done) => {
  var promise = doSomething()
  promise.then(() => {
    expect(sinon_function_spy.callCount).to.equal(1)
  })
  expect(promise).to.be.rejected.notify(done)
})

Then the check on the sinon_function_spy never gets called, since the promise was rejected and doesn't call then.

How can I get both expect statements to reliably execute and return the correct values?


Solution

  • In the case of wanting to assert that the Promise is fulfilled and a call was performed as expected, you don't really need that first part as an assertion. The mocha test case itself will fail if the Promise rejects as long as you are returning it:

    it('test', () => {
      return doSomething()
        .then(() => {
          expect(sinon_function_spy.callCount).to.equal(1)
        });
    });
    

    If the Promise returned by doSomething() rejects, so will the test case. If the expect assertion fails, it will also fail the test case with that failed assertion. If you want to be a bit more explicit:

    it('test', () => {
      return doSomething()
        .then(() => {
          expect(sinon_function_spy.callCount).to.equal(1)
        }, err => {
          expect(err).to.not.exist;
        });
    });
    

    ...you can catch the error. Note that with this flavor of then with two callbacks, the assertion failing in the first callback will not reach the second callback, so it'll just be Mocha that sees the failed assertion.

    Here's how you can do an expected failed Promise:

    it('test', () => {
      return doSomething()
        .then(() => {
          throw new Error('Promise should not have resolved');
        }, err => {
          expect(err).to.exist;
          expect(sinon_function_spy.callCount).to.equal(1)
        });
    })