在Ember App Kit中,有许多测试示例随初始回购一起提供。其中之一是基本的路线单元测试。如果数据在模型钩子中是硬编码的,那么这个测试是微不足道的,如下所示:
test("#model", function(){
deepEqual(route.model(), ['red', 'yellow', 'blue']);
});
如果从ember-data返回一个promise,你如何使用隔离容器测试模型钩子?
以下是测试:
import Activities from 'appkit/routes/activities';
var route;
module("Unit - ActivitiesRoute", {
setup: function(){
var container = isolatedContainer([
'route:activities'
]);
route = container.lookup('route:activities');
}
});
test("#model", function(){
deepEqual(route.model(), ['activity', 'activity2', 'activity3']);
});
实际的Route的模型钩子:
export default Ember.Route.extend({
model: function() {
return this.get('store').find('activity');
}
});
从kingpin2k实施以下不同方法后。以下是结果摘要。
第一种方法:效果很好......但没有承诺。
第二种方法:返回promise对象(似乎已解析),但数组和正确的值都分配给_detail属性。
test("#model", function(){
deepEqual(route.model()['_detail'], ['activity', 'activity2', 'activity3']); //passes
});
我希望在模块设置()中处理商店创建。
...
module("Unit - ActivitiesRoute", {
setup: function(){
var container = isolatedContainer([
'route:activities'
]);
route = container.lookup('route:activities');
var store = {
find: function(type){
return new Em.RSVP.Promise(function(resolve){
resolve(['activity', 'activity2', 'activity3']); // or made up model(s) here
});
}
};
route.set('store', store);
}
});
测试:
test("#model", function(){
deepEqual(route.model(), ['activity', 'activity2', 'activity3']); // ???
});
第三种方法:
...
module('Unit - ActivitiesRoute', {
setup: function() {
var container = isolatedContainer([
'route:activities'
]);
route = container.lookup('route:activities');
var store = {
find: function() {
var promise = new Ember.RSVP.Promise(function(resolve) {
Em.run.later(function() {
resolve(Activity.FIXTURES);
}, 10);
});
return Ember.ArrayProxy.extend(Ember.PromiseProxyMixin).create({
promise: promise
});
}
};
route.set('store', store);
}
});
在测试中,调用route.model()会返回一个空对象{}:
test("#model", function(){
deepEqual(route.model(), Activity.FIXTURES); // returns {}
});
还需要添加asyncTest()而不是test(),并调用start()以防止测试运行器挂起。
asyncTest('#model', function(){
Em.run(function(){
route.model().then(function(result){
ok(result);
equal(result, Activity.FIXTURES);
start();
});
});
});
答案 0 :(得分:5)
简单的方法,它是一个单元测试,所以你真的没有测试商店,所以设置一个模拟商店和结果。
route = container.lookup('route:activities');
var store = {
find: function(type){
equal(type, 'activity', 'type is activity');
return ['activity', 'activity2', 'activity3'];
}
}
route.set('store', store);
更好的是,您也可以复制承诺
route = container.lookup('route:activities');
var store = {
find: function(type){
equal(type, 'activity', 'type is activity');
return new Em.RSVP.Promise(function(resolve){
resolve(['activity', 'activity2', 'activity3']); // or made up model(s) here
});
}
}
route.set('store', store);
如果您想更密切地复制Ember Data,可以使用实现PromiseProxyMixin的ArrayProxy ......
route = container.lookup('route:activities');
var store = {
find: function(type){
equal(type, 'activity', 'type is activity');
var promise = new Ember.RSVP.Promise(function(resolve){
Em.run.later(function(){
resolve(['activity', 'activity2', 'activity3']);
}, 10);
});
return Ember.ArrayProxy.extend(Ember.PromiseProxyMixin).create({
promise: promise
});
}
}
route.set('store', store);
使用您的上一种方法,您应该像这样实现它
test("#model", function(){
route.model().then(function(result){
deepEqual(result, Activity.FIXTURES); // returns {}
});
});
但是,这里有一些棘手的问题,因为它有一个异步响应你需要将它包装在Ember运行循环中
test("#model", function(){
Em.run(function(){
route.model().then(function(result){
deepEqual(result, Activity.FIXTURES); // returns {}
});
});
});