Создание заглушки для службы angularJs с функцией, которая возвращает обещание с использованием sinon

Я пытаюсь проверить контроллер, который вызывает метод в службе. Сервисный метод возвращает обещание, и контроллер немедленно вызывает .then() встроенный после вызова метода обслуживания. Я пытаюсь заглушить службу, используя sinon, а Jasmine продолжает выдавать ошибку, говоря, что then не определено и не является функцией.

Вот контроллер:

var loginModalController = function ($scope, authenticationService) {

    this.submit = submit;

    function submit(user, password) {
        $scope.email = user;
        authenticationService.login(user, password)
                .then(handleSuccessLogin, handleErrorLogin);
    }
}

Вот услуга:

function authenticationService($http, $q, endPointService) {

    var baseUri = endPointService.getApiEndpoint();

    var service = {
        getTermsAndConditions: getTermsAndConditions,
        login: login,
        acceptTerms: acceptTerms
    };

    return service;

    function getTermsAndConditions() {
        ...
    };

    function login(user, password) {
        var deferred = $q.defer();
        $http({ method: 'POST', url: baseUri + '/api/tokens', data: { username: user, password: password } }).
           success(function (data, status, headers, config) {
               $http.defaults.headers.common.Authorization = 'Basic ' + data.EncryptedTokenId;
               deferred.resolve(data);
           }).
           error(function (data, status, headers, config) {
               deferred.reject(status);
           });
        return deferred.promise;
    };

    function acceptTerms() {
        ...
    };
}

А вот и тест:

describe('loginModalController', function () {
    var scope, loginModalController, authenticationServiceMock, localSaverServiceMock;
    var loginInformationMock = { 'firstName': 'Testuser' };

    beforeEach(function () {
        module('clientAppModule');

        inject(function ($rootScope, $controller, authenticationService, localSaverService) {
            scope = $rootScope.$new();

            authenticationServiceMock = sinon.stub(authenticationService)
                .login.returns({ then: function () { return loginInformationMock } });

            localSaverServiceMock = sinon.stub(localSaverService);

            loginModalController = $controller('loginModalController', {
                $scope: scope,
                $state: {},
                authenticationService: authenticationServiceMock,
                errorCodes: {}, 
                localSaverService: localSaverServiceMock
            });
        });
    });

    it('should login', function () {
        loginModalController.submit("test", "test");
    });
});

1 ответ

Решение

Четыре проблемы с моим кодом:

  • Я без необходимости использовал Sinon
  • Я использовал возвращаемое значение stub() вместо того, чтобы просто позволить ему заглушить службу.
  • Я не использовал $q вернуть отложенное обещание, чтобы соответствовать функции входа в систему.
  • Мне нужно было позвонить $digest() в области, чтобы получить отложенное обещание, чтобы решить перед утверждением.

Итак, вот исправленный тестовый код:

beforeEach(module('clientAppModule'));

describe('loginModalController', function () {
    var scope, authenticationService, localSaverService;
    var loginInformationMock = { 'firstName': 'Testuser' };

    beforeEach(inject(function ($injector, $rootScope, $controller, $q) {

        scope = $rootScope.$new();
        scope.$close = function () { };

        authenticationService = $injector.get('authenticationService');
        localSaverService = $injector.get('localSaverService');            

        spyOn(authenticationService, 'login').and.callFake(function () {
            var deferred = $q.defer();
            deferred.resolve(loginInformationMock);

            return deferred.promise;
        });
        spyOn(localSaverService, 'saveLoginInformation').and.stub();

        $controller('loginModalController', {
            $scope: scope,
            $rootScope: {},
            $state: {},
            authenticationService: authenticationService,
            errorCodes: {},
            localSaverService: localSaverService
        });
    }));

    it('should call login on authenticationService', function () {
        // Arrange

        // Act
        scope.submit("test", "test");

        // Assert
        expect(authenticationService.login).toHaveBeenCalled();
    });

    it('should save login info after successful login', function () {
        // Arrange

        // Act
        scope.submit("test", "test");
        scope.$digest();

        // Assert
        expect(localSaverService.saveLoginInformation).toHaveBeenCalled();
    });
});
Другие вопросы по тегам