我有一个拒绝承诺的课程:
Sync.prototype.doCall = function(verb, method, data) {
var self = this;
self.client = P.promisifyAll(new Client());
var res = this.queue.then(function() {
return self.client.callAsync(verb, method, data)
.then(function(res) {
return;
})
.catch(function(err) {
// This is what gets called in my test
return P.reject('Boo');
});
});
this.queue = res.delay(this.options.throttle * 1000);
return res;
};
Sync.prototype.sendNote = function(data) {
var self = this;
return self.doCall('POST', '/Invoice', {
Invoice: data
}).then(function(res) {
return data;
});
};
在我的测试中:
return expect(s.sendNote(data)).to.eventually.be.rejectedWith('Boo');
然而,当测试通过时,它会将错误抛出到控制台。
未处理的拒绝错误:嘘 ...
对于非promise错误,我使用bind来测试以防止在Chai可以包装和测试之前抛出错误:
return expect(s.sendNote.bind(s, data)).to.eventually.be.rejectedWith('Boo');
然而,这不适用于此并返回:
TypeError:[Function] is not a thenable.
测试此方法的正确方法是什么?
答案 0 :(得分:15)
(免责声明:即使是那些不使用Bluebird的人,这也是一个很好的问题。我发布了类似的答案here;这个答案适用于那些没有使用Bluebird的人。)
以下是如何使用chai-as-promised来测试承诺的resolve
和reject
个案例:
var chai = require('chai');
var expect = chai.expect;
var chaiAsPromised = require("chai-as-promised");
chai.use(chaiAsPromised);
...
it('resolves as promised', function() {
return expect(Promise.resolve('woof')).to.eventually.equal('woof');
});
it('rejects as promised', function() {
return expect(Promise.reject('caw')).to.be.rejectedWith('caw');
});
你可以完成同样的事情,而不像这样承诺:
it('resolves as promised', function() {
return Promise.resolve("woof")
.then(function(m) { expect(m).to.equal('woof'); })
.catch(function(e) { throw e }) // use error thrown by test suite
;
});
it('rejects as promised', function() {
return Promise.reject("caw")
.then(function(m) { throw new Error('was not supposed to succeed'); })
.catch(function(m) { expect(m).to.equal('caw'); })
;
});
答案 1 :(得分:6)
我个人使用那个成语:
it('rejects as promised', function() {
return Promise.reject("caw")
.then(
(m) => { assert.fail('was not supposed to succeed'); }
(m) => { /* some extra tests here */ }
);
});
这是极少数情况之一then(onFulfilled, onRejected)
(2个参数)合法使用。
如果您按照其他答案中的建议链接.then(reject).catch(onRejected)
,则每次都会输入catch
处理程序,因为它会捕获前面产生的拒绝then
处理程序 - 如果您不小心检查可能性,可能会导致常青检测。
答案 2 :(得分:5)
您收到错误是因为sendNote被拒绝而且您没有抓住它。
尝试:
var callPromise = self.doCall('POST', '/Invoice', {
Invoice: data
}).then(function(res) {
return data;
});
callPromise.catch(function(reason) {
console.info('sendNote failed with reason:', reason);
});
return callPromise;
看起来你还必须将现有的一个区块移出:
var res = this.queue.then(function() {
return self.client.callAsync(verb, method, data)
.then(function(res) {
return;
});
}).catch(function(err) {
// This is what gets called in my test
return P.reject('Boo');
});
答案 3 :(得分:0)
我遇到了同样的问题,但是在做了很多黑客攻击时,我找到了一个用于测试mocha中被拒绝的承诺的解决方案。
写下面的mocha代码
it('works with resolved and rejected promises', function() {
return yourPromiseFunction("paramifrequired")
.then((result)=>{
//check with should or expect
}).catch((result)=>{
//check with should or expect whether it's rejected with proper result status. set result status in promise function while rejecting accordingly
})
});
注意: - 希望您觉得它很有用。如果您有其他想法建议,请评论我,我是新手探索js世界