Я пытаюсь использовать 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 кажется, что строка для тестового пользователя не откатывается из базы данных между тестами. Я думал, что эти тесты должны были быть транзакционными, но, похоже, они мне не подходят.
Это объясняет, почему тест выполняется правильно в первый раз (и если я очищаю базу данных), но не проходит во второй раз.
Может кто-нибудь объяснить, что я должен изменить, чтобы убедиться, что тесты выполняются транзакционно?
Наконец-то это исправлено, и я надеюсь, что смогу сэкономить кому-нибудь шесть часов отладки, которые потребовались мне, чтобы понять это.
А) мне повезло и я получил работающую версию кода и б) убрав оба набора кода, вот что я обнаружил:
Тест, который мешает
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 за его (тоже правильное) предложение]
Внутри test/test_helper.rb
убедитесь, что у вас есть следующее.
class ActiveSupport::TestCase
self.use_transactional_fixtures = true
#...
end
Несмотря на название "fixtures", это работает и с factory_girl
.