javascript - 无法让 Jasmine 测试成功调用注入(inject)的服务

标签 javascript angularjs unit-testing jasmine karma-runner

我正在尝试使用 Angular 进行单元测试,但遇到了一些障碍。在尝试为这个问题创建一个 plnkr 时:Jasmine test fails making $httpbackend call我遇到了不同的错误。我希望如果有人能帮助解决这个问题,这将有助于解决原来的问题:http://plnkr.co/edit/lJTx0ldR9nEnlYbU5pJd

    (function(){'use strict';
  var RestService = function($http, $rootScope){
    var _postData = function(url, params, data, successFunction,  errorMsg, errorFunction, config) {

            if(config && config.hasOwnProperty('showLoader')){
                $rootScope.showLoader = config.showLoader;
            }

            $http({
                method: 'POST',
                url: url,
                params: params,
                data: data,
                cache: false
            })
                .success(function(data, status, headers, config) {
                    $rootScope.showLoader = false;
                    if (successFunction === undefined) {
                        _defaultSuccessFunction(data, status, headers, config);
                    }
                    else {
                        successFunction(data, status, headers, config);
                    }
                })
                .error(function (data, status, headers, config) {
                    $rootScope.showLoader = false;
                    if(status === 401){
                        _processError(data, status, headers, config, errorMsg, errorFunction);
                    }
                });
        };

        return {
            postData: _postData
        };
  };
  angular.module('ram-utilities.ui.rest.service', []).factory('RestService', ['$http', '$rootScope', RestService]);
})();

(function(){ 'use strict';
    var LoginController = function($scope, RestService){
        var _user = undefined;
        var _message = 'hello';

        var _login = function(user){
            var _success = function(response){
                _message = response.success;
                _user = response.user;
            };

            var _error = function(response){
                _message = response.success;
            };

            RestService.postData('/api/login',  null, {username: user.username, password: user.password}, _success, 'Invalid login, please try again', _error, {showLoader: true});
        };

        $scope.model = {
            login: _login,
            user: _user,
            message: _message
        };
    };

    angular.module('danny',['ram-utilities.ui.rest.service']).controller('LoginController',['$scope', 'RestService',LoginController]);
})();











describe('LoginController', function(){
    var scope, $httpBackend, controller, restService;

    beforeEach(function(){
        module('danny');
    });

    beforeEach(inject(function(_$controller_, _$rootScope_, _$httpBackend_, _RestService_){
        $httpBackend = _$httpBackend_;
        restService = _RestService_;
        scope = _$rootScope_.$new();
        controller = _$controller_('LoginController', {
            $scope: scope,
            RestService: restService
        });
    }));

    afterEach(function() {
        $httpBackend.verifyNoOutstandingExpectation();
        $httpBackend.verifyNoOutstandingRequest();
    });

    describe('successfully logging in', function(){

       it('should redirect to /blog when authenticated', function(){

           var user = {"username":"danny@ravenartmedia.com", "password":"test"};
            expect(user.username).toEqual('danny@ravenartmedia.com');

          $httpBackend.expectPOST('/api/login', user).response(200, {});

          scope.model.login(user);
          $httpBackend.flush();

          expect(scope.model.user).not.toBe(undefined);
       });
    });
});




(function() {
  var jasmineEnv = jasmine.getEnv();
  jasmineEnv.updateInterval = 250;

  /**
   Create the `HTMLReporter`, which Jasmine calls to provide results of each spec and each suite. The Reporter is responsible for presenting results to the user.
   */
  var htmlReporter = new jasmine.HtmlReporter();
  jasmineEnv.addReporter(htmlReporter);

  /**
   Delegate filtering of specs to the reporter. Allows for clicking on single suites or specs in the results to only run a subset of the suite.
   */
  jasmineEnv.specFilter = function(spec) {
    return htmlReporter.specFilter(spec);
  };

  /**
   Run all of the tests when the page finishes loading - and make sure to run any previous `onload` handler

   ### Test Results

   Scroll down to see the results of all of these specs.
   */
  var currentWindowOnload = window.onload;
  window.onload = function() {
    if (currentWindowOnload) {
      currentWindowOnload();
    }

    //document.querySelector('.version').innerHTML = jasmineEnv.versionString();
    execJasmine();
  };

  function execJasmine() {
    jasmineEnv.execute();
  }
})();

谢谢!!!

最佳答案

有几件事:

  1. $httpBackend.expectPOST中使用respond而不是response

  2. data 参数隐藏在 $http.success 处理程序中 - 将其重命名为 postData。还将预期对象传递给成功回调。

var _postData = function(url, params, data, successFunction /* ... */) {
    //...

    $http( {
        method: 'POST',
        url: url,
        params: params,
        data: data,
        cache: false
    } )
        .success( function ( postData, status, headers, config ) {
            // ...
            successFunction({
                success: true,
                user: data
            }, status, headers, config);
        } );

    //...
}
  • LoginController 返回只能通过显式引用(如 $scope.model.user = response.user;)更改的对象 - 只需调用 _user = response.user 不会更改 $scope.model.user,除非您将 _user 包装在闭包中。

  • Working Plunk

  • 关于javascript - 无法让 Jasmine 测试成功调用注入(inject)的服务,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28014719/

    相关文章:

    javascript - 更改 Atom Keymap.cson 中的 JS 缩进

    javascript - 以编程方式移动可拖动位置

    javascript - Angular JS 下拉菜单使用 ng-click 和 ng-show 显示动态内容?

    node.js - PhantomJS 意外退出,退出代码为 -1073741819

    c# - xamarin 中的 NUnit 错误

    javascript - resizeBy 不起作用,不知道为什么

    javascript - 云HTTPS函数: returning a Promise which is inside a Promisee

    jquery - 在 Angular 中使用 ui-sref 时出现错误 : Invalid state ref '' ,

    javascript - 使用 Javascript 的 Firebase 动态链接

    validation - Jersey 自定义验证器单元测试