Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Rspec 3.0 How to mock a method replacing the parameter but with no return value?

Tags:

I've searched a lot and just cannot figure this out although it seems basic. Here's a way simplified example of what I want to do.

Create a simple method that does something but doesn't return anything, such as:

class Test
  def test_method(param)
    puts param
  end
  test_method("hello")
end

But in my rspec test I need to pass a different parameter, such as "goodbye" instead of "hello." I know this has to do with stubs and mocks, and I've looking over the documentation but can't figure it out: https://relishapp.com/rspec/rspec-mocks/v/3-0/docs/method-stubs

If I do:

@test = Test.new
allow(@test).to_receive(:test_method).with("goodbye")

it tells me to stub out a default value but I can't figure out how to do it correctly.

Error message:

received :test_method with unexpected arguments
  expected: ("hello")
  got: ("goodbye")
Please stub a default value first if message might be received with other args as well.     

I am using rspec 3.0, and calling something like

@test.stub(:test_method)

is not allowed.

like image 414
user3060126 Avatar asked Apr 18 '14 02:04

user3060126


2 Answers

How to set a default value that is explained at

and_call_original can configure a default response that can be overriden for specific args

require 'calculator'

RSpec.describe "and_call_original" do
  it "can be overriden for specific arguments using #with" do
    allow(Calculator).to receive(:add).and_call_original
    allow(Calculator).to receive(:add).with(2, 3).and_return(-5)

    expect(Calculator.add(2, 2)).to eq(4)
    expect(Calculator.add(2, 3)).to eq(-5)
  end
end

Source where I came to know about that can be found at https://makandracards.com/makandra/30543-rspec-only-stub-a-method-when-a-particular-argument-is-passed

like image 165
Jignesh Gohel Avatar answered Oct 16 '22 12:10

Jignesh Gohel


For your example, since you don't need to test the actual result of test_method, only that puts gets called in it passing in param, I would just test by setting up the expectation and running the method:

class Test
  def test_method(param)
    puts param
  end
end

describe Test do
  let(:test) { Test.new }

  it 'says hello via expectation' do
    expect(test).to receive(:puts).with('hello')
    test.test_method('hello')
  end

  it 'says goodbye via expectation' do
    expect(test).to receive(:puts).with('goodbye')
    test.test_method('goodbye')
  end
end

What it seems you're attempting to do is set up a test spy on the method, but then I think you're setting up the method stub one level too high (on test_method itself instead of the call to puts inside test_method). If you put the stub on the call to puts, your tests should pass:

describe Test do
  let(:test) { Test.new }

  it 'says hello using a test spy' do
    allow(test).to receive(:puts).with('hello')
    test.test_method('hello')
    expect(test).to have_received(:puts).with('hello')
  end

  it 'says goodbye using a test spy' do
    allow(test).to receive(:puts).with('goodbye')
    test.test_method('goodbye')
    expect(test).to have_received(:puts).with('goodbye')
  end
end
like image 23
Paul Fioravanti Avatar answered Oct 16 '22 12:10

Paul Fioravanti