5

我有以下功能要测试:

// ...
const local = new WeakMap();

export default class User {

  // ...

  async password(password) {
    if (!password) return local.get(this).get('hash'); // remove this for security reasons!
    if (password.length < 6) throw new Error('New password must be at least 6 characters long');
    if (!password.match(passwordPattern)) throw new Error(`New password must match ${passwordPattern}`);
    local.get(this).set('hash', await Password.hash(password));
  }

  // ...

}

现在我想用mochachaichai-as-promised来测试这个功能,做这个测试用例:

import chai from 'chai';
import chaiAsPromised from 'chai-as-promised';

import User from '../../../server/User';

chai.use(chaiAsPromised);
const expect = chai.expect;

describe('.password()', () => {

  const testuser = new User({username: 'Testuser', password: '123abc'});

  // FINDME
  it(`should throw when too short`, () => {
    return expect(testuser.password('1a')).to.eventually.throw();
  });

  // ...

});

在上面的代码中查找// FINDME注释以获取我所指的上下文。

所以发生的事情是测试不等待异步password()函数完成。当我发现 ECMAScript7 异步函数立即返回一个 thenable 时,应该没问题,因为 chai-as-promised 正是使用这个特性。我的猜测是,异步函数不会将错误抛出到这个所说的承诺中,而不是将其抛出到封闭函数本身中,而无法通过expect().

对于热心的人:我使用以下命令使用babel即时转换此代码:

babel-node --experimental node_modules/mocha/bin/_mocha --watch --recursive -R spec --check-leaks

Babel 6 更新:

使用 babel-node 6.4.0,以下工作:

npm install --save-dev babel-preset-stage-0

然后运行:

./node_modules/.bin/babel-node --presets stage-0 node_modules/mocha/bin/_mocha --    
recursive -R spec --check-leaks
4

1 回答 1

8

eventually.throw()我们通过用.be.rejectedchai-as-promised 提供的替换来解决这个问题,因为 anasync function总是返回一个 Promise。起初我并不清楚。如果你愿意,可以看看github上的讨论。

于 2015-04-05T11:30:21.913 回答