Test a controller with success() and error ()

后端 未结 3 1128
不知归路
不知归路 2021-01-31 10:07

I\'m trying to work out the best way to unit test success and error callbacks in controllers. I am able to mock out service methods, as long as the controller only uses the defa

相关标签:
3条回答
  • 2021-01-31 10:38

    As someone had mentioned in a deleted answer, success and error are syntactic sugar added by $http so they aren't there when you create your own promise. You have two options:

    1 - Don't mock the service and use $httpBackend to setup expectations and flush

    The idea is to let your myService act like it normally would without knowing it's being tested. $httpBackend will let you set up expectations and responses, and flush them so you can complete your tests synchronously. $http won't be any wiser and the promise it returns will look and function like a real one. This option is good if you have simple tests with few HTTP expectations.

    'use strict';
    
    describe('SimpleControllerTests', function () {
    
        var scope;
        var expectedResponse = { name: 'this is a mocked response' };
        var $httpBackend, $controller;
    
        beforeEach(module('myApp'));
    
        beforeEach(inject(function(_$rootScope_, _$controller_, _$httpBackend_){ 
            // the underscores are a convention ng understands, just helps us differentiate parameters from variables
            $controller = _$controller_;
            $httpBackend = _$httpBackend_;
            scope = _$rootScope_;
        }));
    
        // makes sure all expected requests are made by the time the test ends
        afterEach(function() {
          $httpBackend.verifyNoOutstandingExpectation();
          $httpBackend.verifyNoOutstandingRequest();
        });
    
        describe('should load data successfully', function() {
    
            beforeEach(function() {
               $httpBackend.expectGET('/api/1').response(expectedResponse);
               $controller('SimpleController', { $scope: scope });
    
               // causes the http requests which will be issued by myService to be completed synchronously, and thus will process the fake response we defined above with the expectGET
               $httpBackend.flush();
            });
    
            it('using loadData()', function() {
              scope.loadData();
              expect(scope.data).toEqual(expectedResponse);
            });
    
            it('using loadData2()', function () {
              scope.loadData2();
              expect(scope.data).toEqual(expectedResponse);
            });
        });
    
        describe('should fail to load data', function() {
            beforeEach(function() {
               $httpBackend.expectGET('/api/1').response(500); // return 500 - Server Error
               $controller('SimpleController', { $scope: scope });
               $httpBackend.flush();
            });
    
            it('using loadData()', function() {
              scope.loadData();
              expect(scope.error).toEqual('ERROR');
            });
    
            it('using loadData2()', function () {
              scope.loadData2();
              expect(scope.error).toEqual('ERROR');
            });
        });           
    });
    

    2 - Return a fully-mocked promise

    If the thing you're testing has complicated dependencies and all the set-up is a headache, you may still want to mock the services and the calls themselves as you have attempted. The difference is that you'll want to fully mock promise. The downside of this can be creating all the possible mock promises, however you could make that easier by creating your own function for creating these objects.

    The reason this works is because we pretend that it resolves by invoking the handlers provided by success, error, or then immediately, causing it to complete synchronously.

    'use strict';
    
    describe('SimpleControllerTests', function () {
    
        var scope;
        var expectedResponse = { name: 'this is a mocked response' };
        var $controller, _mockMyService, _mockPromise = null;
    
        beforeEach(module('myApp'));
    
        beforeEach(inject(function(_$rootScope_, _$controller_){ 
            $controller = _$controller_;
            scope = _$rootScope_;
    
            _mockMyService = {
                get: function() {
                   return _mockPromise;
                }
            };
        }));
    
        describe('should load data successfully', function() {
    
            beforeEach(function() {
    
              _mockPromise = {
                 then: function(successFn) {
                   successFn(expectedResponse);
                 },
                 success: function(fn) {
                   fn(expectedResponse);
                 }
              };
    
               $controller('SimpleController', { $scope: scope, myService: _mockMyService });
            });
    
            it('using loadData()', function() {
              scope.loadData();
              expect(scope.data).toEqual(expectedResponse);
            });
    
            it('using loadData2()', function () {
              scope.loadData2();
              expect(scope.data).toEqual(expectedResponse);
            });
        });
    
        describe('should fail to load data', function() {
            beforeEach(function() {
              _mockPromise = {
                then: function(successFn, errorFn) {
                  errorFn();
                },
                error: function(fn) {
                  fn();
                }
              };
    
              $controller('SimpleController', { $scope: scope, myService: _mockMyService });
            });
    
            it('using loadData()', function() {
              scope.loadData();
              expect(scope.error).toEqual("ERROR");
            });
    
            it('using loadData2()', function () {
              scope.loadData2();
              expect(scope.error).toEqual("ERROR");
            });
        });           
    });
    

    I rarely go for option 2, even in big applications.

    For what it's worth, your loadData and loadData2 http handlers have an error. They reference response.data but the handlers will be called with the parsed response data directly, not the response object (so it should be data instead of response.data).

    0 讨论(0)
  • 2021-01-31 10:51

    Don't mix concerns!

    Using $httpBackend inside a controller is a bad Idea since you are mixing concerns inside your Test. Whether you retrieve data from an Endpoint or not is not a concern of the Controller, is a concern of the DataService you are calling.

    You can see this more clearly if you change the Endpoint Url inside the service you will then have to modify both tests: the service Test and the Controller Test.

    Also as previously mentioned, the use of success and error are syntactic sugar and we should stick to the use of then and catch. But in reality you may find yourself in the need of testing "legacy" code. So for that I'm using this function:

    function generatePromiseMock(resolve, reject) {
        var promise;
        if(resolve) {
            promise = q.when({data: resolve});
        } else if (reject){
            promise = q.reject({data: reject});
        } else {
            throw new Error('You need to provide an argument');
        }
        promise.success = function(fn){
            return q.when(fn(resolve));
        };
        promise.error = function(fn) {
            return q.when(fn(reject));
        };
        return promise;
    }
    

    By calling this function you will get a true promise that respond to then and catch methods when you need to and will also work for the success or error callbacks. Note that the success and error returns a promise itself so it will work with chained then methods.

    (NOTE: On the 4th and 6th line the function returns resolve and reject values inside the data property of an object. This is to mock the Behavior of $http since it returns the data, http Status etc.)

    0 讨论(0)
  • 2021-01-31 10:55

    Yes, do not use $httpbackend in your controller, because we don't need to make real requests, you just need to make sure that one unit is doing it's job exactly as expected, have a look on this simple controller tests, it's easy to understand

    /**
     * @description Tests for adminEmployeeCtrl controller
     */
    (function () {
    
        "use strict";
    
        describe('Controller: adminEmployeeCtrl ', function () {
    
            /* jshint -W109 */
            var $q, $scope, $controller;
            var empService;
            var errorResponse = 'Not found';
    
    
            var employeesResponse = [
                {id:1,name:'mohammed' },
                {id:2,name:'ramadan' }
            ];
    
            beforeEach(module(
                'loadRequiredModules'
            ));
    
            beforeEach(inject(function (_$q_,
                                        _$controller_,
                                        _$rootScope_,
                                        _empService_) {
                $q = _$q_;
                $controller = _$controller_;
                $scope = _$rootScope_.$new();
                empService = _empService_;
            }));
    
            function successSpies(){
    
                spyOn(empService, 'findEmployee').and.callFake(function () {
                    var deferred = $q.defer();
                    deferred.resolve(employeesResponse);
                    return deferred.promise;
                    // shortcut can be one line
                    // return $q.resolve(employeesResponse);
                });
            }
    
            function rejectedSpies(){
                spyOn(empService, 'findEmployee').and.callFake(function () {
                    var deferred = $q.defer();
                    deferred.reject(errorResponse);
                    return deferred.promise;
                    // shortcut can be one line
                    // return $q.reject(errorResponse);
                });
            }
    
            function initController(){
    
                $controller('adminEmployeeCtrl', {
                    $scope: $scope,
                    empService: empService
                });
            }
    
    
            describe('Success controller initialization', function(){
    
                beforeEach(function(){
    
                    successSpies();
                    initController();
                });
    
                it('should findData by calling findEmployee',function(){
                    $scope.findData();
                    // calling $apply to resolve deferred promises we made in the spies
                    $scope.$apply();
                    expect($scope.loadingEmployee).toEqual(false);
                    expect($scope.allEmployees).toEqual(employeesResponse);
                });
            });
    
            describe('handle controller initialization errors', function(){
    
                beforeEach(function(){
    
                    rejectedSpies();
                    initController();
                });
    
                it('should handle error when calling findEmployee', function(){
                    $scope.findData();
                    $scope.$apply();
                    // your error expectations
                });
            });
        });
    }());
    
    0 讨论(0)
提交回复
热议问题