0

我是 rspec 的新手,嘲笑和存根。我慢慢地开始欣赏并围绕孤立测试和模拟/存根的概念。我有一个基本问题,我认为通过代码更容易解​​释:

class NewsItem < ActiveRecord::Base
  ...
  scope :ordered, order("created_at DESC")
  ...
end

现在在我的模型测试中,我希望测试返回 news_items 有序列表的行为。使用 FactoryGirl DB-touching 测试,我实现了如下:

# TODO use mocking and stubbing here
it "should have an ordered method to retrieve news items in the descending order" do
  news_item_x = create(:news_item, created_at: DateTime.new(2012,01,01))
  news_item_y= create(:news_item, created_at: DateTime.new(2011,01,01))
  news_item_z= create(:news_item, created_at: DateTime.new(2012,03,01))

  # .all added to avoid comparison failure between groups of NewsItem objects an ActiveRel group.
  expect(NewsItem.ordered.all).to eql([news_item_z, news_item_x, news_item_y])
end 

我不知道如何将上述测试转换为模拟和存根。这是第一次尝试,但显然我在这里误解了一些核心概念。

xit "should have an ordered method to retrieve news items in the descending order" do

  news_item_x = mock(NewsItem, created_at: DateTime.new(2012,01,01))
  news_item_y = mock(NewsItem, created_at: DateTime.new(2011,01,01))
  news_item_z = mock(NewsItem, created_at: DateTime.new(2012,03,01))

  NewsItem.should_receive(:ordered).and_return([news_item_z, news_item_x, news_item_y])
  # NewsItem.should_receive(:ordered).ordered # not working.

  # this is pointless as it's not checking the order of anything, just the call.
  NewsItem.ordered
end 

在这种测试中模拟/存根是否合适?

任何建议将不胜感激。

判决:

我从@arieljuod 和@zetetic 那里得到了一些很好的答案。对于我最初的问题,在这里嘲笑和存根是否合适?正如@zetetic 指出的那样,答案似乎是否定的。

另一方面,@arieljuod 提供了一种非常好的方法来实际测试我的代码片段(不一定通过模拟和存根)。这两个都是有效的答案。

4

2 回答 2

2

在这种测试中模拟/存根是否合适?

不。

使用模拟和存根的目的是将编写的代码与其依赖项隔离开来。在 的情况下scope,它所依赖的一切都埋在 Rails 框架中。此外,您不需要首先测试框架/库代码的内部——原作者已经这样做了。

于 2013-03-01T03:51:06.207 回答
1

您应该只测试您的“有序”范围在模型上以“created_at DESC”作为参数调用“订单”,至少在那个简单的例子中

describe 'ordered' do
  it 'orders by created_at desc' do
    NewsItem.should_receive(:order).once.with('created_at DESC')
    NewsItem.ordered
  end
end

您可以相信查询将具有您想要的顺序

更复杂的范围可能需要其他规范,但是您始终可以在较小的范围内打破复杂的范围以正确测试它们,并且仅在范围不平凡时才进行真正的数据库查询(就像您首先实际创建对象并运行查询一样) (如果您执行某种奇怪的手动 sql 查询,您应该测试它是否符合您的要求,否则,请信任 rails)

编辑:如评论所述,该代码不起作用,您可以检查 ActiveRelation 对象是否具有所需的顺序集:

describe 'ordered' do
  it 'orders by created_at desc' do
    NewsItem.ordered.order_values.should == ['created_at DESC']
  end
end

这样您就知道活动关系将在查询中使用该顺序

于 2013-03-01T19:52:46.143 回答