NodeJS:如何在使用Mocha,Chai,Sinon进行测试时验证promise内部的函数调用?

时间:2017-09-24 12:01:03

标签: node.js unit-testing mocha sinon chai

我是使用Mocha,Chai和Sinon在NodeJS中进行单元测试的新手。我一直在尝试为API端点编写一个单元测试用例,并且正在努力测试是否在Promise解决方案中从API调用了res.status(200).send();。以下是代码:

controller.js - 具有单元测试功能

const User = require("./user_model");

module.exports = {
    getUserById(req, res) {
        const { params: reqParams } = req,
            { userId } = reqParams;

        User.findOne({ _id: userId })
            .exec()
            .then(function(userFromDb) {
                res.status(200).send();    // unable to test if this is getting called.
            })
            .catch((err) => res.status(500).send());
    }
};

test.js - 是否有单元测试

const sinon = require("sinon"),
    User = require("./user_model"),
    controller = require("../controller");

require("sinon-mongoose");

describe('users', function() {
    it('getUserById', function() {
        const validReqObj = {
            params: {
                userId: "123"
            }
        };

        const mock = sinon.mock(User);
        mock.expects("findOne").withArgs({ _id: validReqObj.params.userId })
            .chain("exec")
            .resolves({ _id: validReqObj.params.userId });

        const validRes = {
            status: function(statusCode) {
                sinon.assert.match(statusCode, 200);

                return {
                    send: function() {

                    }
                };
            }
        };

        controller.getUserById(validReqObj, validRes);

        mock.restore();
        mock.verify();
    });
});

正如您所看到的,我的测试做了两件事:

  1. 模仿mongoose findOne函数并测试findOne的承诺解析是否正常。
  2. 断言使用200状态代码调用res.status()
  3. 它不会测试res.status()是否被调用。例如,如果我从 controller.js 中删除行res.status(200).send();,则测试仍会通过。

    我尝试在res.status上创建spy。它没有按预期工作。

2 个答案:

答案 0 :(得分:2)

问题是你的代码没有任何东西可以指示异步函数何时完成。虽然我们可以在测试中添加一个计时器来解决这个问题,但它有点废话解决方案。

以一种很好的方式解决这个问题的最简单方法是从函数中返回promise:

return User.findOne({ _id: userId })
       ....

在此之后,在您的测试中,您可以简单地执行以下操作:

return controller.getUserById(validReqObj, validRes).then(function() {
    mock.restore();
    mock.verify();
});

注意测试中的返回 - 这允许mocha检测到它是带有承诺的异步测试。

答案 1 :(得分:0)

因为getUserById是异步功能。即使您为User模型创建了模拟,getUserById仍然是异步的。

要创建正确的测试,您可以执行以下操作:

  • getUserById函数返回promise并创建一个测试的promise链。
  • 使用supertest模块发送真实的http请求并验证响应。