thoughtbot--shoulda-matchers/lib/shoulda/matchers/active_model/allow_value_matcher.rb

164 lines
4.5 KiB
Ruby
Raw Normal View History

2010-12-15 22:34:19 +00:00
module Shoulda # :nodoc:
module Matchers
module ActiveModel # :nodoc:
2010-12-15 22:34:19 +00:00
# Ensures that the attribute can be set to the given value or values. If
# multiple values are given the match succeeds only if all given values
# are allowed. Otherwise, the matcher fails at the first bad value in the
# argument list (the remaining arguments are not processed then).
2010-12-15 22:34:19 +00:00
#
# Options:
# * <tt>with_message</tt> - value the test expects to find in
# <tt>errors.on(:attribute)</tt>. Regexp or string. If omitted,
# the test looks for any errors in <tt>errors.on(:attribute)</tt>.
# * <tt>strict</tt> - expects the model to raise an exception when the
# validation fails rather than adding to the errors collection. Used for
# testing `validates!` and the `:strict => true` validation options.
2010-12-15 22:34:19 +00:00
#
# Example:
# it { should_not allow_value('bad').for(:isbn) }
2012-12-20 05:04:27 +00:00
# it { should allow_value('isbn 1 2345 6789 0').for(:isbn) }
2010-12-15 22:34:19 +00:00
#
def allow_value(*values)
2012-03-23 14:10:08 +00:00
if values.empty?
2012-12-20 05:04:27 +00:00
raise ArgumentError, 'need at least one argument'
2012-03-23 14:10:08 +00:00
else
AllowValueMatcher.new(*values)
end
2010-12-15 22:34:19 +00:00
end
class AllowValueMatcher # :nodoc:
include Helpers
def initialize(*values)
@values_to_match = values
@message_finder_factory = ValidationMessageFinder
2012-05-07 14:25:46 +00:00
@options = {}
2010-12-15 22:34:19 +00:00
end
def for(attribute)
@attribute = attribute
self
end
def with_message(message)
2012-05-07 14:25:46 +00:00
@options[:expected_message] = message
2010-12-15 22:34:19 +00:00
self
end
def strict
@message_finder_factory = ExceptionMessageFinder
self
end
2010-12-15 22:34:19 +00:00
def matches?(instance)
@instance = instance
2012-03-23 19:10:33 +00:00
@values_to_match.none? do |value|
@value = value
@instance.send("#{@attribute}=", @value)
2012-03-23 19:10:33 +00:00
errors_match?
end
2010-12-15 22:34:19 +00:00
end
def failure_message_for_should
2010-12-15 22:34:19 +00:00
"Did not expect #{expectation}, got error: #{@matched_error}"
end
def failure_message_for_should_not
2010-12-15 22:34:19 +00:00
"Expected #{expectation}, got #{error_description}"
end
def description
message_finder.allow_description(allowed_values)
2010-12-15 22:34:19 +00:00
end
private
def errors_match?
has_messages? && errors_for_attribute_match?
end
def has_messages?
message_finder.has_messages?
end
def errors_for_attribute_match?
if expected_message
@matched_error = errors_match_regexp? || errors_match_string?
else
errors_for_attribute.compact.any?
2010-09-05 15:23:09 +00:00
end
2010-12-15 22:34:19 +00:00
end
2012-03-23 14:10:08 +00:00
def errors_for_attribute
message_finder.messages
end
2010-12-15 22:34:19 +00:00
def errors_match_regexp?
2012-03-23 14:10:08 +00:00
if Regexp === expected_message
2012-05-07 14:25:46 +00:00
errors_for_attribute.detect { |e| e =~ expected_message }
2010-12-15 22:34:19 +00:00
end
end
def errors_match_string?
2012-03-23 14:10:08 +00:00
if errors_for_attribute.include?(expected_message)
2012-05-07 14:25:46 +00:00
expected_message
2010-12-15 22:34:19 +00:00
end
end
def expectation
2012-03-23 14:10:08 +00:00
includes_expected_message = expected_message ? "to include #{expected_message.inspect}" : ''
[error_source, includes_expected_message, "when #{@attribute} is set to #{@value.inspect}"].join(' ')
end
def error_source
message_finder.source_description
2010-12-15 22:34:19 +00:00
end
def error_description
message_finder.messages_description
end
2012-03-23 14:10:08 +00:00
def allowed_values
if @values_to_match.length > 1
"any of [#{@values_to_match.map(&:inspect).join(', ')}]"
else
@values_to_match.first.inspect
end
end
def expected_message
2012-05-07 14:25:46 +00:00
if @options.key?(:expected_message)
if Symbol === @options[:expected_message]
default_expected_message
2012-03-23 14:10:08 +00:00
else
2012-05-07 14:25:46 +00:00
@options[:expected_message]
2012-03-23 14:10:08 +00:00
end
end
end
def default_expected_message
message_finder.expected_message_from(default_attribute_message)
end
def default_attribute_message
default_error_message(
@options[:expected_message],
:model_name => model_name,
:instance => @instance,
:attribute => @attribute
)
end
2012-03-23 14:10:08 +00:00
def model_name
@instance.class.to_s.underscore
end
def message_finder
@message_finder_factory.new(@instance, @attribute)
end
2012-03-23 14:10:08 +00:00
end
2010-12-15 22:34:19 +00:00
end
end
end