Ответ 1
Вы можете вводить зависимости заглушки при загрузке в своем модуле:
angular.mock.module('curriculumModule', function($provide){
$provide.value('$window', {location:{href:'dummy'}});
});
Я хотел бы unit test выполнить следующий сервис AngularJs:
.factory('httpResponseInterceptor', ['$q', '$location', '$window', 'CONTEXT_PATH', function($q, $location, $window, contextPath){
return {
response : function (response) {
//Will only be called for HTTP up to 300
return response;
},
responseError: function (rejection) {
if(rejection.status === 405 || rejection.status === 401) {
$window.location.href = contextPath + '/signin';
}
return $q.reject(rejection);
}
};
}]);
Я пробовал со следующим пакетом:
describe('Controllers', function () {
var $scope, ctrl;
beforeEach(module('curriculumModule'));
beforeEach(module('curriculumControllerModule'));
beforeEach(module('curriculumServiceModule'));
beforeEach(module(function($provide) {
$provide.constant('CONTEXT_PATH', 'bignibou'); // override contextPath here
}));
describe('CreateCurriculumCtrl', function () {
var mockBackend, location, _window;
beforeEach(inject(function ($rootScope, $controller, $httpBackend, $location, $window) {
mockBackend = $httpBackend;
location = $location;
_window = $window;
$scope = $rootScope.$new();
ctrl = $controller('CreateCurriculumCtrl', {
$scope: $scope
});
}));
it('should redirect to /signin if 401 or 405', function () {
mockBackend.whenGET('bignibou/utils/findLanguagesByLanguageStartingWith.json?language=fran').respond([{"description":"Français","id":46,"version":0}]);
mockBackend.whenPOST('bignibou/curriculum/new').respond(function(method, url, data, headers){
return [401];
});
$scope.saveCurriculum();
mockBackend.flush();
expect(_window.location.href).toEqual("/bignibou/signin");
});
});
});
Однако он не работает со следующим сообщением об ошибке:
PhantomJS 1.9.2 (Linux) Controllers CreateCurriculumCtrl should redirect to /signin if 401 or 405 FAILED
Expected 'http://localhost:9876/context.html' to equal '/bignibou/signin'.
PhantomJS 1.9.2 (Linux) ERROR
Some of your tests did a full page reload!
Я не уверен, что происходит не так и почему. Кто-нибудь может помочь?
Я просто хочу, чтобы $window.location.href
был равен '/bignibou/signin'
.
изменить 1:
Мне удалось заставить его работать следующим образом (благодаря "dskh" ):
beforeEach(module('config', function($provide){
$provide.value('$window', {location:{href:'dummy'}});
}));
Вы можете вводить зависимости заглушки при загрузке в своем модуле:
angular.mock.module('curriculumModule', function($provide){
$provide.value('$window', {location:{href:'dummy'}});
});
Чтобы заставить меня работать для меня, мне пришлось сделать небольшую корректировку. Было бы ошибкой и сказать:
TypeError: 'undefined' is not an object (evaluating '$window.navigator.userAgent')
Итак, я добавил объект navigator.userAgent
, чтобы заставить его работать для меня.
$provide.value('$window', {
location:{
href:'dummy'
},
navigator:{
userAgent:{}
}
});
Я столкнулся с той же проблемой и пошел дальше в своем решении. Я не просто хотел насмехаться, я хотел заменить $window.location.href
шпионом Jasmine для лучшей способности отслеживать сделанные на него изменения. Итак, я узнал из примера apsiller для шпионажа на геттеры/сеттеры и после создания моего макета, я смог заглянуть в собственность, которую я хотел.
Во-первых, вот набор, который показывает, как я высмеивал $window
, с тестом, чтобы продемонстрировать, что шпион работает как ожидалось:
describe("The Thing", function() {
var $window;
beforeEach(function() {
module("app", function ($provide) {
$provide.value("$window", {
//this creates a copy that we can edit later
location: angular.extend({}, window.location)
});
});
inject(function (_$window_) {
$window = _$window_;
});
});
it("should track calls to $window.location.href", function() {
var hrefSpy = spyOnProperty($window.location, 'href', 'set');
console.log($window.location.href);
$window.location.href = "https://www.google.com/";
console.log($window.location.href);
expect(hrefSpy).toHaveBeenCalled();
expect(hrefSpy).toHaveBeenCalledWith("https://www.google.com/");
});
});
Как вы можете видеть выше, шпион генерируется вызовом функции ниже: (он работает как для get
, так и set
)
function spyOnProperty(obj, propertyName, accessType) {
var desc = Object.getOwnPropertyDescriptor(obj, propertyName);
if (desc.hasOwnProperty("value")) {
//property is a value, not a getter/setter - convert it
var value = desc.value;
desc = {
get: function() { return value; },
set: function(input) { value = input; }
}
}
var spy = jasmine.createSpy(propertyName, desc[accessType]).and.callThrough();
desc[accessType] = spy;
Object.defineProperty(obj, propertyName, desc);
return spy;
}
Наконец, здесь скрипка, демонстрирующая это в действии. Я тестировал это против Angular 1.4 и Jasmine 2.3 и 2.4.