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

346 lines
10 KiB
Ruby
Raw Normal View History

module Shoulda
2010-12-15 22:34:19 +00:00
module Matchers
module ActiveModel
# The `allow_value` matcher is used to test that an attribute of a model
# can or cannot be set to a particular value or values. It is most
# commonly used in conjunction with the `validates_format_of` validation.
#
# #### should
#
# In the positive form, `allow_value` asserts that an attribute can be
# set to one or more values, succeeding if none of the values cause the
# record to be invalid:
#
# class UserProfile
# include ActiveModel::Model
# attr_accessor :website_url
#
# validates_format_of :website_url, with: URI.regexp
# end
#
# # RSpec
# describe UserProfile do
# it do
# should allow_value('http://foo.com', 'http://bar.com/baz').
# for(:website_url)
# end
# end
#
# # Test::Unit
# class UserProfileTest < ActiveSupport::TestCase
# should allow_value('http://foo.com', 'http://bar.com/baz').
# for(:website_url)
# end
#
# #### should_not
#
# In the negative form, `allow_value` asserts that an attribute cannot be
# set to one or more values, succeeding if the *first* value causes the
# record to be invalid.
#
# **This can be surprising** so in this case if you need to check that
# *all* of the values are invalid, use separate assertions:
#
# class UserProfile
# include ActiveModel::Model
# attr_accessor :website_url
#
# validates_format_of :website_url, with: URI.regexp
# end
#
# describe UserProfile do
# # One assertion: 'buz' and 'bar' will not be tested
# it { should_not allow_value('fiz', 'buz', 'bar').for(:website_url) }
#
# # Three assertions, all tested separately
# it { should_not allow_value('fiz').for(:website_url) }
# it { should_not allow_value('buz').for(:website_url) }
# it { should_not allow_value('bar').for(:website_url) }
# end
#
# #### Qualifiers
#
# ##### on
#
# Use `on` if your validation applies only under a certain context.
#
# class UserProfile
# include ActiveModel::Model
# attr_accessor :birthday_as_string
#
# validates_format_of :birthday_as_string,
# with: /^(\d+)-(\d+)-(\d+)$/,
# on: :create
# end
#
# # RSpec
# describe UserProfile do
# it do
# should allow_value('2013-01-01').
# for(:birthday_as_string).
# on(:create)
# end
# end
#
# # Test::Unit
# class UserProfileTest < ActiveSupport::TestCase
# should allow_value('2013-01-01').
# for(:birthday_as_string).
# on(:create)
# end
#
# ##### with_message
#
# Use `with_message` if you are using a custom validation message.
#
# class UserProfile
# include ActiveModel::Model
# attr_accessor :state
#
# validates_format_of :state,
# with: /^(open|closed)$/,
# message: 'State must be open or closed'
# end
#
# # RSpec
# describe UserProfile do
# it do
# should allow_value('open', 'closed').
# for(:state).
# with_message('State must be open or closed')
# end
# end
#
# # Test::Unit
# class UserProfileTest < ActiveSupport::TestCase
# should allow_value('open', 'closed').
# for(:state).
# with_message('State must be open or closed')
# end
#
# Use `with_message` with the `:against` option if the attribute the
# validation message is stored under is different from the attribute
# being validated.
#
# class UserProfile
# include ActiveModel::Model
# attr_accessor :sports_team
#
# validate :sports_team_must_be_valid
#
# private
#
# def sports_team_must_be_valid
# if sports_team !~ /^(Broncos|Titans)$/i
# self.errors.add :chosen_sports_team,
# 'Must be either a Broncos fan or a Titans fan'
# end
# end
# end
#
# # RSpec
# describe UserProfile do
# it do
# should allow_value('Broncos', 'Titans').
# for(:sports_team).
# with_message('Must be either a Broncos or Titans fan',
# against: :chosen_sports_team
# )
# end
# end
#
# # Test::Unit
# class UserProfileTest < ActiveSupport::TestCase
# should allow_value('Broncos', 'Titans').
# for(:sports_team).
# with_message('Must be either a Broncos or Titans fan',
# against: :chosen_sports_team
# )
# end
#
# @return [AllowValueMatcher]
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
# @private
class AllowValueMatcher
2010-12-15 22:34:19 +00:00
include Helpers
attr_accessor :attribute_with_message
attr_accessor :options
def initialize(*values)
self.values_to_match = values
self.message_finder_factory = ValidationMessageFinder
self.options = {}
self.after_setting_value_callback = -> {}
2010-12-15 22:34:19 +00:00
end
def for(attribute)
self.attribute_to_set = attribute
self.attribute_to_check_message_against = attribute
2010-12-15 22:34:19 +00:00
self
end
2013-03-04 03:34:38 +00:00
def on(context)
@context = context
self
end
def with_message(message, options={})
self.options[:expected_message] = message
self.options[:expected_message_values] = options.fetch(:values, {})
if options.key?(:against)
self.attribute_to_check_message_against = options[:against]
end
2010-12-15 22:34:19 +00:00
self
end
def strict
self.message_finder_factory = ExceptionMessageFinder
self
end
2014-07-24 04:03:51 +00:00
def _after_setting_value(&callback)
self.after_setting_value_callback = callback
end
2010-12-15 22:34:19 +00:00
def matches?(instance)
self.instance = instance
values_to_match.none? do |value|
self.value = value
set_value(value)
errors_match?
end
2010-12-15 22:34:19 +00:00
end
def failure_message
"Did not expect #{expectation},\ngot error: #{matched_error}"
2010-12-15 22:34:19 +00:00
end
alias failure_message_for_should failure_message
2010-12-15 22:34:19 +00:00
def failure_message_when_negated
"Expected #{expectation},\ngot#{error_description}"
2010-12-15 22:34:19 +00:00
end
alias failure_message_for_should_not failure_message_when_negated
2010-12-15 22:34:19 +00:00
def description
message_finder.allow_description(allowed_values)
2010-12-15 22:34:19 +00:00
end
protected
2010-12-15 22:34:19 +00:00
attr_accessor :values_to_match, :message_finder_factory,
:instance, :attribute_to_set, :attribute_to_check_message_against,
:context, :value, :matched_error, :after_setting_value_callback
def set_value(value)
instance.__send__("#{attribute_to_set}=", value)
after_setting_value_callback.call
end
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
self.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_to_set} 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(', ')}]"
2012-03-23 14:10:08 +00:00
else
values_to_match.first.inspect
2012-03-23 14:10:08 +00:00
end
end
def expected_message
if options.key?(:expected_message)
if Symbol === options[:expected_message]
default_expected_message
2012-03-23 14:10:08 +00:00
else
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],
default_attribute_message_values
)
end
def default_attribute_message_values
defaults = {
2014-01-17 20:20:44 +00:00
model_name: model_name,
instance: instance,
attribute: attribute_to_set,
}
defaults.merge(options[:expected_message_values])
end
2012-03-23 14:10:08 +00:00
def model_name
instance.class.to_s.underscore
2012-03-23 14:10:08 +00:00
end
def message_finder
message_finder_factory.new(instance, attribute_to_check_message_against, context)
end
2012-03-23 14:10:08 +00:00
end
2010-12-15 22:34:19 +00:00
end
end
end