Cómo configurar la fábrica en FactoryGirl con la asociación has_many

89

¿Alguien puede decirme si estoy haciendo la configuración de manera incorrecta?

Tengo los siguientes modelos que tienen asociaciones 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

Estoy usando Rails 3.1.rc4, FactoryGirl 2.0.2, factory_girl_rails 1.1.0 y rspec. Aquí está mi verificación básica de cordura de rspec rspec para la :listingfábrica:

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

Aquí está la fábrica (: listado)

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

Las fábricas :listing_featurey :featureestán configuradas de manera similar.
Si association :featuresse comenta la línea, todas mis pruebas pasan.
Cuando es

association :features, :factory => :feature

el mensaje de error es undefined method 'each' for #<Feature> lo que pensé que tenía sentido para mí porque listing.featuresdevuelve una matriz. Así que lo cambié a

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

y el error que obtengo ahora es ArgumentError: Not registered: features ¿No es sensato generar objetos de fábrica de esta manera, o qué me estoy perdiendo? ¡Muchas gracias por todos y cada uno de los aportes!

Tonys
fuente

Respuestas:

57

La creación de este tipo de asociaciones requiere el uso de devoluciones de llamada de FactoryGirl.

Puede encontrar un conjunto perfecto de ejemplos aquí.

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

Para llevarlo a casa con tu ejemplo.

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
Winfred
fuente
¿Terminaste usando la asociación: características, [: fábrica =>: característica]?
davidtingsu
108

Alternativamente, puede usar un bloque y omitir la associationpalabra clave. Esto hace posible construir objetos sin guardar en la base de datos (de lo contrario, una asociación has_many guardará sus registros en la base de datos, incluso si usa la buildfunción en lugar de create).

FactoryGirl.define do
  factory :listing_with_features, :parent => :listing do |listing|
    features { build_list :feature, 3 }
  end
end
JellicleCat
fuente
5
Este es el maullido del gato. La capacidad de ambos buildy lo createconvierte en el patrón más versátil. Luego use esta estrategia de compilación de FG personalizada gist.github.com/Bartuz/74ee5834a36803d712b7 para post nested_attributes_forprobar las acciones del controlador queaccepts_nested_attributes_for
Chris Beck
4
upvoted, mucho más legible y versátil que la respuesta aceptada IMO
m_x
1
A partir de FactoryBot 5, la associationpalabra clave usa la misma estrategia de compilación para padre e hijo. Entonces, puede construir objetos sin guardar en la base de datos.
Nick
27

Podrías usar trait:

FactoryGirl.define do
  factory :listing do
    ...

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

Con callback, si necesita la creación de una base de datos:

...

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

Úselo en sus especificaciones de esta manera:

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

Esto eliminará la duplicación en sus fábricas y será más reutilizable.

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

ehoffmann
fuente
20

Probé algunos enfoques diferentes y este es el que me funcionó de manera más confiable (adaptado a su caso)

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
Dave Sag
fuente
0

Así es como configuré el mío:

# 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

Y luego en tus pruebas puedes hacer:

@preference_set = FactoryGirl.create(:preference_set_with_preferences)

Espero que ayude.

rii
fuente