Почему factory_girl не работает для меня транзакционно? - строки остаются в базе данных после тестов

Я пытаюсь использовать factory_girl для создания "пользовательской" фабрики (с RSpec), однако она не работает транзакционно и, по-видимому, дает сбой из-за остатков данных предыдущих тестов в тестовой базе данных.

Factory.define :user do |user|
  user.name                   "Joe Blow"
  user.email                  "joe@blow.com" 
  user.password               'password'
  user.password_confirmation  'password'
end

@user = Factory.create(:user)

Запуск первого набора тестов в порядке:

spec spec/ 


...
Finished in 2.758806 seconds

60 examples, 0 failures, 11 pending

Все хорошо и, как и ожидалось, однако запуск тестов снова:

spec spec/ 
...
/Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/validations.rb:1102:in `save_without_dirty!': Validation failed: Email has already been taken (ActiveRecord::RecordInvalid)
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/dirty.rb:87:in `save_without_transactions!'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/transactions.rb:200:in `save!'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/connection_adapters/abstract/database_statements.rb:136:in `transaction'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/transactions.rb:182:in `transaction'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/transactions.rb:200:in `save!'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/transactions.rb:208:in `rollback_active_record_state!'
    from /Library/Ruby/Gems/1.8/gems/activerecord-2.3.8/lib/active_record/transactions.rb:200:in `save!'
    from /Library/Ruby/Gems/1.8/gems/factory_girl-1.2.3/lib/factory_girl/proxy/create.rb:6:in `result'
    from /Library/Ruby/Gems/1.8/gems/factory_girl-1.2.3/lib/factory_girl/factory.rb:316:in `run'
    from /Library/Ruby/Gems/1.8/gems/factory_girl-1.2.3/lib/factory_girl/factory.rb:260:in `create'
    from /Users/petenixey/Rails_apps/resample/spec/controllers/users_controller_spec.rb:7
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/example/example_group_methods.rb:183:in `module_eval'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/example/example_group_methods.rb:183:in `subclass'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/example/example_group_methods.rb:55:in `describe'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/example/example_group_factory.rb:31:in `create_example_group'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/dsl/main.rb:28:in `describe'
    from /Users/petenixey/Rails_apps/resample/spec/controllers/users_controller_spec.rb:3
    from /Library/Ruby/Gems/1.8/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:147:in `load_without_new_constant_marking'
    from /Library/Ruby/Gems/1.8/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:147:in `load'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/runner/example_group_runner.rb:15:in `load_files'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/runner/example_group_runner.rb:14:in `each'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/runner/example_group_runner.rb:14:in `load_files'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/runner/options.rb:133:in `run_examples'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/lib/spec/runner/command_line.rb:9:in `run'
    from /Library/Ruby/Gems/1.8/gems/rspec-1.3.0/bin/spec:5
    from /usr/bin/spec:19:in `load'
    from /usr/bin/spec:19

Попытка исправить - использовать Factory. sequence

Так как у меня есть ограничение уникальности для моего поля электронной почты, я попытался решить проблему с помощью метода sequence factory_girl:

Factory.define :user do |user|
  user.name                   "Joe Blow"
  user.sequence(:email) {|n| "joe#{n}@blow.com" }
  user.password               'password'
  user.password_confirmation  'password'
end

Затем я запустил

rake db:test:prepare
spec spec/
.. # running the tests once executes fine
spec spec/
.. # running them the second time produces the same set of errors as before

. Пользователи, похоже, остаются в базе данных

Если я посмотрю в базе данных /db/test.sqlite3 кажется, что строка для тестового пользователя не откатывается из базы данных между тестами. Я думал, что эти тесты должны были быть транзакционными, но, похоже, они мне не подходят.

Это объясняет, почему тест выполняется правильно в первый раз (и если я очищаю базу данных), но не проходит во второй раз.

Может кто-нибудь объяснить, что я должен изменить, чтобы убедиться, что тесты выполняются транзакционно?

34
задан Arslan Ali 1 October 2015 в 13:19
поделиться

2 ответа

Наконец-то это исправлено, и я надеюсь, что смогу сэкономить кому-нибудь шесть часов отладки, которые потребовались мне, чтобы понять это.

А) мне повезло и я получил работающую версию кода и б) убрав оба набора кода, вот что я обнаружил:

Тест, который мешает

require 'spec_helper'

describe UsersController do

  @user = Factory.create(:user) 
end

Тест, который работает

require 'spec_helper'

describe UsersController do

  it "should make a factory models without choking" do
    @user = Factory.create(:user)   
  end
end

транзакция определяется оператором it "should do something" do ... . Если вы создадите экземпляр фабрики вне этого оператора, он окажется не транзакционным.

Вы также можете поместить его за пределами блока «он должен ..», если он находится в блоке «до… конца».

require 'spec_helper'

describe UsersController do

  before(:each) do
    @user = Factory.create(:user) 
  end

  it 'should make a factory without choking' do
    puts @user.name
    # prints out the correnct name for the user
  end
end

Поэкспериментировав, кажется допустимым определить пользователя вне блока «он». должен do..end "block, пока он находится в блоке" before ... end ". Я предполагаю, что это выполняется только в рамках блока "it should do..end" и поэтому работает нормально.

[Спасибо @jdl за его (тоже правильное) предложение]

52
ответ дан 27 November 2019 в 16:24
поделиться

Внутри test/test_helper.rb убедитесь, что у вас есть следующее.

class ActiveSupport::TestCase
  self.use_transactional_fixtures = true
  #...
end

Несмотря на название "fixtures", это работает и с factory_girl.

3
ответ дан 27 November 2019 в 16:24
поделиться
Другие вопросы по тегам:

Похожие вопросы: