Module: Sequel::Plugins::ValidationClassMethods::ClassMethods
- Defined in:
- lib/sequel/plugins/validation_class_methods.rb
Defined Under Namespace
Classes: Generator
Instance Attribute Summary collapse
-
#validation_reflections ⇒ Object
readonly
A hash of validation reflections for this model class.
-
#validations ⇒ Object
readonly
A hash of validations for this model class.
Instance Method Summary collapse
-
#freeze ⇒ Object
Freeze validation metadata when freezing model class.
-
#has_validations? ⇒ Boolean
Returns true if validations are defined.
-
#skip_superclass_validations ⇒ Object
Instructs the model to skip validations defined in superclasses.
-
#skip_superclass_validations? ⇒ Boolean
Instructs the model to skip validations defined in superclasses.
-
#validate(o) ⇒ Object
Validates the given instance.
-
#validates(&block) ⇒ Object
Defines validations by converting a longhand block into a series of shorthand definitions.
-
#validates_acceptance_of(*atts) ⇒ Object
Validates acceptance of an attribute.
-
#validates_confirmation_of(*atts) ⇒ Object
Validates confirmation of an attribute.
-
#validates_each(*atts, &block) ⇒ Object
Adds a validation for each of the given attributes using the supplied block.
-
#validates_format_of(*atts) ⇒ Object
Validates the format of an attribute, checking the string representation of the value against the regular expression provided by the :with option.
-
#validates_inclusion_of(*atts) ⇒ Object
Validates that an attribute is within a specified range or set of values.
-
#validates_length_of(*atts) ⇒ Object
Validates the length of an attribute.
-
#validates_numericality_of(*atts) ⇒ Object
Validates whether an attribute is a number.
-
#validates_presence_of(*atts) ⇒ Object
Validates the presence of an attribute.
-
#validates_schema_type(*atts) ⇒ Object
Validates whether an attribute has the correct ruby type for the associated database type.
-
#validates_uniqueness_of(*atts) ⇒ Object
Validates only if the fields in the model (specified by atts) are unique in the database.
Instance Attribute Details
#validation_reflections ⇒ Object (readonly)
A hash of validation reflections for this model class. Keys are column symbols, values are an array of two element arrays, with the first element being the validation type symbol and the second being a hash of validation options.
38 39 40 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 38 def validation_reflections @validation_reflections end |
#validations ⇒ Object (readonly)
A hash of validations for this model class. Keys are column symbols, values are arrays of validation procs.
32 33 34 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 32 def validations @validations end |
Instance Method Details
#freeze ⇒ Object
Freeze validation metadata when freezing model class.
41 42 43 44 45 46 47 48 49 50 51 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 41 def freeze @validations.freeze.each_value(&:freeze) @validation_reflections.freeze.each_value do |vs| vs.freeze.each do |v| v.freeze v.last.freeze end end super end |
#has_validations? ⇒ Boolean
Returns true if validations are defined.
74 75 76 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 74 def has_validations? !validations.empty? end |
#skip_superclass_validations ⇒ Object
Instructs the model to skip validations defined in superclasses
81 82 83 84 85 86 87 88 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 81 def skip_superclass_validations superclass.validations.each do |att, procs| if @validations[att] @validations[att] -= procs end end @skip_superclass_validations = true end |
#skip_superclass_validations? ⇒ Boolean
Instructs the model to skip validations defined in superclasses
91 92 93 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 91 def skip_superclass_validations? @skip_superclass_validations end |
#validate(o) ⇒ Object
Validates the given instance.
116 117 118 119 120 121 122 123 124 125 126 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 116 def validate(o) validations.each do |att, procs| v = case att when Array att.map{|a| o.get_column_value(a)} else o.get_column_value(att) end procs.each {|tag, p| p.call(o, att, v)} end end |
#validates(&block) ⇒ Object
Defines validations by converting a longhand block into a series of shorthand definitions. For example:
class MyClass < Sequel::Model
validates do
length_of :name, minimum: 6
length_of :password, minimum: 8
end
end
is equivalent to:
class MyClass < Sequel::Model
validates_length_of :name, minimum: 6
validates_length_of :password, minimum: 8
end
111 112 113 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 111 def validates(&block) Generator.new(self, &block) end |
#validates_acceptance_of(*atts) ⇒ Object
Validates acceptance of an attribute. Just checks that the value is equal to the :accept option. This method is unique in that :allow_nil is assumed to be true instead of false.
Possible Options:
- :accept
-
The value required for the object to be valid (default: ‘1’)
- :message
-
The message to use (default: ‘is not accepted’)
135 136 137 138 139 140 141 142 143 144 145 146 147 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 135 def validates_acceptance_of(*atts) opts = { :message => 'is not accepted', :allow_nil => true, :accept => '1', :tag => :acceptance, }.merge!((atts)) reflect_validation(:acceptance, opts, atts) atts << opts validates_each(*atts) do |o, a, v| o.errors.add(a, opts[:message]) unless v == opts[:accept] end end |
#validates_confirmation_of(*atts) ⇒ Object
Validates confirmation of an attribute. Checks that the object has a _confirmation value matching the current value. For example:
validates_confirmation_of :blah
Just makes sure that object.blah = object.blah_confirmation. Often used for passwords or email addresses on web forms.
Possible Options:
- :message
-
The message to use (default: ‘is not confirmed’)
159 160 161 162 163 164 165 166 167 168 169 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 159 def validates_confirmation_of(*atts) opts = { :message => 'is not confirmed', :tag => :confirmation, }.merge!((atts)) reflect_validation(:confirmation, opts, atts) atts << opts validates_each(*atts) do |o, a, v| o.errors.add(a, opts[:message]) unless v == o.get_column_value(:"#{a}_confirmation") end end |
#validates_each(*atts, &block) ⇒ Object
Adds a validation for each of the given attributes using the supplied block. The block must accept three arguments: instance, attribute and value, e.g.:
validates_each :name, :password do |object, attribute, value|
object.errors.add(attribute, 'is not nice') unless value.nice?
end
Possible Options:
- :allow_blank
-
Whether to skip the validation if the value is blank.
- :allow_missing
-
Whether to skip the validation if the attribute isn’t a key in the values hash. This is different from allow_nil, because Sequel only sends the attributes in the values when doing an insert or update. If the attribute is not present, Sequel doesn’t specify it, so the database will use the table’s default value. This is different from having an attribute in values with a value of nil, which Sequel will send as NULL. If your database table has a non NULL default, this may be a good option to use. You don’t want to use allow_nil, because if the attribute is in values but has a value nil, Sequel will attempt to insert a NULL value into the database, instead of using the database’s default.
- :allow_nil
-
Whether to skip the validation if the value is nil.
- :if
-
A symbol (indicating an instance_method) or proc (which is used to define an instance method) skipping this validation if it returns nil or false.
- :tag
-
The tag to use for this validation.
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 194 def validates_each(*atts, &block) opts = (atts) blank_meth = db.method(:blank_object?).to_proc i = opts[:if] am = opts[:allow_missing] an = opts[:allow_nil] ab = opts[:allow_blank] blk = if i || am || an || ab if i.is_a?(Proc) i = Plugins.def_sequel_method(self, "validation_class_methods_if", 0, &i) end proc do |o,a,v| next if i && !validation_if_proc(o, i) next if an && Array(v).all?(&:nil?) next if ab && Array(v).all?(&blank_meth) next if am && Array(a).all?{|x| !o.values.has_key?(x)} block.call(o,a,v) end else block end tag = opts[:tag] atts.each do |a| a_vals = Sequel.synchronize{validations[a] ||= []} if tag && (old = a_vals.find{|x| x[0] == tag}) old[1] = blk else a_vals << [tag, blk] end end end |
#validates_format_of(*atts) ⇒ Object
Validates the format of an attribute, checking the string representation of the value against the regular expression provided by the :with option.
Possible Options:
- :message
-
The message to use (default: ‘is invalid’)
- :with
-
The regular expression to validate the value with (required).
233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 233 def validates_format_of(*atts) opts = { :message => 'is invalid', :tag => :format, }.merge!((atts)) unless opts[:with].is_a?(Regexp) raise ArgumentError, "A regular expression must be supplied as the :with option of the options hash" end reflect_validation(:format, opts, atts) atts << opts validates_each(*atts) do |o, a, v| o.errors.add(a, opts[:message]) unless v.to_s =~ opts[:with] end end |
#validates_inclusion_of(*atts) ⇒ Object
Validates that an attribute is within a specified range or set of values.
Possible Options:
- :in
-
An array or range of values to check for validity (required)
- :message
-
The message to use (default: ‘is not in range or set: <specified range>’)
337 338 339 340 341 342 343 344 345 346 347 348 349 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 337 def validates_inclusion_of(*atts) opts = (atts) n = opts[:in] unless n && (n.respond_to?(:cover?) || n.respond_to?(:include?)) raise ArgumentError, "The :in parameter is required, and must respond to cover? or include?" end opts[:message] ||= "is not in range or set: #{n.inspect}" reflect_validation(:inclusion, opts, atts) atts << opts validates_each(*atts) do |o, a, v| o.errors.add(a, opts[:message]) unless n.public_send(n.respond_to?(:cover?) ? :cover? : :include?, v) end end |
#validates_length_of(*atts) ⇒ Object
Validates the length of an attribute.
Possible Options:
- :is
-
The exact size required for the value to be valid (no default)
- :maximum
-
The maximum size allowed for the value (no default)
- :message
-
The message to use (no default, overrides :nil_message, :too_long, :too_short, and :wrong_length options if present)
- :minimum
-
The minimum size allowed for the value (no default)
- :nil_message
-
The message to use use if :maximum option is used and the value is nil (default: ‘is not present’)
- :too_long
-
The message to use use if it the value is too long (default: ‘is too long’)
- :too_short
-
The message to use use if it the value is too short (default: ‘is too short’)
- :within
-
The array/range that must include the size of the value for it to be valid (no default)
- :wrong_length
-
The message to use use if it the value is not valid (default: ‘is the wrong length’)
263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 263 def validates_length_of(*atts) opts = { :nil_message => 'is not present', :too_long => 'is too long', :too_short => 'is too short', :wrong_length => 'is the wrong length' }.merge!((atts)) opts[:tag] ||= ([:length] + [:maximum, :minimum, :is, :within].reject{|x| !opts.include?(x)}).join('-').to_sym reflect_validation(:length, opts, atts) atts << opts validates_each(*atts) do |o, a, v| if m = opts[:maximum] o.errors.add(a, opts[:message] || (v ? opts[:too_long] : opts[:nil_message])) unless v && v.size <= m end if m = opts[:minimum] o.errors.add(a, opts[:message] || opts[:too_short]) unless v && v.size >= m end if i = opts[:is] o.errors.add(a, opts[:message] || opts[:wrong_length]) unless v && v.size == i end if w = opts[:within] o.errors.add(a, opts[:message] || opts[:wrong_length]) unless v && w.public_send(w.respond_to?(:cover?) ? :cover? : :include?, v.size) end end end |
#validates_numericality_of(*atts) ⇒ Object
Validates whether an attribute is a number.
Possible Options:
- :message
-
The message to use (default: ‘is not a number’)
- :only_integer
-
Whether only integers are valid values (default: false)
295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 295 def validates_numericality_of(*atts) opts = { :message => 'is not a number', :tag => :numericality, }.merge!((atts)) reflect_validation(:numericality, opts, atts) atts << opts validates_each(*atts) do |o, a, v| begin if opts[:only_integer] Kernel.Integer(v.to_s) else Kernel.Float(v.to_s) end rescue o.errors.add(a, opts[:message]) end end end |
#validates_presence_of(*atts) ⇒ Object
Validates the presence of an attribute. Requires the value not be blank, with false considered present instead of absent.
Possible Options:
- :message
-
The message to use (default: ‘is not present’)
320 321 322 323 324 325 326 327 328 329 330 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 320 def validates_presence_of(*atts) opts = { :message => 'is not present', :tag => :presence, }.merge!((atts)) reflect_validation(:presence, opts, atts) atts << opts validates_each(*atts) do |o, a, v| o.errors.add(a, opts[:message]) if db.send(:blank_object?, v) && v != false end end |
#validates_schema_type(*atts) ⇒ Object
Validates whether an attribute has the correct ruby type for the associated database type. This is generally useful in conjunction with raise_on_typecast_failure = false, to handle typecasting errors at validation time instead of at setter time.
Possible Options:
- :message
-
The message to use (default: ‘is not a valid (integer|datetime|etc.)’)
358 359 360 361 362 363 364 365 366 367 368 369 370 371 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 358 def validates_schema_type(*atts) opts = { :tag => :schema_type, }.merge!((atts)) reflect_validation(:schema_type, opts, atts) atts << opts validates_each(*atts) do |o, a, v| next if v.nil? || (klass = o.send(:schema_type_class, a)).nil? if klass.is_a?(Array) ? !klass.any?{|kls| v.is_a?(kls)} : !v.is_a?(klass) = opts[:message] || "is not a valid #{Array(klass).join(" or ").downcase}" o.errors.add(a, ) end end end |
#validates_uniqueness_of(*atts) ⇒ Object
Validates only if the fields in the model (specified by atts) are unique in the database. Pass an array of fields instead of multiple fields to specify that the combination of fields must be unique, instead of that each field should have a unique value.
This means that the code:
validates_uniqueness_of([:column1, :column2])
validates the grouping of column1 and column2 while
validates_uniqueness_of(:column1, :column2)
validates them separately.
You should also add a unique index in the database, as this suffers from a fairly obvious race condition.
Possible Options:
- :message
-
The message to use (default: ‘is already taken’)
389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 |
# File 'lib/sequel/plugins/validation_class_methods.rb', line 389 def validates_uniqueness_of(*atts) opts = { :message => 'is already taken', :tag => :uniqueness, }.merge!((atts)) reflect_validation(:uniqueness, opts, atts) atts << opts validates_each(*atts) do |o, a, v| error_field = a a = Array(a) v = Array(v) next if v.empty? || !v.all? ds = o.class.where(a.zip(v)) num_dups = ds.count allow = if num_dups == 0 # No unique value in the database true elsif num_dups > 1 # Multiple "unique" values in the database!! # Someone didn't add a unique index false elsif o.new? # New record, but unique value already exists in the database false elsif ds.first === o # Unique value exists in database, but for the same record, so the update won't cause a duplicate record true else false end o.errors.add(error_field, opts[:message]) unless allow end end |