我在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
来解决这个问题,但这看起来很糟糕,可能不会在我更复杂的实施中是可能的。
有没有更好的方法来存根?
谢谢!
答案 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)