How to test message expectations?
JeanMertz opened this issue · 4 comments
I was wondering how I should test message expectations using should_receive
with rspec-given?
Here's an example I have:
context 'persists correct signup data' do
Given do
subject.params = { signup: build(:signup).attributes }
subject.signup.stub(:persist!)
end
When { subject.call }
Then { subject.signup.should_receive(:save).and_return(true) }
end
This fails for not receiving the message.
(#<Signup:0x007fb4598206a8>).save(any args)
expected: 1 time
received: 0 times
I know the method is being called, because I used to check expect{ subject.call }.to change{ User.count }
, but I'd like to rewrite the spec to never hit the database.
I wonder if it's because I am misusing rspec-given in this case? The reason I ask is that I call the method in the When
clause, and after that set the expectation in the Then
clause. This seems counter to I normally write it in RSpec (first an expectation, then the action).
I think you are not doing something right. It looks like you are testing that a stubbed method returns true... Which in this case you haven't told it to, and would be pointless to test this because you are trying to test a stubbed method.
I would say do not worry about testing persistence. If you merely test that the record is valid you can assume it will persist... "foo.should be_valid"
What mocking framework are you using? With most mock libraries, you have to setup the expectations before the When code is run (i.e. in a Given or before block).
If want to check expectations in a Then clause, then you need to use a mock framework that supports spies. Flexmock will support spies. I've heard that RSpec mocks also support them (or will soon? I don't have details).
See https://github.com/jimweirich/flexmock#spies for details on using spies with flexmock.
It looks like you are testing that a stubbed method returns true
No, I haven't stubbed save
, I've only stubbed persist!
(so it doesn't touch the database). This is not a standard AR model, so that's why I want to test if save is called correctly.
However, I'm not quite sure if I can use should_receive
on a non-stubbed method. I'm not even sure if I've written the correct expectation using the default rspec mocks.
Anyway, thank you for your responses, I will continue investigating why this fails.
Looks like spies will be in RSpec 2.14 (which isn't out yet). In the meantime, you can try out https://github.com/technicalpickles/rspec-spies