ruby-on-rails - DatabaseCleaner 似乎不会在诉讼之间清理

标签 ruby-on-rails ruby ruby-on-rails-3 rspec database-cleaner

救世主。

在每个 RSpec 示例之后,我在清理数据库时遇到了问题。 问题是,当我运行 rspec 命令时,users_controller_spec.rb 提示记录比示例预期的多。事实上,如果我使用 rails c 进行检查,就会按照它所说的那样创建记录。 当我单独运行这个套件时,它会成功,所以我认为这是因为 DatabaseCleaner 没有清除其他规范创建的用户记录(用户记录的数量与 users_controller_spec 示例声称的额外记录相匹配)。它们是在 before :all block 中创建的(如果重要的话)。

这是我的rails_helper.rb

# This file is copied to spec/ when you run 'rails generate rspec:install'
ENV['RAILS_ENV'] ||= 'test'
require 'spec_helper'
require File.expand_path('../../config/environment', __FILE__)
require 'rspec/rails'

# Add additional requires below this line. Rails is not loaded until this point!
require 'devise'
require 'admin/v1/dashboard_controller'
# Requires supporting ruby files with custom matchers and macros, etc, in
Dir[Rails.root.join('spec/support/**/*.rb')].each { |f| require f }

# Checks for pending migrations before tests are run.
# If you are not using ActiveRecord, you can remove this line.
ActiveRecord::Migration.maintain_test_schema!

RSpec.configure do |config|
  # Remove this line if you're not using ActiveRecord or ActiveRecord fixtures
  config.fixture_path = "#{::Rails.root}/spec/fixtures"

  config.include Devise::Test::ControllerHelpers, type: :controller
  config.include ControllerMacros, type: :controller
  # If you're not using ActiveRecord, or you'd prefer not to run each of your
  # examples within a transaction, remove the following line or assign false
  # instead of true.
  config.use_transactional_fixtures = true

  config.include FactoryGirl::Syntax::Methods

  config.infer_spec_type_from_file_location!

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

  config.around(:each) do |example|
    DatabaseCleaner.cleaning do
      example.run
    end
  end
end

users_controller.rb

describe 'GET #index' do
  it 'populates an array of users' do
    user1 = create(:user)
    user2 = create(:user)
    get :index
    expect(assigns(:users)).to match_array([user1, user2])
  end
  it 'renders :index template' do
    get :index, {}
    expect(response).to render_template :index
  end
end

更新 1:这是创建额外的 user 记录的地方

    require 'rails_helper'

describe Admin::V1::MessagesController do
  let(:admin_user) do
    admin_user = double('admin_user')
    allow(request.env['warden']).to receive(:authenticate!).and_return(admin_user)
    allow(controller).to receive(:current_admin_v1_admin_user).and_return(admin_user)
    p '==='
  end
  before { login_admin_user admin_user }

  describe 'GET #index' do
    it 'renders :index template' do
      get :index, {}
      expect(response).to render_template :index
    end
  end

  describe 'GET #get_users' do
    before :all do
      @user1 = create(:user, nickname: 'hiro')
      @user2 = create(:user, nickname: 'elise')
    end
    context 'with params' do
      it 'populates an array of users matching on nickname' do
        get :get_users, format: :json, query: 'h'
        expect(assigns(:users)).to match_array([@user1])
      end
    end
    context 'without params' do
      it 'populates an array of all users' do
        get :get_users, format: :json
        expect(assigns(:users)).to match_array([@user1, @user2])
      end
    end
  end

  describe 'GET #get_messages' do
    before :all do
      @user1 = create(:user)
      @user2 = create(:user)
      @message1 = create(:message, user_id: @user1.id)
      @message2 = create(:message, user_id: @user1.id)
      @message3 = create(:message, user_id: @user2.id)
    end
    context 'with user_id' do
      it 'populates an array of messages with the user_id' do
        get :get_messages, format: :json, user_id: @user1.id
        expect(assigns(:messages)).to match_array([@message1, @message2])
      end
    end
  end
end

最佳答案

不幸的是,RSpec 的 before(:all) 不能很好地处理事务测试。 before(:all) 中的代码在事务打开之前运行,这意味着在事务中止时不会回滚在那里创建的任何记录。您有责任在 after(:all) 中手动清理这些项目。

参见 rspec-rails#496Using before(:all) in RSpec will cause you lots of trouble unless you know what you are doing

  after(:all) do
    # before/after(:all) is not transactional; see https://www.relishapp.com/rspec/rspec-rails/docs/transactions
    DatabaseCleaner.clean_with(:truncation)
  end

关于ruby-on-rails - DatabaseCleaner 似乎不会在诉讼之间清理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39972969/

相关文章:

html - Rails 管理员自定义 : embed html view into dashboard

ruby-on-rails - Heroku 模式加载 : Permission denied for database "postgres" User

ruby-on-rails - Windows 7 上的 MySQL gem 安装错误

ruby:如果缺少则安装 gem 的代码

ruby-on-rails - 如何检查日期是否有效?

ruby-on-rails - 查找所有在 18 岁之前购买汽车的用户

ruby-on-rails - 如何在 Heroku 中查看详细日志,类似于我在开发环境中看到的日志?

sql - 事件记录查询。按多个表中的多列分组

javascript - 数组随机返回的 Jasmine 测试

ruby - application 和 actioncable 不会共享 cookie