1
0
Fork 0
mirror of https://github.com/thoughtbot/factory_bot.git synced 2022-11-09 11:43:51 -05:00
thoughtbot--factory_bot/lib/factory_girl/factory.rb

226 lines
5.7 KiB
Ruby
Raw Normal View History

require "active_support/core_ext/hash/keys"
require "active_support/inflector"
module FactoryGirl
# Raised when a factory is defined that attempts to instantiate itself.
class AssociationDefinitionError < RuntimeError
2009-01-07 11:48:29 -05:00
end
# Raised when a callback is defined that has an invalid name
class InvalidCallbackNameError < RuntimeError
end
2010-06-07 15:51:18 -04:00
# Raised when a factory is defined with the same name as a previously-defined factory.
class DuplicateDefinitionError < RuntimeError
2009-01-08 11:43:07 -05:00
end
class Factory
attr_reader :name #:nodoc:
2011-08-12 10:35:41 -04:00
def factory_name
puts "WARNING: factory.factory_name is deprecated. Use factory.name instead."
name
end
def build_class #:nodoc:
@build_class ||= class_name.to_s.camelize.constantize
end
2010-06-07 15:51:18 -04:00
def default_strategy #:nodoc:
@default_strategy || (parent && parent.default_strategy) || :create
2009-10-10 00:46:19 -04:00
end
2010-06-07 15:51:18 -04:00
def initialize(name, options = {}) #:nodoc:
assert_valid_options(options)
2011-09-23 16:33:39 -04:00
@name = name.to_s.underscore.to_sym
@parent = options[:parent]
@aliases = options[:aliases] || []
@traits = options[:traits] || []
2011-09-23 16:33:39 -04:00
@class_name = options[:class]
@default_strategy = options[:default_strategy]
@defined_traits = []
@children = []
@attribute_list = AttributeList.new
@compiled = false
end
def allow_overrides
@compiled = false
@attribute_list.overridable
self
end
def allow_overrides?
@attribute_list.overridable?
end
def add_child(factory)
@children << factory unless @children.include?(factory)
2011-08-09 20:29:02 -04:00
end
2011-08-12 10:35:41 -04:00
2011-08-12 16:16:17 -04:00
def define_trait(trait)
@defined_traits << trait
2011-08-09 20:29:02 -04:00
end
def add_callback(name, &block)
2011-09-16 16:24:16 -04:00
@attribute_list.add_callback(Callback.new(name, block))
2011-08-12 14:38:33 -04:00
end
def attributes
ensure_compiled
AttributeList.new.tap do |list|
@traits.reverse.map { |name| trait_by_name(name) }.each do |trait|
list.apply_attributes(trait.attributes)
end
list.apply_attributes(@attribute_list)
list.apply_attributes(parent.attributes) if parent
2011-09-16 16:06:32 -04:00
end
end
def run(proxy_class, overrides) #:nodoc:
ensure_compiled
proxy = proxy_class.new(build_class)
2011-09-16 16:24:16 -04:00
callbacks.each { |callback| proxy.add_callback(callback) }
overrides = overrides.symbolize_keys
attributes.each do |attribute|
factory_overrides = overrides.select { |attr, val| attribute.aliases_for?(attr) }
if factory_overrides.empty?
attribute.add_to(proxy)
else
factory_overrides.each { |attr, val| proxy.set(attr, val, attribute.ignored); overrides.delete(attr) }
end
end
overrides.each { |attr, val| proxy.set(attr, val) }
proxy.result(@to_create_block)
end
2009-09-15 15:47:47 -04:00
def human_names
names.map {|name| name.to_s.humanize.downcase }
2009-09-15 15:47:47 -04:00
end
def associations
attributes.select {|attribute| attribute.association? }
end
2011-08-12 16:16:17 -04:00
def trait_by_name(name)
if existing_attribute = trait_for(name)
2011-08-12 10:46:15 -04:00
existing_attribute
elsif parent
parent.trait_by_name(name)
2011-08-12 10:35:41 -04:00
else
2011-08-12 16:16:17 -04:00
FactoryGirl.trait_by_name(name)
2011-08-09 20:29:02 -04:00
end
end
2011-08-12 10:35:41 -04:00
# Names for this factory, including aliases.
#
# Example:
#
# factory :user, :aliases => [:author] do
# # ...
# end
#
# FactoryGirl.create(:author).class
# # => User
#
# Because an attribute defined without a value or block will build an
# association with the same name, this allows associations to be defined
# without factories, such as:
#
# factory :user, :aliases => [:author] do
# # ...
# end
#
# factory :post do
# author
# end
#
# FactoryGirl.create(:post).author.class
# # => User
def names
2011-09-23 16:33:39 -04:00
[name] + @aliases
end
def to_create(&block)
@to_create_block = block
end
2011-09-16 16:06:32 -04:00
def callbacks
attributes.callbacks
end
def ensure_compiled
compile unless @compiled
end
def declare_attribute(declaration)
@attribute_list.declare_attribute(declaration)
end
protected
def class_name #:nodoc:
@class_name || (parent && parent.class_name) || name
end
private
def compile
inherit_factory(parent) if parent
declarations.each do |declaration|
declaration.to_attributes.each do |attribute|
define_attribute(attribute)
end
end
@compiled = true
end
def inherit_factory(parent) #:nodoc:
parent.ensure_compiled
allow_overrides if parent.allow_overrides?
parent.add_child(self)
end
def declarations
@attribute_list.declarations
end
def define_attribute(attribute)
if attribute.respond_to?(:factory) && attribute.factory == self.name
raise AssociationDefinitionError, "Self-referencing association '#{attribute.name}' in factory '#{self.name}'"
end
@attribute_list.define_attribute(attribute)
end
def assert_valid_options(options)
options.assert_valid_keys(:class, :parent, :default_strategy, :aliases, :traits)
2011-01-19 19:26:27 -05:00
if options[:default_strategy]
assert_valid_strategy(options[:default_strategy])
puts "WARNING: default_strategy is deprecated."
puts "Override to_create if you need to prevent a call to #save!."
end
end
2010-06-07 15:51:18 -04:00
def assert_valid_strategy(strategy)
unless Proxy.const_defined? strategy.to_s.camelize
raise ArgumentError, "Unknown strategy: #{strategy}"
end
2009-01-08 11:43:07 -05:00
end
2011-08-12 16:16:17 -04:00
def trait_for(name)
@defined_traits.detect {|trait| trait.name == name }
end
def parent
return unless @parent
FactoryGirl.factory_by_name(@parent)
end
end
end