I have this function:
let removePresentation = function(presentationName, callback) {
  let rimraf = require('rimraf');
  callback();
  callback();
  callback();
  if(!presentationName || !presentationName.trim()) {
    callback();
    return;
  }
  presentationName = presentationName.replace('.zip', '');
  rimraf('./presentations/' + presentationName, function(err) {
    if(err) {
      console.log(err);
    }
    callback();
  });
};
exports.removePresentation = removePresentation;
and I am trying to test it with the following:
var chai = require('chai'),
expect = require('chai').expect,
sinonChai = require('sinon-chai'),
sinon = require('sinon'),
mock = require('mock-require');
chai.use(sinonChai);
describe('removePresentation', function() {
  var sandbox;
  var callback;
  var rimrafSpy;
  beforeEach(function() {
    sandbox = sinon.sandbox.create();
    mock('../business/communications_business', {});
    rimrafSpy = sinon.spy();
    callback = sinon.spy();
    mock('rimraf', rimrafSpy);
  });
  afterEach(function() {
    sandbox.restore();
  });
  it('should call rimraf if presentation name is valid', function(done) {
    let RoomStateBusiness = require('../business/roomstate_business');
    RoomStateBusiness.removePresentation('name.zip', callback);
    expect(rimrafSpy).to.have.been.calledWith('./presentations/name');
    expect(callback).to.have.been.called.once;
    done();
  });
  it('should not call rimraf if presentation name is null', function(done) {
    let RoomStateBusiness = require('../business/roomstate_business');
    RoomStateBusiness.removePresentation(null, callback);
    expect(rimrafSpy).not.to.have.been.called;
    expect(callback).to.have.been.called.once;
    done();
  });
  it('should not call rimraf if presentation name is whitespace', function(done) {
    let RoomStateBusiness = require('../business/roomstate_business');
    RoomStateBusiness.removePresentation('      ', callback);
    expect(rimrafSpy).not.to.have.been.called;
    expect(callback).to.have.been.called.once;
    done();
  });
  it('should not call rimraf if presentation name is empty string', function(done) {
    let RoomStateBusiness = require('../business/roomstate_business');
    RoomStateBusiness.removePresentation('', callback);
    expect(rimrafSpy).not.to.have.been.called;
    expect(callback).to.have.been.called.once;
    done();
  });
});
Even though I am clearly calling callback() multiple times (whilst testing only), expect(callback).to.have.been.called.once; is always asserting to true. I have checked on the Chai api that that expects the call to be exactly once, although it is always passing no matter how many times I call callback(). What am I doing wrong?
you've passed the test: you have succeeded (in this trial or evaluation) idiom.
The word passed is the past tense of the verb to pass. The verb pass, when used in present tense would look like this: I will pass the ball to you. If you substituted the word pass for passed, I passed the ball to you, it signifies that this happened previously. That is has already happened.
There is no such assertion as expect(fn).to.have.been.called.once.
As per sinon-chai docs, there is only:
expect(fn).to.have.been.calledexpect(fn).to.have.been.calledOnceThis is a known problem with chai and why getter-only-assertions are a bad thing. Chai allows you to write a piece of code which looks like a property access (ie. the assertion itself does not end with a function call) to assert...whatever you want to assert. This uses property getters to execute the necessary code.
The problem is that if you make a typo or other mistake, the expression will simply evaluate to undefined (you are accessing a property which does not exist) and no assertion code is ever executed, resulting in a test passing (because tests only fail if an exception is thrown).
In your case, there is an assertion for called, and this most likely returns an object. Unfortunately, that object does not have assertion once, so there is no code executed and the test passes.
There are 2 options available to you:
Proxy support (not sure where Proxy support was added, probably Node.js 5 or 6) - chai introduced a safeguard against these issues by proxying all property access through a Proxy object which checks if you are using a valid assertionundefined is not a function errorThe second option is highly preferred, in my opinion, as there can be no doubt on the correctness of the test case. Chai proxy support can still be turned off even on supported platforms.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With