在测试使用promises的Node模块时如何断言?

时间:2015-01-29 16:37:16

标签: javascript node.js mocha q sinon

我有一个节点控制器“simpleController”,它依赖于“DataAccess”的实例。控制器在“DataAccess”实例上调用函数“getData”并返回一个promise。如果承诺被拒绝,控制器应返回500的状态。

我想在“simpleController”上编写一个测试但是模拟“DataAccess”的实例,这样我就可以返回失败的promise并断言“simpleController”行为与预期一致。

我试图编写一个单元测试来执行此操作但测试超时。

(function (simpleController) {

    var DataAccess = require('./DataAccess');

    var dataAccessInstance;

    simpleController.init = function (dataSource) {
        dataAccessInstance = new DataAccess(dataSource);
    };

    simpleController.setMockDependencies = function (mockDataAccessInstance) {
        dataAccessInstance = mockDataAccessInstance;
    };

    simpleController.performAction = function (req, res) {

        var entityId = req.query.id;

        dataAccessInstance.getData(entityId)
            .then(function (data){
                res.status(200).json(data);
            },
            function (err) {
                res.status(200).json(err);
            });
    }

})(module.exports);



/* global describe, it */
'use strict';

var assert = require('assert');
var sinon = require('sinon');
var Q = require('q');
var DataAccess = require('../lib/temp/DataAccess');

describe('Simple Controller Tests', function () {

    var simpleController;
    var dataAccess;

    beforeEach(function (done) {

        simpleController = require('../lib/temp/simpleController');

        var dataSource = 'dummy';
        dataAccess = new DataAccess(dataSource);

        done();
    });


    it('should return status of 500 if data access fails', function (done) {

        // Arrange
        var req = { query: { id: '12345678' } };
        var res = {};

        var deferred = Q.defer();

        var stub = sinon.stub(dataAccess, 'getData', function() {
            return deferred.promise;
        });

        simpleController.setMockDependencies(dataAccess);

        deferred.promise.fail(function () {

            // Assert
            assert.ok(stub.called);
            assert.equal(res.status, 500);

            done();

        });

        // Act
        simpleController.performAction(req, res);

        deferred.reject();
    });

});



(function () {

    function DataAccess(dataSource) {
        this.dataSource = dataSource;
    }

    DataAccess.prototype.getData = function (entityId) {
        throw new Error('Not implemented');
    };

    module.exports = DataAccess;
})();

非常感谢任何帮助。

1 个答案:

答案 0 :(得分:2)

据我所知,您遇到的问题是您在两个地方为您的承诺提供了故障处理程序。一次在测试中调用deferred.promise.fail,然后在控制器中将第二个参数传递给dataAccessInstance.getData(entityId).then。控制器中的那个可能会替换您在测试中添加的那个。我建议删除这段代码(它并没有真正做任何有用的事情):

deferred.promise.fail(function () {

        // Assert
        assert.ok(stub.called);
        assert.equal(res.status, 500);

        done();

    });

而不是在测试中使用res的空对象,添加状态方法,并在那里执行断言:

it('should return status of 500 if data access fails', function (done) {

    // Arrange
    var req = { query: { id: '12345678' } };
    var res = {};

    var deferred = Q.defer();

    var stub = sinon.stub(dataAccess, 'getData', function() {
        return deferred.promise;
    });

    simpleController.setMockDependencies(dataAccess);

    res.status = function (status) {

        // Assert
        assert.ok(stub.called);
        assert.equal(status, 500);

        done();

    };

    // Act
    simpleController.performAction(req, res);

    deferred.reject();
});

另外,请注意,在您的控制器中,您使用200表示成功和失败,我认为您打算使用500表示失败。最后,您应该在承诺链的末尾添加.done():

dataAccessInstance.getData(entityId)
       .then(function (data){
           res.status(200).json(data);
       },
       function (err) {
           res.status(500).json(err);
       })
       .done();

(见http://documentup.com/kriskowal/q/#tutorial/the-end)。这将避免在测试失败的情况下超时。