Rspec - 每次测试后如何清理数据库

时间:2015-04-06 07:03:31

标签: ruby-on-rails rspec capybara factory-bot database-cleaner

我有一个Capybara的功​​能规格用于登录页面,我正在使用FactoryGirl + DatabaseCleaner

require 'rails_helper'

feature 'Admin signs in' do

  background do
    FactoryGirl.create(:user)
  end

  scenario 'with valid credentials' do
    visit admin_root_path
    fill_in 'user_email', :with => 'email@email.com'
    fill_in 'user_password', :with => 'testpassword'
    click_button 'Sign in'
    expect(page).to have_content('Dashboard')
  end

  scenario 'with invalid credentials' do
    visit admin_root_path
    fill_in 'user_email', :with => 'email@email.com'
    fill_in 'user_password', :with => 'wrongpassword'
    click_button 'Sign in'
    expect(page).to have_content('Admin Login')
  end

end

运行测试,我收到以下错误:

1) Admin signs in test with invalid credentials
 Failure/Error: FactoryGirl.create(:user)
 ActiveRecord::RecordInvalid:
   Validation failed: Email has already been taken

我认为DatabaseCleaner会恢复更改,但看起来用户记录会持续存在于数据库中,直到第二个场景块。

如何确保在第一个场景之后清理数据库?

我按照此post

配置了数据库清理程序
# support/database_cleaner_spec.rb

RSpec.configure do |config|
  config.before(:suite) do
    DatabaseCleaner.clean_with(:truncation)
  end

  config.before(:each) do
    DatabaseCleaner.strategy = :transaction
  end

  config.before(:each, :js => true) do
    DatabaseCleaner.strategy = :truncation
  end

  config.before(:each) do
    DatabaseCleaner.start
  end

  config.after(:each) do
    DatabaseCleaner.clean
  end

end

我还用以下内容更新了规范帮助文件:

config.use_transactional_fixtures = false

5 个答案:

答案 0 :(得分:6)

我错误地认为 spec / support 文件夹中的配置文件已自动加载,但事实证明我必须取消注释 spec / rails_helper.rb中的以下行

Dir[Rails.root.join('spec/support/**/*.rb')].each { |f| require f }

DatabaseCleaner配置文件是正确的,它根本就没有加载。

答案 1 :(得分:2)

确保您在spec / rails_helper.rb

中具有以下配置
RSpec.configure do |config|
 config.use_transactional_fixtures = true
end

我们的想法是使用干净的数据库启动每个示例,创建该示例所需的任何数据,然后通过简单地回滚示例末尾的事务来删除该数据。

答案 2 :(得分:0)

使用以下设置:

config.before(:suite) do
  DatabaseCleaner.clean_with(:truncation)
end

答案 3 :(得分:0)

如果您在测试中的某个方法中使用MySQL并更改表(重置自动增量或任何DDL操作),则不会清除事务策略will fail和您的数据库。

为了修复,你必须声明一个这样的配置块:

config.before(:each, :altering_database => true) do
  DatabaseCleaner.strategy = :truncation
end

并将此配置添加到您的测试环境中:

context "when you alter the DB", :altering_database => true do...

注意:这会降低您的测试速度,因此请注意不要滥用它。

答案 4 :(得分:0)

  config.before(:example) do
    DatabaseCleaner.clean_with(:truncation)
  end

为我工作!