Angularjs 单元测试解决承诺
Angularjs unit testing resolving promises
我可能做错了,但我不知道如何解决它。
我想测试一个使用资源 (ngResource) 的控制器,我想使用 Spy 作为资源的测试替身,这样它实际上不会进行 http 调用。在下面的代码中,我只想测试控制器中的搜索功能。
控制器:
controllers = angular.module('app.controllers');
controllers.controller('landingCtrl', ['$scope', '$q', 'categoryResource', function ($scope, $q, categoryResource) {
$scope.search = function (text) {
console.log('searching for: ' + text);
var deferred = $q.defer();
categoryResource.query({ searchTerm: text }, function (result) {
if (result.length == 0) {
deferred.resolve(['No results found']);
} else {
deferred.resolve(result);
}
});
return deferred.promise;
}
}]);
服务:
var services = angular.module('app.services');
services.factory('categoryResource', ['$resource', function ($resource) {
var resource = $resource('/api/category');
return resource;
}]);
landingCtrl 规范:
describe('Controller: landingCtrl ', function () {
var $q,
$rootScope,
$scope;
beforeEach(module('ngResource'));
beforeEach(module('app.services'));
beforeEach(module('app.controllers'));
beforeEach(inject(function (_$rootScope_, _$q_) {
$q = _$q_;
$rootScope = _$rootScope_;
}));
// mock any depencencies, like scope. $resource or $http
beforeEach(inject(function ($controller, $injector, categoryResource) {
$scope = $rootScope.$new();
spyOn(categoryResource, 'query').andCallFake(function (searchText) {
console.log('query fake being called');
var deferred = $q.defer();
deferred.resolve(['Test', 'Testing', 'Protester']);
return deferred.promise;
});
landingCtrl = $controller('landingCtrl', {
'$scope': $scope,
'$q': $q,
'categoryResource': categoryResource
});
}));
afterEach(inject(function ($rootScope) {
$rootScope.$apply();
}));
it('should return words with "test" in them"', function () {
$scope.search('test').then(function (results) {
console.log(results);
expect(results).toContain('Test');
});
$scope.$apply();
});
});
测试执行时没有错误,但它在没有解决承诺的情况下通过了,所以我在 "then" 函数中的代码永远不会被调用。我做错了什么?
我用上面的代码创建了一个 plunker 和一个应该会失败的测试:
您的规范在嘲笑 categoryResource.query()
,因此它 returns 是一个承诺,但您的控制器并不期望如此。它调用 query()
并传递一个回调,并在该回调中执行它的操作。换句话说,您的规范并未测试您的控制器的功能。
这是您的固定规格:
spyOn(categoryResource, 'query').andCallFake(function (searchText, callback) {
console.log('query fake being called');
callback(['Test', 'Testing', 'Protester']);
});
...
it('should return words with "test" in them"', function () {
var results;
$scope.search('test').then(function (_results_) {
console.log(results);
results = _results_;
});
$scope.$apply();
expect(results).toContain('Test');
});
请注意,我已将期望值移到 then()
回调之外,因此如果未解决承诺,您的测试就会中断。
我可能做错了,但我不知道如何解决它。
我想测试一个使用资源 (ngResource) 的控制器,我想使用 Spy 作为资源的测试替身,这样它实际上不会进行 http 调用。在下面的代码中,我只想测试控制器中的搜索功能。
控制器:
controllers = angular.module('app.controllers');
controllers.controller('landingCtrl', ['$scope', '$q', 'categoryResource', function ($scope, $q, categoryResource) {
$scope.search = function (text) {
console.log('searching for: ' + text);
var deferred = $q.defer();
categoryResource.query({ searchTerm: text }, function (result) {
if (result.length == 0) {
deferred.resolve(['No results found']);
} else {
deferred.resolve(result);
}
});
return deferred.promise;
}
}]);
服务:
var services = angular.module('app.services');
services.factory('categoryResource', ['$resource', function ($resource) {
var resource = $resource('/api/category');
return resource;
}]);
landingCtrl 规范:
describe('Controller: landingCtrl ', function () {
var $q,
$rootScope,
$scope;
beforeEach(module('ngResource'));
beforeEach(module('app.services'));
beforeEach(module('app.controllers'));
beforeEach(inject(function (_$rootScope_, _$q_) {
$q = _$q_;
$rootScope = _$rootScope_;
}));
// mock any depencencies, like scope. $resource or $http
beforeEach(inject(function ($controller, $injector, categoryResource) {
$scope = $rootScope.$new();
spyOn(categoryResource, 'query').andCallFake(function (searchText) {
console.log('query fake being called');
var deferred = $q.defer();
deferred.resolve(['Test', 'Testing', 'Protester']);
return deferred.promise;
});
landingCtrl = $controller('landingCtrl', {
'$scope': $scope,
'$q': $q,
'categoryResource': categoryResource
});
}));
afterEach(inject(function ($rootScope) {
$rootScope.$apply();
}));
it('should return words with "test" in them"', function () {
$scope.search('test').then(function (results) {
console.log(results);
expect(results).toContain('Test');
});
$scope.$apply();
});
});
测试执行时没有错误,但它在没有解决承诺的情况下通过了,所以我在 "then" 函数中的代码永远不会被调用。我做错了什么?
我用上面的代码创建了一个 plunker 和一个应该会失败的测试:
您的规范在嘲笑 categoryResource.query()
,因此它 returns 是一个承诺,但您的控制器并不期望如此。它调用 query()
并传递一个回调,并在该回调中执行它的操作。换句话说,您的规范并未测试您的控制器的功能。
这是您的固定规格:
spyOn(categoryResource, 'query').andCallFake(function (searchText, callback) {
console.log('query fake being called');
callback(['Test', 'Testing', 'Protester']);
});
...
it('should return words with "test" in them"', function () {
var results;
$scope.search('test').then(function (_results_) {
console.log(results);
results = _results_;
});
$scope.$apply();
expect(results).toContain('Test');
});
请注意,我已将期望值移到 then()
回调之外,因此如果未解决承诺,您的测试就会中断。