Docunext


Rspec 2 Shoulda Rails 3.1 Awesome

October 26th, 2011

I've been digging deeper into Rspec, and along the way I've found some truly amazing features enabled by combining Rspec and shoulda from thoughtbot.

A Brief History of Rspec and Shuolda

First a bit of background - Ruby has been dominated by Ruby on Rails for quite some time, and it ships with Test::Unit. Test::Unit does the job, obviously - to this day tons of developers use Test::Unit on a daily basis. But the fact remains - there is room for improvement. Case in point, Rails 3 actually switched to MiniTest, a Test::Unit clone. Not that Rspec was or is perfect - its gone through a major shift from version 1 to version 2. I thought version 1 was cool, Rspec version 2 is wicked awesome.

How about Shoulda? Shoulda is a gem created by the folks at ThoughtBot - where giant robots smash into other, not-so-giant-robots. They've contributed some seriously awesome gems to the Ruby on Rails community, including Paperclip, high_voltage, and Factory Girl.

Shoulda is certainly in their greatest hits, in my humble opinion. Just think - Shoulda is good to go with Test::Unit, MiniTest, or Rspec, though it originated in the world of Test::Unit. When I first used it, I took it for granted, and assumed it was just part of the Rails testing suite. After reading up on how well it interacts with Rspec, I'm very grateful for Shoulda.

RSpec and Shoulda = Awesome

Note, a lot of the awesomeness I've discovered is thanks to this awesome presentation by Lark.

Take a look at these Rspec examples with Shouda:

describe BlogPost do
  it { should be_invalid }
end

That Rspec example can expand into this beauty:

describe BlogPost do
  subject { BlogPost.new :title => 'foo', :body => 'bar' }
  it "sets published timestamp" do
    subject.publish!
    subject.published_on.should == Date.today
  end
end

and furthermore into this:

describe BlogPost do
  subject do
    BlogPost.new(:title => 'foo', :body => 'bar').tap do |p|
      p.publish!
    end
  end
  it { should be_valid }
  its(:errors) { should be_empty }
  its(:title) { should == 'foo' }
  its(:body) { should == 'bar' }
  its(:published_on) { should == Date.today }
end

Wicked awesome, huh?

As Lark puts it, "Hold it right there."

expect do
  foo.bar
end.to change {baz.quux}.from('corge').to('grault')

Nuff said. Actually, no, I've got more to discuss.

Describe, Context, and It

Using Rspec and Shoulda can be challenging because its so flexible, but sticking to the core has helped me. I also try to keep my tests consistent and to the bare minimum. Like this for example:

describe Make do
  let (:make) { Make.make }
  it { should respond_to :name }
end

It is so to the point! But what about a real test? Not just an obvious confirmation that the associations are still setup right. For me, that's when I bring in context. Note that context and describe are interchangeable, but in my humble opinion, they are different. I use describe to frame a test, and context to position it. I use describe to describe and object, and context to put that object in a situation.

While it can be passed a string to explain what it should be doing, including shoulda will combine the two together. That is truly awesome.

What About Rails 3??

Actually, Rspec 2 and Shoulda are awesome on their own, it just turns out that I'm using them extensively on a bunch of rails 3.1 apps and engines, so that's what got me researching them. I am convinced that Rails 3.1 is totally awesome, too, and the fact that these projects all work together is a thing of beauty, in my humble opinion.

¥

Yearly Indexes: 2003 2004 2006 2007 2008 2009 2010 2011 2012 2013 2015 2019 2020 2022