Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

rspec 2: detect call to method but still have it perform its function

Tags:

rspec2

I want to check if a method was called exactly(n) times, but I still want that method to perform its original function. Consider a simple thumbnailing system that caches the thumbnail file and make sure ImageMagick's "convert" executable that creates the thumbnail is only called on the first request.

  it "this passes: should detect a cached version" do
    thumbnail_url = thumbnail_url_for("images/something.jpg")
    get thumbnail_url
    last_response.should be_ok
    Sinatra::Thumbnail.should_not_receive(:convert)
    get thumbnail_url
    last_response.should be_ok
  end

  it "this fails:  should detect a cached version" do
    Sinatra::Thumbnail.should_receive(:convert).exactly(1).times
    thumbnail_url = thumbnail_url_for("images/something.jpg")
    get thumbnail_url
    last_response.should be_ok
    get thumbnail_url
    last_response.should be_ok
 end

In my case I get away with my first attempt, but there could be cases where I don't. The second one fails because the call Thumbnail.convert is detected but the method itself doesn't do anything. Is there some way to just detect the call to the method and have it do it's original thing?

BTW: I suspect this question is very similar, but then I get lost in the description and also it's unanswered...

like image 883
Joao Tavora Avatar asked Aug 26 '11 14:08

Joao Tavora


2 Answers

Now there's an and_call_original method precisely for this use case. (RSpec 2.12)

Sinatra::Thumbnails.should_receive(:convert).and_call_original

The documentation can be found on the same page referenced by Joao, here.

See also: the changelog

like image 83
kpassa615 Avatar answered Jan 01 '23 09:01

kpassa615


Yay! I think I figured it out!

it "should detect a cached version" do
  original_method = Sinatra::Thumbnails.method(:convert)
  Sinatra::Thumbnails.should_receive(:convert).exactly(1).times do |*args|
    original_method.call(*args)
  end
  thumbnail_url = thumbnail_url_for("images/something.jpg") # 
  get thumbnail_url
  last_response.should be_ok
  get thumbnail_url
  last_response.should be_ok
end

It's documented (poorly, in my opinion) in here at the very end...

like image 23
Joao Tavora Avatar answered Jan 01 '23 07:01

Joao Tavora