Як створити фабрику в FactoryGirl з асоціацією has_many


89

Хтось може сказати мені, чи я просто йду з налаштуваннями неправильно?

У мене є такі моделі, які мають has_many.through асоціації:

class Listing < ActiveRecord::Base
  attr_accessible ... 

  has_many :listing_features
  has_many :features, :through => :listing_features

  validates_presence_of ...
  ...  
end


class Feature < ActiveRecord::Base
  attr_accessible ...

  validates_presence_of ...
  validates_uniqueness_of ...

  has_many :listing_features
  has_many :listings, :through => :listing_features
end


class ListingFeature < ActiveRecord::Base
  attr_accessible :feature_id, :listing_id

  belongs_to :feature  
  belongs_to :listing
end

Я використовую Rails 3.1.rc4, FactoryGirl 2.0.2, factory_girl_rails 1.1.0 та rspec. Ось моя основна перевірка осудності rspec rspec на :listingзаводі:

it "creates a valid listing from factory" do
  Factory(:listing).should be_valid
end

Ось завод (: перелік)

FactoryGirl.define do
  factory :listing do
    headline    'headline'
    home_desc   'this is the home description'
    association :user, :factory => :user
    association :layout, :factory => :layout
    association :features, :factory => :feature
  end
end

:listing_featureІ :featureзаводи так само установка.
Якщо association :featuresрядок коментується, тоді всі мої тести проходять.
Коли воно є

association :features, :factory => :feature

повідомлення про помилку - це те, undefined method 'each' for #<Feature> що, на мою думку, для мене має сенс, оскільки listing.featuresповертає масив. Тож я змінив його на

association :features, [:factory => :feature]

і помилка, яку я отримую зараз, полягає в тому, ArgumentError: Not registered: features чи просто не є розумним генерувати заводські об’єкти таким чином, або чого мені не вистачає? Щиро дякуємо за будь-які вклади!

Відповіді:


57

Створення подібних асоціацій вимагає використання зворотних викликів FactoryGirl.

Повний набір прикладів можна знайти тут.

https://thoughtbot.com/blog/aint-no-calla-back-girl

Щоб принести його додому на свій приклад.

Factory.define :listing_with_features, :parent => :listing do |listing|
  listing.after_create { |l| Factory(:feature, :listing => l)  }
  #or some for loop to generate X features
end

ти в кінцевому підсумку використовував асоціацію: особливості, [: factory =>: особливість]?
davidtingsu

108

Крім того, ви можете використовувати блок і пропустити associationключове слово. Це дозволяє створювати об'єкти без збереження в базі даних (інакше асоціація has_many збереже ваші записи в базі даних, навіть якщо ви використовуєте buildфункцію замість create).

FactoryGirl.define do
  factory :listing_with_features, :parent => :listing do |listing|
    features { build_list :feature, 3 }
  end
end

5
Це нявкання кота. Здатність до обох buildі createробить його найбільш універсальним візерунком. Потім використовуйте цю спеціальну стратегію побудови FG gist.github.com/Bartuz/74ee5834a36803d712b7 для post nested_attributes_forтестування дій контролераaccepts_nested_attributes_for
Кріс Бек

4
проголосував, набагато читабельніший та універсальніший за прийняту відповідь IMO
m_x

1
Починаючи з FactoryBot 5, associationключове слово використовує однакову стратегію побудови для батьків та дітей. Отже, він може будувати об'єкти без збереження в базі даних.
Нік

27

Ви можете використовувати trait:

FactoryGirl.define do
  factory :listing do
    ...

    trait :with_features do
      features { build_list :feature, 3 }
    end
  end
end

З callback, якщо вам потрібно створити БД:

...

trait :with_features do
  after(:create) do |listing|
    create_list(:feature, 3, listing: listing)
  end
end

Використовуйте у своїх специфікаціях, як це:

let(:listing) { create(:listing, :with_features) }

Це призведе до видалення дублікатів на ваших заводах і стане більш багаторазовим.

https://robots.thoughtbot.com/remove-duplication-with-factorygirls-traits


20

Я спробував кілька різних підходів, і це той, який для мене працював найнадійніше (адаптований до вашої справи)

FactoryGirl.define do
  factory :user do
    # some details
  end

  factory :layout do
    # some details
  end

  factory :feature do
    # some details
  end

  factory :listing do
    headline    'headline'
    home_desc   'this is the home description'
    association :user, factory: :user
    association :layout, factory: :layout
    after(:create) do |liztng|
      FactoryGirl.create_list(:feature, 1, listing: liztng)
    end
  end
end

0

Ось як я налаштував свій:

# Model 1 PreferenceSet
class PreferenceSet < ActiveRecord::Base
  belongs_to :user
  has_many :preferences, dependent: :destroy
end

#Model 2 Preference

class Preference < ActiveRecord::Base    
  belongs_to :preference_set
end



# factories/preference_set.rb

FactoryGirl.define do
  factory :preference_set do
    user factory: :user
    filter_name "market, filter_structure"

    factory :preference_set_with_preferences do
      after(:create) do |preference|
        create(:preference, preference_set: preference)
        create(:filter_structure_preference, preference_set: preference)
      end
    end
  end

end

# factories/preference.rb

FactoryGirl.define do
  factory :preference do |p|
    filter_name "market"
    filter_value "12"
  end

  factory :filter_structure_preference, parent: :preference do
    filter_name "structure"
    filter_value "7"
  end
end

А потім у своїх тестах ви можете зробити:

@preference_set = FactoryGirl.create(:preference_set_with_preferences)

Сподіваюся, що це допомагає.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.