如何测试Ember模型的具有关系依赖性的计算属性?

时间:2014-06-09 15:51:44

标签: javascript testing ember.js qunit ember-qunit

我正在编写Qunit测试以测试Ember模型,但很难测试具有关系依赖性的计算属性(计算属性会触发另一个模型的计算属性)。

正在测试的模型(CoffeeScript):

Customer = DS.Model.extend
  firstName:      DS.attr('string')
  lastName:       DS.attr('string')
  phones:         DS.attr('embedded-list')
phone: (->
    @get('phones.firstObject.number')
  ).property('phones.firstObject.number')

fullName: (->
    [@get('lastName'), @get('firstName')].join(' ') )
  ).property('firstName','lastName')

会议模式:

Meeting = DS.Model.extend
  customers: DS.hasMany('customer')

  startAt:   DS.attr('isodate')
  status:    DS.attr()
  objective: DS.attr()

 customerPhones: (->
    phones = []
    @get('customers').forEach (c) ->
      c.get('phones').forEach (ph) ->
        phones.push(ph.number)
    phones
  ).property('customers.@each.phones')


  firstCustomer: (->
    @get('customers.firstObject')
  ).property('customers.firstObject')

 firstCustomerFullName: (->
    @get('firstCustomer.fullName')
  ).property('firstCustomer.fullName')

现在,测试customerPhonesfirstCustomerFullName让我真的很难过......

我的测试如下:

`import { test, moduleForModel } from 'ember-qunit';`

moduleForModel('meeting', 'App.Meeting',{
   needs: ['model:customer']
   setup: ->
     Ember.run do (t = @)->
       ->
        customer = t.store().createRecord 'customer', firstName: 'John', lastName: 'Smith', phones:[]
        customer.get('phones').addObject(Ember.Object.create({tag: 'home', number: '111222333'}))
        customer.get('phones').addObject(Ember.Object.create({tag: 'work', number: '444555666'}))

        t.subject().set('customers.content', Ember.ArrayProxy.create({content: []}));
        t.subject().get('customers.content').pushObject(customer)    
 teardown: ->
  },(container, context) ->
      container.register 'store:main', DS.Store
      container.register 'adapter:application', DS.FixtureAdapter
      context.__setup_properties__.store = -> container.lookup('store:main')
)

test "it's a DS.Model", -> ok(@subject())

test "attributes: can be created with valid values", ->
  meeting = @subject({objective: 'Follow up'})
  Ember.run ->
    equal(meeting.get('objective', 'Follow up'))


test "properties: firstCustomer & firstCustomerFullName & firstCustomerPhone", ->
  meeting = @subject()
  Ember.run ->
    equal(meeting.get('firstCustomer.fullName'),  'Smith John')
    equal(meeting.get('firstCustomer.phone'),     '111222333')

现在,我在这个测试中使用了一些技术,我在Stack Overflow的答案中找到了,但我现在似乎无法找到它。

几天前,这完全有效,现在(我知道这似乎是胡说八道)每当我运行测试时,都会出错:

  

断言失败:您无法在此关系中添加“会议”记录(仅允许“会议”)

我不知道错误在哪里,也不知道如何解决。花了一整天的时间,没有结果。

我该如何解决这个问题?

2 个答案:

答案 0 :(得分:2)

好的,到目前为止,我的评论太多了,所以我打算做一个WIP答案。

  • 我删除了大部分的运行循环,它们只是异步进程所必需的。

  • 我将部分计算属性更改为computed.alias属性

phone: (->
  @get('phones.firstObject.number')
).property('phones.firstObject.number')

phone: Ember.computed.alias('phones.firstObject.number')
  • 我撕掉了大部分的设置,Ember Data急切地自己加载商店,并且会使用fixture ID等而不指定它。 (这部分可以放回去,在这种情况下也没有必要)。

  },(container, context) ->
  container.register 'store:main', DS.Store
  container.register 'adapter:application', DS.FixtureAdapter
  context.__setup_properties__.store = -> container.lookup('store:main')
  • 我事先道歉,我不是coffeescript的粉丝,所以我把它全部放在js中。现在的问题是,如果您仍然遇到任何问题,我们可能需要找出您正在使用的Ember,ED和Ember Qunit的版本。

http://emberjs.jsbin.com/OxIDiVU/625/edit

答案 1 :(得分:1)

我发现这个问题正在寻找'如何对使用hasMany'的计算属性进行单元测试。

以下是我如何做到的简单示例(感谢Kitler):

冰箱型号:

foods: DS.hasMany('food', {async: true}),

inDateFoods: Ember.computed('foods.@each.{ignoreEndDate,endDate}', function() {
  let foods = this.get('foods');
  let now = moment();
  return foods.filter(f => f.get(ignoreEndDate) || moment(c.get('endDate')).isAfter(now));
})

所以说我们现在想在单元测试中测试inDateFoods?然后在冰箱模型测试文件中执行此操作:

import Ember from 'ember';
import { moduleForModel, test } from 'ember-qunit';
import Fridge from '../../../models/fridge';

Fridge.reopen({
  foods: Ember.computed(() => [])
});

moduleForModel('fridge', 'Unit | Model | fridge', {
  // Specify the other units that are required for this test.
  needs: ['model:food']
});

test('filters correctly', function(assert) {
  assert.expect(1);
  let model = this.subject();
  model.pushObject(Ember.Object.create({foods: [{id: 1, ignoreEndDate: false, endDate: '2050-03-08T00:00:00'});

  assert.equal(model.get('inDateFoods.length'), 1);
});

他们在这里的关键是重新打开你的模型以删除有多个,并在执行this.subject之后推送对象。在重新开放之前,我们收到错误All elements of a hasMany relationship must be instances of DS.Model, you passed [[object Object]] error