module Remarkable
module ActiveModel
module Matchers
class AllowValuesForMatcher < Remarkable::ActiveModel::Base #:nodoc:
include Remarkable::Negative
arguments :collection => :attributes, :as => :attribute
optional :message
optional :in, :splat => true
optional :allow_nil, :allow_blank, :default => true
collection_assertions :is_valid?, :is_invalid?, :allow_nil?, :allow_blank?
default_options :message => /.*/
before_assert do
first_value = @options[:in].is_a?(Array) ? @options[:in].first : @options[:in]
@in_range = first_value.is_a?(Range)
@options[:in] = if @in_range
first_value.to_a[0,2] + first_value.to_a[-2,2]
else
[*@options[:in]].compact
end
@options[:in].uniq!
end
protected
def is_valid?
assert_collection :value, valid_values do |value|
good?(value)
end
end
def is_invalid?
assert_collection :value, invalid_values do |value|
bad?(value)
end
end
def valid_values
@options[:in]
end
def invalid_values
[]
end
def interpolation_options
options = if @in_range
{ :in => (@options[:in].first..@options[:in].last).inspect }
elsif @options[:in].is_a?(Array)
{ :in => array_to_sentence(@options[:in], true, '[]') }
else
{ :in => @options[:in].inspect }
end
options.merge!(:behavior => @behavior.to_s)
end
end
# Ensures that the attribute can be set to the given values. It checks
# for any message in the given attribute unless a :message is explicitely
# given.
#
# == Options
#
# * :allow_nil - when supplied, validates if it allows nil or not.
# * :allow_blank - when supplied, validates if it allows blank or not.
# * :message - value the test expects to find in errors[:attribute].
# Regexp, string or symbol. Default = /.*/
#
# == Examples
#
# should_allow_values_for :isbn, "isbn 1 2345 6789 0", "ISBN 1-2345-6789-0"
# it { should allow_values_for(:isbn, "isbn 1 2345 6789 0", "ISBN 1-2345-6789-0") }
#
def allow_values_for(attribute, *args, &block)
options = args.extract_options!
AllowValuesForMatcher.new(attribute, options.merge!(:in => args), &block).spec(self)
end
end
end
end