2014-01-23 13:07:36 -05:00
|
|
|
module Shoulda
|
2010-12-15 17:34:19 -05:00
|
|
|
module Matchers
|
2014-01-23 13:07:36 -05:00
|
|
|
module ActiveModel
|
|
|
|
# The `validate_numericality_of` matcher tests usage of the
|
|
|
|
# `validates_numericality_of` validation.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :gpa
|
|
|
|
#
|
|
|
|
# validates_numericality_of :gpa
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it { should validate_numericality_of(:gpa) }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:gpa)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# #### Qualifiers
|
|
|
|
#
|
2015-03-28 11:03:31 -04:00
|
|
|
# ##### on
|
|
|
|
#
|
|
|
|
# Use `on` if your validation applies only under a certain context.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :number_of_dependents
|
|
|
|
#
|
|
|
|
# validates_numericality_of :number_of_dependents, on: :create
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:number_of_dependents).
|
|
|
|
# on(:create)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:number_of_dependents).on(:create)
|
|
|
|
# end
|
|
|
|
#
|
2014-01-23 13:07:36 -05:00
|
|
|
# ##### only_integer
|
|
|
|
#
|
|
|
|
# Use `only_integer` to test usage of the `:only_integer` option. This
|
|
|
|
# asserts that your attribute only allows integer numbers and disallows
|
|
|
|
# non-integer ones.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :age
|
|
|
|
#
|
|
|
|
# validates_numericality_of :age, only_integer: true
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it { should validate_numericality_of(:age).only_integer }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:age).only_integer
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### is_less_than
|
|
|
|
#
|
|
|
|
# Use `is_less_than` to test usage of the the `:less_than` option. This
|
|
|
|
# asserts that the attribute can take a number which is less than the
|
|
|
|
# given value and cannot take a number which is greater than or equal to
|
|
|
|
# it.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :number_of_cars
|
|
|
|
#
|
|
|
|
# validates_numericality_of :number_of_cars, less_than: 2
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:number_of_cars).
|
|
|
|
# is_less_than(2)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:number_of_cars).
|
|
|
|
# is_less_than(2)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### is_less_than_or_equal_to
|
|
|
|
#
|
|
|
|
# Use `is_less_than_or_equal_to` to test usage of the
|
|
|
|
# `:less_than_or_equal_to` option. This asserts that the attribute can
|
|
|
|
# take a number which is less than or equal to the given value and cannot
|
|
|
|
# take a number which is greater than it.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :birth_year
|
|
|
|
#
|
|
|
|
# validates_numericality_of :birth_year, less_than_or_equal_to: 1987
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:birth_year).
|
|
|
|
# is_less_than_or_equal_to(1987)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:birth_year).
|
|
|
|
# is_less_than_or_equal_to(1987)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### is_equal_to
|
|
|
|
#
|
|
|
|
# Use `is_equal_to` to test usage of the `:equal_to` option. This asserts
|
|
|
|
# that the attribute can take a number which is equal to the given value
|
|
|
|
# and cannot take a number which is not equal.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :weight
|
|
|
|
#
|
|
|
|
# validates_numericality_of :weight, equal_to: 150
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it { should validate_numericality_of(:weight).is_equal_to(150) }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:weight).is_equal_to(150)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### is_greater_than_or_equal_to
|
|
|
|
#
|
|
|
|
# Use `is_greater_than_or_equal_to` to test usage of the
|
|
|
|
# `:greater_than_or_equal_to` option. This asserts that the attribute can
|
|
|
|
# take a number which is greater than or equal to the given value and
|
|
|
|
# cannot take a number which is less than it.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :height
|
|
|
|
#
|
|
|
|
# validates_numericality_of :height, greater_than_or_equal_to: 55
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:height).
|
|
|
|
# is_greater_than_or_equal_to(55)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:height).
|
|
|
|
# is_greater_than_or_equal_to(55)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### is_greater_than
|
|
|
|
#
|
|
|
|
# Use `is_greater_than` to test usage of tthe `:greater_than` option.
|
|
|
|
# This asserts that the attribute can take a number which is greater than
|
|
|
|
# the given value and cannot take a number less than or equal to it.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :legal_age
|
|
|
|
#
|
|
|
|
# validates_numericality_of :legal_age, greater_than: 21
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:legal_age).
|
|
|
|
# is_greater_than(21)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:legal_age).
|
|
|
|
# is_greater_than(21)
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### even
|
|
|
|
#
|
|
|
|
# Use `even` to test usage of the `:even` option. This asserts that the
|
|
|
|
# attribute can take odd numbers and cannot take even ones.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :birth_month
|
|
|
|
#
|
|
|
|
# validates_numericality_of :birth_month, even: true
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it { should validate_numericality_of(:birth_month).even }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:birth_month).even
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### odd
|
|
|
|
#
|
|
|
|
# Use `odd` to test usage of the `:odd` option. This asserts that the
|
|
|
|
# attribute can take a number which is odd and cannot take a number which
|
|
|
|
# is even.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :birth_day
|
|
|
|
#
|
|
|
|
# validates_numericality_of :birth_day, odd: true
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it { should validate_numericality_of(:birth_day).odd }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:birth_day).odd
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### with_message
|
|
|
|
#
|
|
|
|
# Use `with_message` if you are using a custom validation message.
|
|
|
|
#
|
|
|
|
# class Person
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :number_of_dependents
|
|
|
|
#
|
|
|
|
# validates_numericality_of :number_of_dependents,
|
|
|
|
# message: 'Number of dependents must be a number'
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Person do
|
|
|
|
# it do
|
|
|
|
# should validate_numericality_of(:number_of_dependents).
|
|
|
|
# with_message('Number of dependents must be a number')
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PersonTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:number_of_dependents).
|
|
|
|
# with_message('Number of dependents must be a number')
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# ##### allow_nil
|
|
|
|
#
|
|
|
|
# Use `allow_nil` to assert that the attribute allows nil.
|
|
|
|
#
|
|
|
|
# class Age
|
|
|
|
# include ActiveModel::Model
|
|
|
|
# attr_accessor :age
|
|
|
|
#
|
|
|
|
# validates_numericality_of :age, allow_nil: true
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # RSpec
|
|
|
|
# describe Post do
|
|
|
|
# it { should validate_numericality_of(:age).allow_nil }
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# # Test::Unit
|
|
|
|
# class PostTest < ActiveSupport::TestCase
|
|
|
|
# should validate_numericality_of(:age).allow_nil
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# @return [ValidateNumericalityOfMatcher]
|
2010-12-15 17:34:19 -05:00
|
|
|
#
|
|
|
|
def validate_numericality_of(attr)
|
|
|
|
ValidateNumericalityOfMatcher.new(attr)
|
|
|
|
end
|
|
|
|
|
2014-01-23 13:07:36 -05:00
|
|
|
# @private
|
2012-10-16 13:45:06 -04:00
|
|
|
class ValidateNumericalityOfMatcher
|
2014-01-12 20:43:36 -05:00
|
|
|
NUMERIC_NAME = 'numbers'
|
2012-10-23 12:30:18 -04:00
|
|
|
NON_NUMERIC_VALUE = 'abcd'
|
2014-04-26 11:13:57 -04:00
|
|
|
DEFAULT_DIFF_TO_COMPARE = 1
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
attr_reader :diff_to_compare
|
2012-10-23 12:30:18 -04:00
|
|
|
|
2012-04-24 18:00:26 -04:00
|
|
|
def initialize(attribute)
|
2012-10-16 13:45:06 -04:00
|
|
|
@attribute = attribute
|
|
|
|
@submatchers = []
|
2014-02-26 22:26:37 -05:00
|
|
|
@diff_to_compare = DEFAULT_DIFF_TO_COMPARE
|
2014-12-03 17:06:23 -05:00
|
|
|
@strict = false
|
2012-10-16 13:45:06 -04:00
|
|
|
add_disallow_value_matcher
|
2012-04-24 18:00:26 -04:00
|
|
|
end
|
2011-10-26 22:21:06 -04:00
|
|
|
|
2014-12-03 17:06:23 -05:00
|
|
|
def strict
|
|
|
|
@submatchers.each(&:strict)
|
|
|
|
@strict = true
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2011-10-26 22:21:06 -04:00
|
|
|
def only_integer
|
2014-02-26 22:26:37 -05:00
|
|
|
prepare_submatcher(
|
|
|
|
NumericalityMatchers::OnlyIntegerMatcher.new(@attribute)
|
|
|
|
)
|
2013-02-19 05:09:13 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-21 17:31:48 -05:00
|
|
|
def allow_nil
|
2014-02-26 22:26:37 -05:00
|
|
|
prepare_submatcher(
|
|
|
|
AllowValueMatcher.new(nil)
|
|
|
|
.for(@attribute)
|
|
|
|
.with_message(:not_a_number)
|
|
|
|
)
|
2014-02-21 17:31:48 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def odd
|
|
|
|
prepare_submatcher(
|
|
|
|
NumericalityMatchers::OddNumberMatcher.new(@attribute)
|
|
|
|
)
|
2013-02-19 05:09:13 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def even
|
|
|
|
prepare_submatcher(
|
|
|
|
NumericalityMatchers::EvenNumberMatcher.new(@attribute)
|
|
|
|
)
|
2013-02-19 05:09:13 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def is_greater_than(value)
|
|
|
|
prepare_submatcher(comparison_matcher_for(value, :>).for(@attribute))
|
2013-02-19 05:09:13 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def is_greater_than_or_equal_to(value)
|
|
|
|
prepare_submatcher(comparison_matcher_for(value, :>=).for(@attribute))
|
2013-02-19 05:09:13 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def is_equal_to(value)
|
|
|
|
prepare_submatcher(comparison_matcher_for(value, :==).for(@attribute))
|
2011-10-26 22:21:06 -04:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def is_less_than(value)
|
|
|
|
prepare_submatcher(comparison_matcher_for(value, :<).for(@attribute))
|
2013-03-19 00:50:39 -04:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def is_less_than_or_equal_to(value)
|
|
|
|
prepare_submatcher(comparison_matcher_for(value, :<=).for(@attribute))
|
2013-03-19 00:50:39 -04:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2010-12-15 17:34:19 -05:00
|
|
|
def with_message(message)
|
2012-10-23 12:30:18 -04:00
|
|
|
@submatchers.each { |matcher| matcher.with_message(message) }
|
2010-12-15 17:34:19 -05:00
|
|
|
self
|
|
|
|
end
|
|
|
|
|
|
|
|
def matches?(subject)
|
2012-10-16 13:45:06 -04:00
|
|
|
@subject = subject
|
2015-01-22 20:47:43 -05:00
|
|
|
failing_submatchers.empty?
|
2010-12-15 17:34:19 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def description
|
2014-12-03 17:06:23 -05:00
|
|
|
description_parts = ["only allow #{allowed_types} for #{@attribute}"]
|
|
|
|
|
|
|
|
if comparison_descriptions.present?
|
|
|
|
description_parts << comparison_descriptions
|
|
|
|
end
|
|
|
|
|
|
|
|
if @strict
|
|
|
|
description_parts.insert(1, 'strictly')
|
|
|
|
description_parts.join(', ')
|
|
|
|
else
|
|
|
|
description_parts.join(' ')
|
|
|
|
end
|
2012-10-16 13:45:06 -04:00
|
|
|
end
|
|
|
|
|
2013-12-24 06:24:27 -05:00
|
|
|
def failure_message
|
2015-01-22 20:47:43 -05:00
|
|
|
last_failing_submatcher.failure_message
|
2013-04-12 13:11:07 -04:00
|
|
|
end
|
2013-12-24 06:24:27 -05:00
|
|
|
alias failure_message_for_should failure_message
|
2013-04-12 13:11:07 -04:00
|
|
|
|
2013-12-24 06:24:27 -05:00
|
|
|
def failure_message_when_negated
|
2015-01-22 20:47:43 -05:00
|
|
|
last_failing_submatcher.failure_message_when_negated
|
2010-12-15 17:34:19 -05:00
|
|
|
end
|
2013-12-24 06:24:27 -05:00
|
|
|
alias failure_message_for_should_not failure_message_when_negated
|
2010-12-15 17:34:19 -05:00
|
|
|
|
2012-03-30 10:36:55 -04:00
|
|
|
private
|
|
|
|
|
2012-10-16 13:45:06 -04:00
|
|
|
def add_disallow_value_matcher
|
2012-10-23 12:30:18 -04:00
|
|
|
disallow_value_matcher = DisallowValueMatcher.new(NON_NUMERIC_VALUE).
|
|
|
|
for(@attribute).
|
|
|
|
with_message(:not_a_number)
|
|
|
|
|
|
|
|
add_submatcher(disallow_value_matcher)
|
2012-04-03 20:20:50 -04:00
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def prepare_submatcher(submatcher)
|
|
|
|
add_submatcher(submatcher)
|
|
|
|
if submatcher.respond_to?(:diff_to_compare)
|
|
|
|
update_diff_to_compare(submatcher)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def comparison_matcher_for(value, operator)
|
|
|
|
NumericalityMatchers::ComparisonMatcher
|
|
|
|
.new(self, value, operator)
|
|
|
|
.for(@attribute)
|
|
|
|
end
|
|
|
|
|
2012-10-16 13:45:06 -04:00
|
|
|
def add_submatcher(submatcher)
|
|
|
|
@submatchers << submatcher
|
2011-10-26 22:21:06 -04:00
|
|
|
end
|
|
|
|
|
2014-02-26 22:26:37 -05:00
|
|
|
def update_diff_to_compare(matcher)
|
|
|
|
@diff_to_compare = [@diff_to_compare, matcher.diff_to_compare].max
|
|
|
|
end
|
|
|
|
|
2015-01-22 20:47:43 -05:00
|
|
|
def submatchers_and_results
|
|
|
|
@_submatchers_and_results ||=
|
|
|
|
@submatchers.map do |matcher|
|
|
|
|
{ matcher: matcher, matched: matcher.matches?(@subject) }
|
|
|
|
end
|
2012-10-23 12:30:18 -04:00
|
|
|
end
|
|
|
|
|
2015-01-22 20:47:43 -05:00
|
|
|
def failing_submatchers
|
|
|
|
submatchers_and_results.
|
|
|
|
select { |x| !x[:matched] }.
|
|
|
|
map { |x| x[:matcher] }
|
2013-04-12 13:11:07 -04:00
|
|
|
end
|
|
|
|
|
2015-01-22 20:47:43 -05:00
|
|
|
def last_failing_submatcher
|
|
|
|
failing_submatchers.last
|
2012-10-16 13:45:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def allowed_types
|
2014-01-12 20:43:36 -05:00
|
|
|
allowed_array = submatcher_allowed_types
|
|
|
|
allowed_array.empty? ? NUMERIC_NAME : allowed_array.join(', ')
|
2012-10-16 13:45:06 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def submatcher_allowed_types
|
2014-01-12 20:43:36 -05:00
|
|
|
@submatchers.inject([]){|m, s| m << s.allowed_type if s.respond_to?(:allowed_type); m }
|
|
|
|
end
|
|
|
|
|
|
|
|
def comparison_descriptions
|
|
|
|
description_array = submatcher_comparison_descriptions
|
2014-12-03 17:06:23 -05:00
|
|
|
description_array.empty? ? '' : 'which are ' + submatcher_comparison_descriptions.join(' and ')
|
2014-01-12 20:43:36 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def submatcher_comparison_descriptions
|
2014-02-26 22:26:37 -05:00
|
|
|
@submatchers.inject([]) do |arr, submatcher|
|
|
|
|
if submatcher.respond_to? :comparison_description
|
|
|
|
arr << submatcher.comparison_description
|
|
|
|
end
|
|
|
|
arr
|
|
|
|
end
|
2012-03-30 10:36:55 -04:00
|
|
|
end
|
|
|
|
end
|
2010-12-15 17:34:19 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|