views:

3679

answers:

11

Whenever I run rspec tests for my Rails application it takes forever and a day of overhead before it actually starts running tests. Why is rspec so slow? Is there a way to speed up Rails' initial load or single out the part of my Rails app I need (e.g. ActiveRecord stuff only) so it doesn't load absolutely everything to run a few tests?

+2  A: 

Are you running this over Rails? If so, it's not RSpec's initialization that's slow, it's Rails'. Rails has to initialize the entire codebase and yours before running the specs. Well, it doesn't have to, but it does. RSpec runs pretty fast for me under my small non-rails projects.

Loren Segal
Thanks for this answer. I've clarified the question to be Rails-specific.
Adrian Dunston
Then yea, its rails, not rspec that's slowing you down for sure.
Loren Segal
+15  A: 

You should be able to to speed up your script/spec calls by running script/spec_server in a separate terminal window, then adding the additional -X parameter to your spec calls.

Scott Matthewman
To be clear, you would then do something like this: spec --color ./spec -X
sbwoodside
spec_server is now deprecated in favor of spork. (http://github.com/timcharper/spork)
austinfromboston
+3  A: 

Running tests can be really slow because the whole rails environment has to load (try script/console) and only then can all tests run. You should use autotest which keeps the environment loaded and will check which files you edit. When you edit and save a file, only the tests that depend on these will run automatically and quickly.

Marc-André Lafortune
+2  A: 

If you're using a Mac I recommend using Rspactor over autotest as it uses a lot fewer resources for polling changed files than autotest. There is both a full Cocoa version

RSpactor.app

or the gem version that I maintain at Github

sudo gem install pelle-rspactor

While these don't speed up individual rspec tests, they feel much faster as they auto run the affected spec's within a second of you hitting save.

Pelle
cool idea, but link needs update and it doesn't seem to work with latest rspec.
sbwoodside
I recommend using mislavs version now.sudo gem install mislav-rspactor
Pelle
+1  A: 

If you are on a Windows environment then there is probably little you can do as Rails seems to startup really slowly under Windows. I had the same experience on Windows and had to move my setup to a Linux VM to make it really zippy (I was also using autotest).

DEfusion
+1. I run Rails and RSpec on Linux and have no slowness complaints, albeit on a fairly powerful machine. My colleague uses Windows, on identical hardware, and it can take a full minute to load the Rails environment.
Sam Stokes
+4  A: 

Why is rspec so slow? because it loads all the environement, loads fixtures and all that jazz.

Is there a way to speed up Rails' initial load you could try using mocks instead of relying on the database, this is actually correct for unit testing and will definitly speed up your unit tests. Additionnaly using the spec server as mentionned by @Scott Matthewman can help, same with the autotest from zentest mentionned by @Marc-Andre Lafortune

Is there a way to single out the part of my Rails app I need (e.g. ActiveRecord stuff only) so it doesn't load absolutely everything to run a few tests? what about this

rake test:recent

I am not sure how the rspec task integrate with this but you could definitely use the test:recent task as a template to do the same with rspec tests if the.

rake test:rspec:recent

doesn't exist yet

Jean
I second ZenTest....> spork/spec_server
scottschulthess
There is a new autotest tool for ruby : Infinity test @ http://github.com/tomas-stefano/infinity_test
Jean
+3  A: 

because it loads all the environement, loads fixtures and all that jazz.

The real culprit is if you run it using rake spec, it runs the db:test:prepare task.

This task drops your entire test database and re-creates it from scratch. This seems ridiculous to me, but that's what it does (the same thing happens when you run rake:test:units etc).

You can easily work around this using the spec application which rspec installs as part of the rspec gem.

Like this:

cd railsapp
spec spec # run all specs without rebuilding the whole damn database
spec spec/models # run model specs only

cd spec
spec controllers/user* # run specs for controllers that start with user
Orion Edwards
Dropping the DB and recreating it ensures that the data from the fixtures is loaded correctly and that no modifications are left from previous runs. running spec directly doesn't guarantee that. if you are going to be dependant on the db you must recreate it from scratch, mock objects ftw !
Jean
+4  A: 

I think the "zen" experience you're looking for is to run spec_server and autospec in the background, with the result being near-instant tests when you save a file.

However, I'm having problems getting these two programs to communicate.

I found an explanation here:

I've noticed that autotest doesn't send commands to the spec_server. Instead it reloads the entire Rails environment and your application's plugins everytime it executes. This causes autotest to run significantly slower than script server, because when you run the script/spec command the specs are sent to the spec_server which already has your Rails environment fired up and ready to go. If you happen to install a new plugin or something like that, then you'll have to restart the spec_server.

But, how do we fix this issue? I'm guessing it would involve downloading ZenTest and changing code for the autotest program, but don't have time to try it out right now.

Nappy
+1  A: 

As of rspec-rails-1.2.7, spec_server is deprecated in favor of the spork gem.

scottschulthess
+1  A: 

The main reason is that require takes forever on windows, for some reason.

Tips for speedup: spork now works with windows, I believe.

You can try "faster_require" which caches locations:

http://github.com/rdp/faster_require

GL. -rp

rogerdpack
A: 

I definitely suggest checking out spork.

http://spork.rubyforge.org/

The railstutorial specifically addresses this, and gives a workaround to get spork running nicely in rails 3.0 (as of this moment, spork is not rails 3 ready out of the box). Of course, if you're not on rails 3.0, then you should be good to go.

The part of the tutorial showing how to get spork running in rails 3.0

http:// railstutorial .org/chapters/static-pages#sec:spork

Checking when spork is rails 3.0 ready

http:// www.railsplugins .org/plugins/440-spork

(sorry for the 2 broken hyperlinks, I'm too junior on this site to post more than 1, I guess)