我的项目正在使用AngularJS + Kendo-UI。我试图测试一个使用Kendo-UI Grid的控制器:
angular.module('myApp')('DevicesCtrl', function ($scope) {
$scope.gridOptions = {
dataSource: {
transport: {
read: {
url: "/devices",
dataType: "json"
}
},
change: function(){
var view = this.view();
$scope.devices = [];
$.each(view, function(i, v) {
$scope.devices.push({id: v.id, description: v.name, status: v.status == 0 ? "failure" : "success"});
});
$scope.$apply();
}
},
columns: [
{
field: "name",
title: "Name",
width: 250,
template: function (item) {
var itemStatus = item.status == 0 ? 'failure' : 'success';
return '<div label size="small" operation="' + itemStatus + '" label="' + item.name + '"></div>';
}
},
{
field: "status",
title: "Status"
}
]
};
});
当我编写单元测试时,我预计会调用GET请求:
describe('deviceCtrl', function () {
var scope, deviceCtrl, httpBackend, timeout;
beforeEach(module("myApp"));
beforeEach(module('src/modules/devices/views/device.html'));
beforeEach(inject(function ($controller, $rootScope, $httpBackend, $timeout, $state) {
scope = $rootScope.$new();
httpBackend = $httpBackend;
timeout = $timeout;
httpBackend.expectGET('languages/en_US.json').respond({});
deviceCtrl = $controller("DeviceCtrl", {
$scope: scope
});
$state.go("devices");
timeout.flush()
}));
it('should load the switch list', function () {
httpBackend.expectGET("/devices").respond(
[{"id":"1","name":"name 1","status":"1"},
{"id":"2","name":"name 2","status":"2"}]
);
httpBackend.flush();
});
});
但期望永远不会满足,没有提出要求。 所以我的问题是:有没有办法让Kendo Grid / Datasource进行这个调用,所以我可以模拟它?
我看到了一些关于如何使用Mockjax(http://www.telerik.com/forums/best-practice-mocking-with-datasource)的示例,但我更喜欢使用角度库来做到这一点。
答案 0 :(得分:6)
经过大量研究后,我可以找到一些方法来测试使用Kendo Datasource的控制器。
Kendo有自己的方式来进行Ajax调用以获取数据,并且不使用常规的Angular $ http来执行此操作,因此使用Angular工具(角度模拟)进行测试有点棘手。 我们来看看选项:
1 - 使用常规的Angular方式进行Ajax调用。
Kendo让我们可以改变获取数据的方式,而不是:
dataSource: new kendo.data.DataSource({
transport: {
read: {
url: "/devices,
dataType: "json"
}
},
change: function(){
var view = this.view();
$scope.devices = [];
$.each(view, function(i, v) {
$scope.devices.push({id: v.id, description: v.name, status: v.status == 0 ? "failure" : "success"});
});
$scope.$apply();
}
});
我们将改为:
dataSource: new kendo.data.DataSource({
transport: {
read: function(options){
$http.get("/devices").then(function(response){
options.success(response.data);
$scope.devices = [];
response.data.forEach(function(v){
$scope.devices.push({id: v.id, description: v.name, status: v.status == 0 ? "failure" : "success"});
});
});
}
}
});
然后我们可以使用常规$ httpBackend.expectGET(url)来模拟Ajax调用。 我个人喜欢这种方法,因为我们有更多的控制权。 Ps。:使用函数内部的变量“options”,我们可以访问Grid过滤器,排序和分页值。
2 - 模拟Kendo Ajax Call。
通过这种方式,我们在Controller中几乎没有任何改变,唯一需要改变的是使用新的kendo.data.DataSource({})创建一个新的Datasource而不是仅传递选项。 这需要,因为我们在测试用例中调用了read函数。 我尝试了不同的方法来模拟这个Ajax请求,但我唯一可以使它工作的是使用名为Jasmine-Ajax的工具。 在测试用例中,我们会写一些类似的东西:
var request = jasmine.Ajax.requests.mostRecent();
request.response(MockData);
我希望这可以帮助其他人。