mirror of
https://github.com/thoughtbot/factory_bot.git
synced 2022-11-09 11:43:51 -05:00
b86ec136b6
to make use of did_you_mean gem. The did_you_mean gem only supports NameError, NoMethodError and KeyError. However, for NameError the message does also need to match a certain format and we can not use a custom message like 'Factory not registered ...'. Therefore using KeyError is to only logical conclusion. The did_you_mean gem makes use of the receiver attributes, but in Ruby > 2.5 it is not possible to set the receiver and key attributes on a KeyError (they are only set when the KeyError is raised in C). We explored monkey patching KeyError for earlier versions of Ruby, but it was a problematic solution. Instead we can rescue the original KeyError, take the message from it, which will already include the did_you_mean message, then customize the message and re-raise a new KeyError with that customized message. Starting in Ruby 2.6 this will not be necessary anymore https://bugs.ruby-lang.org/issues/14313, so maybe we can get rid of it for FactoryBot 6 or 7. Fixes #992 Co-authored-by: Daniel Colson <danieljamescolson@gmail.com>
184 lines
4.4 KiB
Ruby
184 lines
4.4 KiB
Ruby
describe "modifying factories" do
|
|
include FactoryBot::Syntax::Methods
|
|
|
|
before do
|
|
define_model("User", name: :string, admin: :boolean, email: :string, login: :string)
|
|
|
|
FactoryBot.define do
|
|
sequence(:email) { |n| "user#{n}@example.com" }
|
|
|
|
factory :user do
|
|
email
|
|
|
|
after(:create) do |user|
|
|
user.login = user.name.upcase if user.name
|
|
end
|
|
|
|
factory :admin do
|
|
admin { true }
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
context "simple modification" do
|
|
before do
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
name { "Great User" }
|
|
end
|
|
end
|
|
end
|
|
|
|
subject { create(:user) }
|
|
its(:name) { should eq "Great User" }
|
|
its(:login) { should eq "GREAT USER" }
|
|
|
|
it "doesn't allow the factory to be subsequently defined" do
|
|
expect do
|
|
FactoryBot.define { factory :user }
|
|
end.to raise_error(FactoryBot::DuplicateDefinitionError, "Factory already registered: user")
|
|
end
|
|
|
|
it "does allow the factory to be subsequently modified" do
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
name { "Overridden again!" }
|
|
end
|
|
end
|
|
|
|
expect(create(:user).name).to eq "Overridden again!"
|
|
end
|
|
end
|
|
|
|
context "adding callbacks" do
|
|
before do
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
name { "Great User" }
|
|
after(:create) do |user|
|
|
user.name = user.name.downcase
|
|
user.login = nil
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
subject { create(:user) }
|
|
|
|
its(:name) { should eq "great user" }
|
|
its(:login) { should be_nil }
|
|
end
|
|
|
|
context "reusing traits" do
|
|
before do
|
|
FactoryBot.define do
|
|
trait :rockstar do
|
|
name { "Johnny Rockstar!!!" }
|
|
end
|
|
end
|
|
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
rockstar
|
|
email { "#{name}@example.com" }
|
|
end
|
|
end
|
|
end
|
|
|
|
subject { create(:user) }
|
|
|
|
its(:name) { should eq "Johnny Rockstar!!!" }
|
|
its(:email) { should eq "Johnny Rockstar!!!@example.com" }
|
|
its(:login) { should eq "JOHNNY ROCKSTAR!!!" }
|
|
end
|
|
|
|
context "redefining attributes" do
|
|
before do
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
email { "#{name}-modified@example.com" }
|
|
name { "Great User" }
|
|
end
|
|
end
|
|
end
|
|
|
|
context "creating user" do
|
|
context "without overrides" do
|
|
subject { create(:user) }
|
|
|
|
its(:name) { should eq "Great User" }
|
|
its(:email) { should eq "Great User-modified@example.com" }
|
|
end
|
|
|
|
context "overriding the email" do
|
|
subject { create(:user, email: "perfect@example.com") }
|
|
|
|
its(:name) { should eq "Great User" }
|
|
its(:email) { should eq "perfect@example.com" }
|
|
end
|
|
|
|
context "overriding the name" do
|
|
subject { create(:user, name: "wonderful") }
|
|
|
|
its(:name) { should eq "wonderful" }
|
|
its(:email) { should eq "wonderful-modified@example.com" }
|
|
end
|
|
end
|
|
|
|
context "creating admin" do
|
|
context "without overrides" do
|
|
subject { create(:admin) }
|
|
|
|
its(:name) { should eq "Great User" }
|
|
its(:email) { should eq "Great User-modified@example.com" }
|
|
its(:admin) { should be true }
|
|
end
|
|
|
|
context "overriding the email" do
|
|
subject { create(:admin, email: "perfect@example.com") }
|
|
|
|
its(:name) { should eq "Great User" }
|
|
its(:email) { should eq "perfect@example.com" }
|
|
its(:admin) { should be true }
|
|
end
|
|
|
|
context "overriding the name" do
|
|
subject { create(:admin, name: "wonderful") }
|
|
|
|
its(:name) { should eq "wonderful" }
|
|
its(:email) { should eq "wonderful-modified@example.com" }
|
|
its(:admin) { should be true }
|
|
end
|
|
end
|
|
end
|
|
|
|
it "doesn't overwrite already defined child's attributes" do
|
|
FactoryBot.modify do
|
|
factory :user do
|
|
admin { false }
|
|
end
|
|
end
|
|
expect(create(:admin)).to be_admin
|
|
end
|
|
|
|
it "allows for overriding child classes" do
|
|
FactoryBot.modify do
|
|
factory :admin do
|
|
admin { false }
|
|
end
|
|
end
|
|
|
|
expect(create(:admin)).not_to be_admin
|
|
end
|
|
|
|
it "raises an exception if the factory was not defined before" do
|
|
modify_unknown_factory = -> do
|
|
FactoryBot.modify do
|
|
factory :unknown_factory
|
|
end
|
|
end
|
|
|
|
expect(modify_unknown_factory).to raise_error(KeyError)
|
|
end
|
|
end
|