module Remarkable
module ActiveRecord
module Matchers
class ValidateAssociatedMatcher < Remarkable::ActiveRecord::Base
arguments :collection => :associations, :as => :association, :block => :block
optional :message, :builder
collection_assertions :find_association?, :is_valid?
default_options :message => :invalid
protected
def find_association?
reflection = @subject.class.reflect_on_association(@association)
raise ScriptError, "Could not find association #{@association} on #{subject_class}." unless reflection
associated_object = if builder = @options[:builder] || @block
builder.call(@subject)
elsif [:belongs_to, :has_one].include?(reflection.macro)
@subject.send(:"build_#{@association}") rescue nil
else
@subject.send(@association).build rescue nil
end
raise ScriptError, "The association object #{@association} could not be built. You can give me " <<
":builder as option or a block which returns an association." unless associated_object
raise ScriptError, "The associated object #{@association} is not invalid. You can give me " <<
":builder as option or a block which returns an invalid association." if associated_object.save
return true
end
def is_valid?
return false if @subject.valid?
error_message_to_expect = error_message_from_model(@subject, :base, @options[:message])
# In Rails 2.1.2, the error on association returns a symbol (:invalid)
# instead of the message, so we check this case here.
@subject.errors.on(@association) == @options[:message] ||
assert_contains(@subject.errors.on(@association), error_message_to_expect)
end
end
# Ensures that the model is invalid if one of the associations given is
# invalid. It tries to build the association automatically. In has_one
# and belongs_to cases, it will build it like this:
#
# @model.build_association
# @project.build_manager
#
# In has_many and has_and_belongs_to_many to cases it will build it like
# this:
#
# @model.association.build
# @project.tasks.build
#
# The object returned MUST be invalid and it's likely the case, since the
# associated object is empty when calling build. However, if the associated
# object has to be manipulated to be invalid, you will have to give :builder
# as option or a block to manipulate it:
#
# should_validate_associated(:tasks) do |project|
# project.tasks.build(:captcha => 'i_am_a_bot')
# end
#
# In the case above, the associated object task is only invalid when the
# captcha attribute is set. So we give a block to the matcher that tell
# exactly how to build an invalid object.
#
# The example above can also be written as:
#
# should_validate_associated :tasks, :builder => proc{ |p| p.tasks.build(:captcha => 'i_am_a_bot') }
#
# == Options
#
# * :builder - a proc to build the association
#
# * :message - value the test expects to find in errors.on(:attribute).
# Regexp, string or symbol. Default = I18n.translate('activerecord.errors.messages.invalid')
#
# == Examples
#
# should_validate_associated :tasks
# should_validate_associated(:tasks){ |p| p.tasks.build(:captcha => 'i_am_a_bot') }
# should_validate_associated :tasks, :builder => proc{ |p| p.tasks.build(:captcha => 'i_am_a_bot') }
#
# it { should validate_associated(:tasks) }
# it { should validate_associated(:tasks){ |p| p.tasks.build(:captcha => 'i_am_a_bot') } }
# it { should validate_associated(:tasks, :builder => proc{ |p| p.tasks.build(:captcha => 'i_am_a_bot') }) }
#
def validate_associated(*args, &block)
ValidateAssociatedMatcher.new(*args, &block).spec(self)
end
end
end
end