Class: DataMapper::Validate::NumericValidator

Inherits:
GenericValidator show all
Defined in:
lib/dm-validations/numeric_validator.rb

Overview

Author:

  • Guy van den Berg

Since:

  • 0.9

Instance Attribute Summary

Attributes inherited from GenericValidator

#if_clause, #unless_clause

Instance Method Summary collapse

Methods inherited from GenericValidator

#==, #add_error, #execute?, #field_name

Constructor Details

#initialize(field_name, options = {}) ⇒ NumericValidator

Returns a new instance of NumericValidator.

Since:

  • 0.9



10
11
12
13
14
# File 'lib/dm-validations/numeric_validator.rb', line 10

def initialize(field_name, options={})
  super
  @field_name, @options = field_name, options
  @options[:integer_only] = false unless @options.has_key?(:integer_only)
end

Instance Method Details

#call(target) ⇒ Object

Since:

  • 0.9



16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
# File 'lib/dm-validations/numeric_validator.rb', line 16

def call(target)
  value = target.send(field_name)
  return true if @options[:allow_nil] && value.nil?

  value = value.kind_of?(BigDecimal) ? value.to_s('F') : value.to_s

  error_message = @options[:message]
  precision     = @options[:precision]
  scale         = @options[:scale]

  if @options[:integer_only]
    return true if value =~ /\A[+-]?\d+\z/
    error_message ||= '%s must be an integer'.t(Extlib::Inflection.humanize(@field_name))
  else
    # FIXME: if precision and scale are not specified, can we assume that it is an integer?
    #        probably not, as floating point numbers don't have hard
    #        defined scale. the scale floats with the length of the
    #        integral and precision. Ie. if precision = 10 and integral
    #        portion of the number is 9834 (4 digits), the max scale will
    #        be 6 (10 - 4). But if the integral length is 1, max scale
    #        will be (10 - 1) = 9, so 1.234567890.
    #        In MySQL somehow you can hard-define scale on floats. Not
    #        quite sure how that works...
    if precision && scale
      #handles both Float when it has scale specified and BigDecimal
      if precision > scale && scale > 0
        return true if value =~ /\A[+-]?(?:\d{1,#{precision - scale}}|\d{0,#{precision - scale}}\.\d{1,#{scale}})\z/
      elsif precision > scale && scale == 0
        return true if value =~ /\A[+-]?(?:\d{1,#{precision}}(?:\.0)?)\z/
      elsif precision == scale
        return true if value =~ /\A[+-]?(?:0(?:\.\d{1,#{scale}})?)\z/
      else
        raise ArgumentError, "Invalid precision #{precision.inspect} and scale #{scale.inspect} for #{field_name} (value: #{value.inspect} #{value.class})"
      end
    elsif precision && scale.nil?
      # for floats, if scale is not set

      #total number of digits is less or equal precision
      return true if value.gsub(/[^\d]/,'').length <= precision

      #number of digits before decimal == precision, and the number is x.0. same as scale = 0
      return true if value =~ /\A[+-]?(?:\d{1,#{precision}}(?:\.0)?)\z/
    else
      return true if value =~ /\A[+-]?(?:\d+|\d*\.\d+)\z/
    end
    error_message ||= '%s must be a number'.t(Extlib::Inflection.humanize(@field_name))
  end

  add_error(target, error_message, @field_name)

  # TODO: check the gt, gte, lt, lte, and eq options

  return false
end