You want your specs to use transactions for speed 🐎🐎🐎
But as soon as you try it with Capybara, things go wrong 💻💥
Don't flip tables. Use this instead.
For a detailed explanation of how this works, refer to the introductory blog post.
Right now this gem automatically handles the following things:
- ActiveRecord
- Sequel
- jQuery
- Angular
Tested on Capybara 2.4.x, may not work on other major versions.
Don't see something you want? I'd love a pull request, or even just a friendly inquiry!
Add it to your Gemfile, of course:
group :test do
gem 'transactional_capybara'
end
And then initialize it in your tests…
In rails_helper.rb
(or spec_helper.rb
):
require "transactional_capybara/rspec"
If you're using ActiveRecord::Base.maintain_test_schema!
in your rails_helper.rb
(or spec_helper.rb
), make sure it is invoked before requiring transactional_capybara. Failure to do so might cause some non-deterministic connection failures.
Your database connection is automatically shared between threads, and all specs tagged with js: true
will wait for AJAX requests to finish before continuing.
Wow. Much convenience. So relax.
Somewhere near the beginning of your test initialization:
TransactionalCapybara.share_connection
And then make sure to define a hook that will run after each Capybara test:
after :each do
TransactionalCapybara::AjaxHelpers.wait_for_ajax(page)
end
You might have situations where you need to wait for AJAX calls to complete at times other than teardown. For example, you might have a pattern like this if you access models directly for either setup or verification of results:
visit '/page-that-fires-ajax'
Model.where(whatever).first
This can still fail! The ideal solution is to avoid direct database manipulation in integration tests. However, if you insist on doing this, you can stay safe by waiting for AJAX to complete before continuing:
visit '/page-that-fires-ajax'
TransactionalCapybara::AjaxHelpers.wait_for_ajax(page)
Model.where(whatever).first
If you'd like the helper more easily accessible, just mix the AjaxHelpers
module into your test suite.
In RSpec, you can do this in the config block:
RSpec.configure do |config|
config.include TransactionalCapybara::AjaxHelpers
end
Or in any example group:
describe 'awesome web stuff' do
include TransactionalCapybara::AjaxHelpers
end
Now the helper is easily available:
visit '/page-that-fires-ajax'
wait_for_ajax
Model.where(whatever).first
For this gem to be able to help with AJAX, it needs to be invoked before DatabaseCleaner rolls back the transaction.
You should be good to go if your setup looks like this:
config.around(:each) do |example|
DatabaseCleaner.cleaning do
example.run
end
end
But if you're using before/after hooks to clean, like:
before :each do
DatabaseCleaner.start
end
after :each do
DatabaseCleaner.clean
end
Then you need to make sure the AJAX hook runs first. Declare it before the DatabaseCleaner cleanup and you should be set:
before :each do
DatabaseCleaner.start
end
after :each do
TransactionalCapybara::AjaxHelpers.wait_for_ajax(page)
DatabaseCleaner.clean
end
If you want to have shared database connections with sequel just
add the option single_threaded: true
to your sequel connection in test.
- Fork it
- Create your feature branch (
git checkout -b my-new-feature
) - Make your changes. Don't forget to add a test!
- Commit your changes (
git commit -am 'Add some feature'
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request
cp spec/support/config.yml{.example,} # Edit as necessary
DB=sqlite rspec spec
You can run the specs with other databases as well, but you will have to create the databases and users manually first.
You can run the tests against all combinations of supported databases, web drivers, and ORMs with:
rake test:all
That rake task may be examined to find all the options currently under test, as well.