角度资源测试:$ httpBackend.flush()导致意外请求

Bar*_*ski 7 javascript jasmine angularjs gruntjs

我想测试angularjs资源.

'use strict';

/**
 * AddressService provides functionality to use address resource in easy way.
 *
 * This is an example usage of method:
 *
 * `get`:
 *
     var a = AddressService.get({id: '1'},
         function (data) {
            // Work here with your resource
         }
     );
 *
 */
App.factory('AddressService', function ($resource, $rootScope) {
    var url = [
            $rootScope.GLOBALS.API_PATH,
            $rootScope.GLOBALS.API_VERSION,
            'models/address/:id'
        ].join('/'),

        actions = {
            'get': {
                method: 'GET',
                params: {id: '@id'}
            }
        };
    return $resource(url, {}, actions);
});
Run Code Online (Sandbox Code Playgroud)

我创建了测试:

'use strict';

var $httpBackend;

describe('Service: AddressService', function () {

    beforeEach(module('myApp'));

    beforeEach(inject(function ($injector) {
        var url_get = 'api/v1/models/address/5';

        var response_get = {
            address_line_1: '8 Austin House Netly Road',
            address_line_2: 'Ilford IR2 7ND'
        };

        $httpBackend = $injector.get('$httpBackend');

        $httpBackend.whenGET(url_get).respond(response_get);

    }));

    describe('AddressService get test', function () {
        it('Tests get address', inject(function (AddressService) {
            var address = AddressService.get({ id: '5'});
            $httpBackend.flush();
            expect(address.address_line_1).toEqual('8 Austin House Netly Road');
            expect(address.address_line_2).toEqual('Ilford IR2 7ND');
        }));
    });
});
Run Code Online (Sandbox Code Playgroud)

我没有很好的角度经验.我在karma.config.js中设置了jasmine.AngularJS v1.0.6 Yoeman和Grunt管理项目.

我试试 grunt test

Running "karma:unit" (karma) task
INFO [karma]: Karma server started at http://localhost:8080/
INFO [launcher]: Starting browser Chrome
INFO [Chrome 27.0 (Linux)]: Connected on socket id RFFUY5bW8Hb5eTu0n-8L
Chrome 27.0 (Linux) Service: AddressService AddressService get Tests get address FAILED
        Error: Unexpected request: GET views/home.html
        No more request expected
            at Error (<anonymous>)
            at $httpBackend (/home/bart/y/projects/x/frontend/app/components/angular-mocks/angular-mocks.js:910:9)
            at sendReq (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:9087:9)
            at $http (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:8878:17)
            at Function.$http.(anonymous function) [as get] (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:9021:18)
            at $q.when.then.then.next.locals (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:7394:34)
            at wrappedCallback (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:6797:59)
            at wrappedCallback (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:6797:59)
            at /home/bart/y/projects/x/frontend/app/components/angular/angular.js:6834:26
            at Object.Scope.$eval (/home/bart/y/projects/x/frontend/app/components/angular/angular.js:8011:28)
        Error: Declaration Location
            at window.jasmine.window.inject.angular.mock.inject (/home/bart/y/projects/x/frontend/app/components/angular-mocks/angular-mocks.js:1744:25)
            at null.<anonymous> (/home/bart/y/projects/x/frontend/test/spec/services/userdata/AddressService.js:32:30)
            at null.<anonymous> (/home/bart/y/projects/x/frontend/test/spec/services/userdata/AddressService.js:31:5)
            at /home/bart/y/projects/x/frontend/test/spec/services/userdata/AddressService.js:5:1
..................................................................
Chrome 27.0 (Linux): Executed 67 of 67 (1 FAILED) (0.343 secs / 0.179 secs)
Warning: Task "karma:unit" failed. Use --force to continue.
Run Code Online (Sandbox Code Playgroud)

如果我$httpBackend.flush()在测试中删除.测试正在过去.但我undefined来自address.我看到了一些示例:示例全部使用flush()但是只有我得到了愚蠢的异常:Error: Unexpected request: GET views/home.html

views/home.html是我在项目目录中的视图.我不知道如何解决我的问题,我找不到任何解决方案.我不知何故错过了这一点?

任何人都可以在我的代码中看到错误吗?所有建议将不胜感激.

编辑

我发现我需要使用它:

$httpBackend.when('GET', /\.html$/).passThrough();
Run Code Online (Sandbox Code Playgroud)

但另一个问题是我得到:

TypeError: Object #<Object> has no method 'passThrough'
Run Code Online (Sandbox Code Playgroud)

Bar*_*ski 8

我终于找到了解决方案:

我试着按照这个:其他帖子

所以我补充说$templateCache:

'use strict';

var $httpBackend;

describe('Service: AddressService', function () {

    beforeEach(module('myApp'));

    beforeEach(inject(function ($injector, $templateChache) {
        $templateCache.put('views/home.html', '.<template-goes-here />');
        var url_get = 'api/v1/models/address/5';

        var response_get = {
            address_line_1: '8 Austin House Netly Road',
            address_line_2: 'Ilford IR2 7ND'
        };

        $httpBackend = $injector.get('$httpBackend');

        $httpBackend.whenGET(url_get).respond(response_get);

    }));

    describe('AddressService get test', function () {
        it('Tests get address', inject(function (AddressService) {
            var address = AddressService.get({ id: '5'});
            $httpBackend.flush();
            expect(address.address_line_1).toEqual('8 Austin House Netly Road');
            expect(address.address_line_2).toEqual('Ilford IR2 7ND');
        }));
    });
});
Run Code Online (Sandbox Code Playgroud)

  • 我同意.这似乎是一个非常糟糕的黑客忽略这个问题.我现在遇到同样的问题,我不确定为什么会这样...... (5认同)
  • 为什么templateCache很重要?它似乎与API服务无关. (2认同)
  • 这些是单元测试,您必须提取特定单位(在这种情况下为服务)进行测试.删除所有其他依赖项甚至模板,这些模板只是在这里模拟.我不知道更好的方法也不知道核心角流量.这个问题尚未结束. (2认同)

SSt*_*ley 8

该错误是由您的测试尝试加载应用程序的主页面引起的.由于您尚未告知测试期望此服务器调用,因此会返回错误.有关这一点的说明,请参阅$ httpBackend的文档.

您的$ templateCache解决方法是为单元测试指令而设计的,而不是其他任何指令.你非常接近:

$httpBackend.when('GET', /\.html$/).passThrough();
Run Code Online (Sandbox Code Playgroud)

由于您不需要对实际的模板文件执行任何操作,因此将其添加到beforeEach()块是一种安全而简单的方法.

$httpBackend.whenGET(/\.html$/).respond('');
Run Code Online (Sandbox Code Playgroud)

这样可以阻止您获得TypeError.

TypeError: Object #<Object> has no method 'passThrough'
Run Code Online (Sandbox Code Playgroud)

在升级到新版本的Angular JS并使用respond('')工作正常后,我在单元测试中遇到了同样的问题.

  • 难道不能避免 $http 服务向应用程序主页发出这样的请求吗? (2认同)