使用模拟工厂测试Jasmine中的嵌套承诺

时间:2015-01-13 10:31:46

标签: javascript angularjs mocking jasmine promise

我正在构建AngularJS应用程序,我正在尝试对其进行单元测试。我想注入模拟服务而不是真正的依赖。

这是我要测试的服务:

.factory('LoginService',['Login', function(Login){
    return {
        getActiveUser: function() {
            return Login.query({}).$promise.then(
                function (users) {
                    return users[0];
                },
                function (error) {
                    console.log(error);
                }
            );
        }
    };
}]);

这是一个简单的服务,它从所有用户列表中返回第一个用户。贝娄是我用嘲笑写单元测试的尝试,但没有成功。

....
beforeEach(module('loginModule', function($provide) {

    mockedService = {
        getActiveUser: function() {
            return {
                then: function(callback) {
                    return callback({name: 'John'});
                }
            };
        }
    };

    $provide.value('LoginService', mockedService);

}));

beforeEach(inject(function (_LoginService_,  _Login_, _$httpBackend_, $rootScope) {
    LoginService = _LoginService_;
    Login = _Login_;
    scope = $rootScope;
    httpBackend = _$httpBackend_;

    spyOn(mockedService, 'getActiveEnvironment').andCallThrough();
}));

describe('getActiveUser', function () {
    it('should call the function', function() {
        LoginService.getActiveUser();
        expect(mockedService.getActiveUser).toHaveBeenCalled();
    });

    it('should return active user', function() {
        var user = {};

        mockedService.getActiveUser().then(
            function (env) {
                user = env;
            }
        )

        expect(user.name).toBe("John");

    });
});

我能够使用间谍进行测试通过并直接注入服务方法。现在我想实现相同的目标,但是使用模拟实例。第一次测试通过,但第二次测试失败。有人知道吗?

1 个答案:

答案 0 :(得分:1)

尝试以这种方式模拟服务:

mockedService = {
    getActiveUser: function() {
        return $q.when({username: 'Thomas'});
    }
}
然后你可以窥探它:

spyOn(mockedService, 'getActiveUser').and.callThrough();

并且你可以运行它

mockedService.getActiveUser().then(function(env) {
    user = env;
});

您也可以手动创建类似承诺的API:

mockedService = {
    getActiveUser: function() {
        return {
            then: function(callback) {
                return callback({username: 'Thomas'});
            }
        };
    }
}