在RSpec 3.2中存储实例变量

时间:2015-03-29 18:33:58

标签: ruby-on-rails ruby rspec

我在Rails中使用RSpec 3.2,并想知道如果实例变量不能通过getter方法公开访问(例如使用attr_accessor或类似方法),我将如何存根实例变量。

考虑下面的简单示例 -

require 'rails_helper'

class A
  def initialize
    @x = 3
  end

  def add(n)
    sum = @x + n
    puts "Sum is #{sum}"
    sum
  end
end


RSpec.describe A do
  before(:all) do
    @a = A.new
  end

  it 'computes the sum correctly' do
    # For this test I want to stub the value of @x and return 5
    allow(@a).to receive(:x) { 5 }
    # 5 + 8 should return 13
    expect(@a.add(8)).to eq(13)
  end
end

在这种情况下,尝试存根@x是不可能的,因为该类永远不会收到x的消息或方法调用。 RSpec输出证实了这一点 -

Failures:

  1) A computes the sum correctly
     Failure/Error: allow(@a).to receive(:@x) { 5 }
       #<A:0x007fe6e66ab8d0 @x=3> does not implement: @x
     # ./spec/unit/test_spec.rb:25:in `block (2 levels) in <top (required)>'

我可以通过使实例变量@x可访问(attr_accesssor :x)并将调用替换为@x self.x来解决这个问题,但这看起来很糟糕,可能不会在我更复杂的实施中是可能的。

有没有更好的方法来存根?

谢谢!

1 个答案:

答案 0 :(得分:9)

我认为这不是一种正确的做法。 Rspec应该测试类的接口行为,而不是内部实现。

由于您没有使用访问者,因此可以使用#instance_variable_set&amp; #instance_variable_get来操纵并获取实例变量。

获取和设置如下:

@a.instance_variable_set(:@x, 5)
@a.instance_variable_get(:@x)
#=> 5

在您的代码中:

@a.instance_variable_set(:@x, 5)
expect(@a.add(8)).to eq(13)