在RSpec 3.2中存储实例变量

use*_*003 2 ruby rspec ruby-on-rails

我正在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
Run Code Online (Sandbox Code Playgroud)

@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)>'
Run Code Online (Sandbox Code Playgroud)

我可以通过使实例变量@xaccess(attr_accesssor :x)并将调用替换为@xwith 来解决这个问题self.x,但这看起来很麻烦,在我更复杂的实现中可能无法实现.

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

谢谢!

Sha*_*med 10

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

由于您没有使用访问器,因此可以使用#instance_variable_set&#instance_variable_get来操作和获取实例变量.

获取和设置如下:

@a.instance_variable_set(:@x, 5)
@a.instance_variable_get(:@x)
#=> 5
Run Code Online (Sandbox Code Playgroud)

在你的代码中:

@a.instance_variable_set(:@x, 5)
expect(@a.add(8)).to eq(13)
Run Code Online (Sandbox Code Playgroud)