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

178 lines
5.1 KiB
Ruby
Raw Normal View History

module FactoryGirl
class << self
attr_accessor :factories #:nodoc:
end
self.factories = {}
def self.factory_by_name(name)
factories[name.to_sym] or raise ArgumentError.new("No such factory: #{name.to_s}")
end
2010-09-07 19:20:02 -04:00
def self.register_factory(factory, options = {})
name = options[:as] || factory.name
if self.factories[name]
raise DuplicateDefinitionError, "Factory already defined: #{name}"
end
self.factories[name] = factory
end
2010-06-07 15:51:18 -04:00
# 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:
attr_reader :attributes #:nodoc:
2010-06-07 15:51:18 -04:00
def factory_name
puts "WARNING: factory.factory_name is deprecated. Use factory.name instead."
name
end
def class_name #:nodoc:
@options[:class] || name
2009-01-07 11:48:29 -05:00
end
def build_class #:nodoc:
@build_class ||= class_for(class_name)
end
2010-06-07 15:51:18 -04:00
def default_strategy #:nodoc:
@options[:default_strategy] || :create
2009-10-09 23:46:19 -05:00
end
2010-06-07 15:51:18 -04:00
def initialize(name, options = {}) #:nodoc:
assert_valid_options(options)
@name = factory_name_for(name)
@options = options
@attributes = []
end
def inherit_from(parent) #:nodoc:
@options[:class] ||= parent.class_name
@options[:default_strategy] ||= parent.default_strategy
new_attributes = []
parent.attributes.each do |attribute|
unless attribute_defined?(attribute.name)
new_attributes << attribute.clone
end
end
@attributes.unshift *new_attributes
end
def define_attribute(attribute)
name = attribute.name
# TODO: move these checks into Attribute
if attribute_defined?(name)
raise AttributeDefinitionError, "Attribute already defined: #{name}"
end
if attribute.respond_to?(:factory) && attribute.factory == self.name
raise AssociationDefinitionError, "Self-referencing association '#{name}' in factory '#{self.name}'"
end
@attributes << attribute
end
def add_callback(name, &block)
unless [:after_build, :after_create, :after_stub].include?(name.to_sym)
raise InvalidCallbackNameError, "#{name} is not a valid callback name. Valid callback names are :after_build, :after_create, and :after_stub"
end
@attributes << Attribute::Callback.new(name.to_sym, block)
end
def run (proxy_class, overrides) #:nodoc:
proxy = proxy_class.new(build_class)
overrides = symbolize_keys(overrides)
overrides.each {|attr, val| proxy.set(attr, val) }
passed_keys = overrides.keys.collect {|k| FactoryGirl.aliases_for(k) }.flatten
@attributes.each do |attribute|
unless passed_keys.include?(attribute.name)
attribute.add_to(proxy)
end
end
proxy.result
end
2009-09-15 15:47:47 -04:00
def human_name(*args, &block)
name.to_s.gsub('_', ' ')
2009-09-15 15:47:47 -04:00
end
def associations
attributes.select {|attribute| attribute.is_a?(Attribute::Association) }
end
private
2009-09-15 15:47:47 -04:00
def class_for (class_or_to_s)
if class_or_to_s.respond_to?(:to_sym)
class_name = variable_name_to_class_name(class_or_to_s)
class_name.split('::').inject(Object) do |object, string|
object.const_get(string)
end
else
class_or_to_s
end
end
def factory_name_for(class_or_to_s)
if class_or_to_s.respond_to?(:to_sym)
class_or_to_s.to_sym
else
class_name_to_variable_name(class_or_to_s).to_sym
end
end
def attribute_defined? (name)
!@attributes.detect {|attr| attr.name == name && !attr.is_a?(Attribute::Callback) }.nil?
end
def assert_valid_options(options)
invalid_keys = options.keys - [:class, :parent, :default_strategy]
unless invalid_keys == []
raise ArgumentError, "Unknown arguments: #{invalid_keys.inspect}"
end
assert_valid_strategy(options[:default_strategy]) if options[:default_strategy]
end
2010-06-07 15:51:18 -04:00
def assert_valid_strategy(strategy)
unless Proxy.const_defined? variable_name_to_class_name(strategy)
raise ArgumentError, "Unknown strategy: #{strategy}"
end
2009-01-08 11:43:07 -05:00
end
# Based on ActiveSupport's underscore inflector
def class_name_to_variable_name(name)
name.to_s.gsub(/::/, '/').
gsub(/([A-Z]+)([A-Z][a-z])/,'\1_\2').
gsub(/([a-z\d])([A-Z])/,'\1_\2').
tr("-", "_").
downcase
end
# Based on ActiveSupport's camelize inflector
def variable_name_to_class_name(name)
name.to_s.
gsub(/\/(.?)/) { "::#{$1.upcase}" }.
gsub(/(?:^|_)(.)/) { $1.upcase }
end
# From ActiveSupport
def symbolize_keys(hash)
hash.inject({}) do |options, (key, value)|
options[(key.to_sym rescue key) || key] = value
options
end
end
end
end