module Remarkable
module ActiveRecord
module Matchers
class AssociationMatcher < Remarkable::ActiveRecord::Base #:nodoc:
arguments :macro, :collection => :associations, :as => :association
optionals :through, :source, :source_type, :class_name, :foreign_key, :dependent, :join_table, :as
optionals :select, :conditions, :include, :group, :having, :order, :limit, :offset, :finder_sql, :counter_sql
optionals :uniq, :readonly, :validate, :autosave, :polymorphic, :counter_cache, :default => true
collection_assertions :association_exists?, :macro_matches?, :through_exists?, :source_exists?,
:join_table_exists?, :foreign_key_exists?, :polymorphic_exists?,
:counter_cache_exists?, :options_match?
protected
def association_exists?
reflection
end
def macro_matches?
reflection.macro == @macro
end
def through_exists?
return true unless @options.key?(:through)
reflection.through_reflection rescue false
end
def source_exists?
return true unless @options.key?(:through)
reflection.source_reflection rescue false
end
def join_table_exists?
return true unless reflection.macro == :has_and_belongs_to_many
::ActiveRecord::Base.connection.tables.include?(reflection.options[:join_table])
end
def foreign_key_exists?
return true unless foreign_key_table
table_has_column?(foreign_key_table, reflection_foreign_key)
end
def polymorphic_exists?
return true unless @options[:polymorphic]
table_has_column?(subject_class.table_name, reflection_foreign_key.sub(/_id$/, '_type'))
end
def counter_cache_exists?
return true unless @options[:counter_cache]
table_has_column?(reflection.klass.table_name, reflection.counter_cache_column.to_s)
end
def options_match?
actual_options = {}
@options.keys.each do |key|
method = :"reflection_#{key}"
@options[key] = @options[key].to_s
actual_options[key] = (respond_to?(method, true) ? send(method) : reflection.options[key]).to_s
end
return @options == actual_options, :actual => actual_options.inspect
end
private
def reflection
@reflection ||= subject_class.reflect_on_association(@association.to_sym)
end
# Rescue nil to avoid raising errors in invalid through associations
def reflection_class_name
reflection.class_name.to_s rescue nil
end
def reflection_foreign_key
reflection.primary_key_name.to_s
end
def table_has_column?(table_name, column)
::ActiveRecord::Base.connection.columns(table_name, 'Remarkable column retrieval').any?{|c| c.name == column }
end
# In through we don't check the foreign_key, because it's spread
# accross the through and the source reflection which should be tested
# with their own macros.
#
# In cases a join table exists (has_and_belongs_to_many and through
# associations), we check the foreign key in the join table.
#
# On belongs to, the foreign_key is in the subject class table and in
# other cases it's on the reflection class table.
#
def foreign_key_table
if reflection.options.key?(:through)
nil
elsif reflection.macro == :has_and_belongs_to_many
reflection.options[:join_table]
elsif reflection.macro == :belongs_to
subject_class.table_name
else
reflection.klass.table_name
end
end
def interpolation_options
options = { :macro => Remarkable.t(@macro, :scope => matcher_i18n_scope, :default => @macro.to_s), :options => @options.inspect }
if @subject && reflection
options.merge!(
:actual_macro => Remarkable.t(reflection.macro, :scope => matcher_i18n_scope, :default => reflection.macro.to_s),
:subject_table => subject_class.table_name.inspect,
:reflection_table => reflection.klass.table_name.inspect,
:foreign_key_table => foreign_key_table.inspect,
:polymorphic_column => reflection_foreign_key.sub(/_id$/, '_type').inspect,
:counter_cache_column => reflection.counter_cache_column.to_s.inspect,
:join_table => reflection.options[:join_table].inspect,
:foreign_key => reflection_foreign_key.inspect
) rescue nil # rescue to allow specs to run properly
end
options
end
end
# Ensure that the belongs_to relationship exists. Will also test that the
# subject table has the association_id column.
#
# == Options
#
# * :class_name - the expected associted class name.
# * :foreign_key - the expected foreign key in the subject table.
# * :dependent - the expected dependent value for the association.
# * :readonly - checks wether readonly is true or false.
# * :validate - checks wether validate is true or false.
# * :autosave - checks wether autosave is true or false.
# * :counter_cache - the expected dependent value for the association.
# It also checks if the column actually exists in the table.
# * :polymorphic - if the association should be polymorphic or not.
# When true it also checks for the association_type column in the subject table.
#
# Plus all supported sql conditions options: :select, :conditions, :order,
# :limit, :offset, :include, :group, :having.
#
# == Examples
#
# should_belong_to :parent, :polymorphic => true
# it { should belong_to(:parent) }
#
def belong_to(*associations, &block)
AssociationMatcher.new(:belongs_to, *associations, &block).spec(self)
end
# Ensures that the has_and_belongs_to_many relationship exists, if the join
# table is in place and if the foreign_key column exists.
#
# == Options
#
# * :class_name - the expected associted class name.
# * :join_table - the expected join table name.
# * :foreign_key - the expected foreign key in the association table.
# * :uniq - checks wether uniq is true or false.
# * :readonly - checks wether readonly is true or false.
# * :validate - checks wether validate is true or false.
# * :autosave - checks wether autosave is true or false.
#
# Plus all supported sql conditions options: :select, :conditions, :order,
# :limit, :offset, :include, :group, :having.
#
# == Examples
#
# should_have_and_belong_to_many :posts, :cars
# it{ should have_and_belong_to_many :posts, :cars }
#
def have_and_belong_to_many(*associations, &block)
AssociationMatcher.new(:has_and_belongs_to_many, *associations, &block).spec(self)
end
# Ensures that the has_many relationship exists. Will also test that the
# associated table has the required columns. It works by default with
# polymorphic association (:as does not have to be supplied).
#
# == Options
#
# * :class_name - the expected associted class name.
# * :through - the expected join model which to perform the query.
# It also checks if the through table exists.
# * :source - the source of the through association.
# * :source_type - the source type of the through association.
# * :foreign_key - the expected foreign key in the associated table.
# When used with :through, it will check for the foreign key in the join table.
# * :dependent - the expected dependent value for the association.
# * :uniq - checks wether uniq is true or false.
# * :readonly - checks wether readonly is true or false.
# * :validate - checks wether validate is true or false.
# * :autosave - checks wether autosave is true or false.
#
# Plus all supported sql conditions options: :select, :conditions, :order,
# :limit, :offset, :include, :group, :having.
#
# == Examples
#
# should_have_many :friends
# should_have_many :enemies, :through => :friends
# should_have_many :enemies, :dependent => :destroy
#
# it{ should have_many(:friends) }
# it{ should have_many(:enemies, :through => :friends) }
# it{ should have_many(:enemies, :dependent => :destroy) }
#
def have_many(*associations, &block)
AssociationMatcher.new(:has_many, *associations, &block).spec(self)
end
# Ensures that the has_many relationship exists. Will also test that the
# associated table has the required columns. It works by default with
# polymorphic association (:as does not have to be supplied).
#
# == Options
#
# * :class_name - the expected associted class name.
# * :through - the expected join model which to perform the query.
# It also checks if the through table exists.
# * :source - the source of the through association.
# * :source_type - the source type of the through association.
# * :foreign_key - the expected foreign key in the associated table.
# When used with :through, it will check for the foreign key in the join table.
# * :dependent - the expected dependent value for the association.
# * :validate - checks wether validate is true or false.
# * :autosave - checks wether autosave is true or false.
#
# Plus all supported sql conditions options: :select, :conditions, :order,
# :limit, :offset, :include, :group, :having.
#
# == Examples
#
# should_have_one :universe
# it{ should have_one(:universe) }
#
def have_one(*associations, &block)
AssociationMatcher.new(:has_one, *associations, &block).spec(self)
end
end
end
end