AngularJS - 使用$ http服务的测试服务方法

dnc*_*253 19 unit-testing angularjs

我的服务功能看起来像这样:

addStatement: function(user, action, object) {
            var statement = {
                    user: user,
                    action: action,
                    object: object
            };
            $http({
                method: 'POST', 
                url: '/foo/bar', 
                data: statement, 
                headers: {Authorization: 'Basic YWxhZGRpbjpvcGVuIHNlc2FtZQ=='}
            }).success(function(data) {
                alert("success: " + data);
            });
        }
Run Code Online (Sandbox Code Playgroud)

我想为这个方法编写单元测试,但我不确定如何使它工作.基本上,我想测试发送的数据是否已从函数参数中正确构造,并且发送了正确的Authorization标头.

我一直在阅读如何使用$ httpBackend,但是那里的例子是测试一个控制器,只是在发出请求并返回时在$ scope上发生了变化.有没有办法实现我想要的测试?或者我说错了什么?

pko*_*rce 37

测试服务与测试控制器没有太大差别,因此原理是相同的.基本上你需要:

  • 注入测试对象
  • 设置模拟(如果有的话) - 在这里你使用$ httpBackend mock在正确的轨道上
  • 在测试对象上运行方法并验证结果

如果您在测试服务方法导致$ http POST调用后,您可以像这样编写测试(省略非必要部分):

beforeEach(module('MyApp'));
beforeEach(inject(function(MyService, _$httpBackend_) {
    service = MyService;
    $httpBackend = _$httpBackend_; // angular strips the underscores so
                                   // we don't need to use unique names
    // https://docs.angularjs.org/api/ngMock/function/angular.mock.inject
}));

it('should invoke service with right paramaeters', function() {
    $httpBackend.expectPOST('/foo/bar', {
        "user": "testUser",
        "action": "testAction",
        "object": {}
    }, function(headers){
        return headers.Authorization === 'Basic YWxhZGRpbjpvcGVuIHNlc2FtZQ==';
    }).respond({});
    service.addStatement('testUser', 'testAction', {});
    $httpBackend.flush();
});
Run Code Online (Sandbox Code Playgroud)

以下是工作中的jsFiddle,说明了这个测试的实际效果:http://jsfiddle.net/pkozlowski_opensource/MkrjZ/2/

最后一句话:最好不要在单元测试中使用.alert(),因为这些警报会暂停执行测试.