couchrest_extended_document 1.0.0.beta5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/LICENSE +176 -0
- data/README.md +68 -0
- data/Rakefile +68 -0
- data/THANKS.md +19 -0
- data/examples/model/example.rb +144 -0
- data/history.txt +159 -0
- data/lib/couchrest/casted_array.rb +25 -0
- data/lib/couchrest/casted_model.rb +55 -0
- data/lib/couchrest/extended_document.rb +323 -0
- data/lib/couchrest/mixins/attribute_protection.rb +74 -0
- data/lib/couchrest/mixins/callbacks.rb +532 -0
- data/lib/couchrest/mixins/class_proxy.rb +120 -0
- data/lib/couchrest/mixins/collection.rb +260 -0
- data/lib/couchrest/mixins/design_doc.rb +127 -0
- data/lib/couchrest/mixins/document_queries.rb +82 -0
- data/lib/couchrest/mixins/extended_attachments.rb +73 -0
- data/lib/couchrest/mixins/properties.rb +162 -0
- data/lib/couchrest/mixins/validation.rb +245 -0
- data/lib/couchrest/mixins/views.rb +148 -0
- data/lib/couchrest/mixins.rb +11 -0
- data/lib/couchrest/property.rb +50 -0
- data/lib/couchrest/support/couchrest.rb +19 -0
- data/lib/couchrest/support/rails.rb +42 -0
- data/lib/couchrest/typecast.rb +175 -0
- data/lib/couchrest/validation/auto_validate.rb +156 -0
- data/lib/couchrest/validation/contextual_validators.rb +78 -0
- data/lib/couchrest/validation/validation_errors.rb +125 -0
- data/lib/couchrest/validation/validators/absent_field_validator.rb +74 -0
- data/lib/couchrest/validation/validators/confirmation_validator.rb +107 -0
- data/lib/couchrest/validation/validators/format_validator.rb +122 -0
- data/lib/couchrest/validation/validators/formats/email.rb +66 -0
- data/lib/couchrest/validation/validators/formats/url.rb +43 -0
- data/lib/couchrest/validation/validators/generic_validator.rb +120 -0
- data/lib/couchrest/validation/validators/length_validator.rb +139 -0
- data/lib/couchrest/validation/validators/method_validator.rb +89 -0
- data/lib/couchrest/validation/validators/numeric_validator.rb +109 -0
- data/lib/couchrest/validation/validators/required_field_validator.rb +114 -0
- data/lib/couchrest/validation.rb +245 -0
- data/lib/couchrest_extended_document.rb +21 -0
- data/spec/couchrest/attribute_protection_spec.rb +150 -0
- data/spec/couchrest/casted_extended_doc_spec.rb +79 -0
- data/spec/couchrest/casted_model_spec.rb +406 -0
- data/spec/couchrest/extended_doc_attachment_spec.rb +148 -0
- data/spec/couchrest/extended_doc_inherited_spec.rb +40 -0
- data/spec/couchrest/extended_doc_spec.rb +868 -0
- data/spec/couchrest/extended_doc_subclass_spec.rb +99 -0
- data/spec/couchrest/extended_doc_view_spec.rb +529 -0
- data/spec/couchrest/property_spec.rb +648 -0
- data/spec/fixtures/attachments/README +3 -0
- data/spec/fixtures/attachments/couchdb.png +0 -0
- data/spec/fixtures/attachments/test.html +11 -0
- data/spec/fixtures/more/article.rb +35 -0
- data/spec/fixtures/more/card.rb +22 -0
- data/spec/fixtures/more/cat.rb +22 -0
- data/spec/fixtures/more/course.rb +25 -0
- data/spec/fixtures/more/event.rb +8 -0
- data/spec/fixtures/more/invoice.rb +17 -0
- data/spec/fixtures/more/person.rb +9 -0
- data/spec/fixtures/more/question.rb +6 -0
- data/spec/fixtures/more/service.rb +12 -0
- data/spec/fixtures/more/user.rb +22 -0
- data/spec/fixtures/views/lib.js +3 -0
- data/spec/fixtures/views/test_view/lib.js +3 -0
- data/spec/fixtures/views/test_view/only-map.js +4 -0
- data/spec/fixtures/views/test_view/test-map.js +3 -0
- data/spec/fixtures/views/test_view/test-reduce.js +3 -0
- data/spec/spec.opts +5 -0
- data/spec/spec_helper.rb +49 -0
- data/utils/remap.rb +27 -0
- data/utils/subset.rb +30 -0
- metadata +200 -0
@@ -0,0 +1,66 @@
|
|
1
|
+
# encoding: binary
|
2
|
+
|
3
|
+
# Extracted from dm-validations 0.9.10
|
4
|
+
#
|
5
|
+
# Copyright (c) 2007 Guy van den Berg
|
6
|
+
#
|
7
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
8
|
+
# a copy of this software and associated documentation files (the
|
9
|
+
# "Software"), to deal in the Software without restriction, including
|
10
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
11
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
12
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
13
|
+
# the following conditions:
|
14
|
+
#
|
15
|
+
# The above copyright notice and this permission notice shall be
|
16
|
+
# included in all copies or substantial portions of the Software.
|
17
|
+
#
|
18
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
19
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
20
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
21
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
22
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
23
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
24
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
25
|
+
|
26
|
+
module CouchRest
|
27
|
+
module Validation
|
28
|
+
module Format
|
29
|
+
module Email
|
30
|
+
|
31
|
+
def self.included(base)
|
32
|
+
CouchRest::Validation::FormatValidator::FORMATS.merge!(
|
33
|
+
:email_address => [ EmailAddress, lambda { |field, value| '%s is not a valid email address'.t(value) }]
|
34
|
+
)
|
35
|
+
end
|
36
|
+
|
37
|
+
# RFC2822 (No attribution reference available)
|
38
|
+
EmailAddress = begin
|
39
|
+
alpha = "a-zA-Z"
|
40
|
+
digit = "0-9"
|
41
|
+
atext = "[#{alpha}#{digit}\!\#\$\%\&\'\*+\/\=\?\^\_\`\{\|\}\~\-]"
|
42
|
+
dot_atom_text = "#{atext}+([.]#{atext}*)*"
|
43
|
+
dot_atom = "#{dot_atom_text}"
|
44
|
+
qtext = '[^\\x0d\\x22\\x5c\\x80-\\xff]'
|
45
|
+
text = "[\\x01-\\x09\\x11\\x12\\x14-\\x7f]"
|
46
|
+
quoted_pair = "(\\x5c#{text})"
|
47
|
+
qcontent = "(?:#{qtext}|#{quoted_pair})"
|
48
|
+
quoted_string = "[\"]#{qcontent}+[\"]"
|
49
|
+
atom = "#{atext}+"
|
50
|
+
word = "(?:#{atom}|#{quoted_string})"
|
51
|
+
obs_local_part = "#{word}([.]#{word})*"
|
52
|
+
local_part = "(?:#{dot_atom}|#{quoted_string}|#{obs_local_part})"
|
53
|
+
no_ws_ctl = "\\x01-\\x08\\x11\\x12\\x14-\\x1f\\x7f"
|
54
|
+
dtext = "[#{no_ws_ctl}\\x21-\\x5a\\x5e-\\x7e]"
|
55
|
+
dcontent = "(?:#{dtext}|#{quoted_pair})"
|
56
|
+
domain_literal = "\\[#{dcontent}+\\]"
|
57
|
+
obs_domain = "#{atom}([.]#{atom})*"
|
58
|
+
domain = "(?:#{dot_atom}|#{domain_literal}|#{obs_domain})"
|
59
|
+
addr_spec = "#{local_part}\@#{domain}"
|
60
|
+
pattern = /^#{addr_spec}$/
|
61
|
+
end
|
62
|
+
|
63
|
+
end # module Email
|
64
|
+
end # module Format
|
65
|
+
end # module Validation
|
66
|
+
end # module CouchRest
|
@@ -0,0 +1,43 @@
|
|
1
|
+
# Extracted from dm-validations 0.9.10
|
2
|
+
#
|
3
|
+
# Copyright (c) 2007 Guy van den Berg
|
4
|
+
#
|
5
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
# a copy of this software and associated documentation files (the
|
7
|
+
# "Software"), to deal in the Software without restriction, including
|
8
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
# the following conditions:
|
12
|
+
#
|
13
|
+
# The above copyright notice and this permission notice shall be
|
14
|
+
# included in all copies or substantial portions of the Software.
|
15
|
+
#
|
16
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
+
|
24
|
+
module CouchRest
|
25
|
+
module Validation
|
26
|
+
module Format
|
27
|
+
module Url
|
28
|
+
|
29
|
+
def self.included(base)
|
30
|
+
CouchRest::Validation::FormatValidator::FORMATS.merge!(
|
31
|
+
:url => [ Url, lambda { |field, value| '%s is not a valid URL'.t(value) }]
|
32
|
+
)
|
33
|
+
end
|
34
|
+
|
35
|
+
Url = begin
|
36
|
+
# Regex from http://www.igvita.com/2006/09/07/validating-url-in-ruby-on-rails/
|
37
|
+
/(^$)|(^(http|https):\/\/[a-z0-9]+([\-\.]{1}[a-z0-9]+)*\.[a-z]{2,5}(([0-9]{1,5})?\/.*)?$)/ix
|
38
|
+
end
|
39
|
+
|
40
|
+
end # module Url
|
41
|
+
end # module Format
|
42
|
+
end # module Validation
|
43
|
+
end # module CouchRest
|
@@ -0,0 +1,120 @@
|
|
1
|
+
# -*- coding: utf-8 -*-
|
2
|
+
|
3
|
+
# Extracted from dm-validations 0.9.10
|
4
|
+
#
|
5
|
+
# Copyright (c) 2007 Guy van den Berg
|
6
|
+
#
|
7
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
8
|
+
# a copy of this software and associated documentation files (the
|
9
|
+
# "Software"), to deal in the Software without restriction, including
|
10
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
11
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
12
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
13
|
+
# the following conditions:
|
14
|
+
#
|
15
|
+
# The above copyright notice and this permission notice shall be
|
16
|
+
# included in all copies or substantial portions of the Software.
|
17
|
+
#
|
18
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
19
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
20
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
21
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
22
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
23
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
24
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
25
|
+
|
26
|
+
module CouchRest
|
27
|
+
module Validation
|
28
|
+
|
29
|
+
# All validators extend this base class. Validators must:
|
30
|
+
#
|
31
|
+
# * Implement the initialize method to capture its parameters, also calling
|
32
|
+
# super to have this parent class capture the optional, general :if and
|
33
|
+
# :unless parameters.
|
34
|
+
# * Implement the call method, returning true or false. The call method
|
35
|
+
# provides the validation logic.
|
36
|
+
#
|
37
|
+
# @author Guy van den Berg
|
38
|
+
class GenericValidator
|
39
|
+
|
40
|
+
attr_accessor :if_clause, :unless_clause
|
41
|
+
attr_reader :field_name
|
42
|
+
|
43
|
+
# Construct a validator. Capture the :if and :unless clauses when present.
|
44
|
+
#
|
45
|
+
# @param field<String, Symbol> The property specified for validation
|
46
|
+
#
|
47
|
+
# @option :if<Symbol, Proc> The name of a method or a Proc to call to
|
48
|
+
# determine if the validation should occur.
|
49
|
+
# @option :unless<Symbol, Proc> The name of a method or a Proc to call to
|
50
|
+
# determine if the validation should not occur
|
51
|
+
# All additional key/value pairs are passed through to the validator
|
52
|
+
# that is sub-classing this GenericValidator
|
53
|
+
#
|
54
|
+
def initialize(field, opts = {})
|
55
|
+
@if_clause = opts.delete(:if)
|
56
|
+
@unless_clause = opts.delete(:unless)
|
57
|
+
end
|
58
|
+
|
59
|
+
# Add an error message to a target resource. If the error corresponds to a
|
60
|
+
# specific field of the resource, add it to that field, otherwise add it
|
61
|
+
# as a :general message.
|
62
|
+
#
|
63
|
+
# @param <Object> target the resource that has the error
|
64
|
+
# @param <String> message the message to add
|
65
|
+
# @param <Symbol> field_name the name of the field that caused the error
|
66
|
+
#
|
67
|
+
# TODO - should the field_name for a general message be :default???
|
68
|
+
#
|
69
|
+
def add_error(target, message, field_name = :general)
|
70
|
+
target.errors.add(field_name, message)
|
71
|
+
end
|
72
|
+
|
73
|
+
# Call the validator. "call" is used so the operation is BoundMethod and
|
74
|
+
# Block compatible. This must be implemented in all concrete classes.
|
75
|
+
#
|
76
|
+
# @param <Object> target the resource that the validator must be called
|
77
|
+
# against
|
78
|
+
# @return <Boolean> true if valid, otherwise false
|
79
|
+
def call(target)
|
80
|
+
raise NotImplementedError, "CouchRest::Validation::GenericValidator::call must be overriden in a subclass"
|
81
|
+
end
|
82
|
+
|
83
|
+
# Determines if this validator should be run against the
|
84
|
+
# target by evaluating the :if and :unless clauses
|
85
|
+
# optionally passed while specifying any validator.
|
86
|
+
#
|
87
|
+
# @param <Object> target the resource that we check against
|
88
|
+
# @return <Boolean> true if should be run, otherwise false
|
89
|
+
def execute?(target)
|
90
|
+
if unless_clause = self.unless_clause
|
91
|
+
if unless_clause.is_a?(Symbol)
|
92
|
+
return false if target.send(unless_clause)
|
93
|
+
elsif unless_clause.respond_to?(:call)
|
94
|
+
return false if unless_clause.call(target)
|
95
|
+
end
|
96
|
+
end
|
97
|
+
|
98
|
+
if if_clause = self.if_clause
|
99
|
+
if if_clause.is_a?(Symbol)
|
100
|
+
return target.send(if_clause)
|
101
|
+
elsif if_clause.respond_to?(:call)
|
102
|
+
return if_clause.call(target)
|
103
|
+
end
|
104
|
+
end
|
105
|
+
|
106
|
+
true
|
107
|
+
end
|
108
|
+
|
109
|
+
def ==(other)
|
110
|
+
self.class == other.class &&
|
111
|
+
self.field_name == other.field_name &&
|
112
|
+
self.class == other.class &&
|
113
|
+
self.if_clause == other.if_clause &&
|
114
|
+
self.unless_clause == other.unless_clause &&
|
115
|
+
self.instance_variable_get(:@options) == other.instance_variable_get(:@options)
|
116
|
+
end
|
117
|
+
|
118
|
+
end # class GenericValidator
|
119
|
+
end # module Validation
|
120
|
+
end # module CouchRest
|
@@ -0,0 +1,139 @@
|
|
1
|
+
# Extracted from dm-validations 0.9.10
|
2
|
+
#
|
3
|
+
# Copyright (c) 2007 Guy van den Berg
|
4
|
+
#
|
5
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
# a copy of this software and associated documentation files (the
|
7
|
+
# "Software"), to deal in the Software without restriction, including
|
8
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
# the following conditions:
|
12
|
+
#
|
13
|
+
# The above copyright notice and this permission notice shall be
|
14
|
+
# included in all copies or substantial portions of the Software.
|
15
|
+
#
|
16
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
+
|
24
|
+
module CouchRest
|
25
|
+
module Validation
|
26
|
+
|
27
|
+
##
|
28
|
+
#
|
29
|
+
# @author Guy van den Berg
|
30
|
+
# @since 0.9
|
31
|
+
class LengthValidator < GenericValidator
|
32
|
+
|
33
|
+
def initialize(field_name, options)
|
34
|
+
super
|
35
|
+
@field_name = field_name
|
36
|
+
@options = options
|
37
|
+
|
38
|
+
@min = options[:minimum] || options[:min]
|
39
|
+
@max = options[:maximum] || options[:max]
|
40
|
+
@equal = options[:is] || options[:equals]
|
41
|
+
@range = options[:within] || options[:in]
|
42
|
+
|
43
|
+
@validation_method ||= :range if @range
|
44
|
+
@validation_method ||= :min if @min && @max.nil?
|
45
|
+
@validation_method ||= :max if @max && @min.nil?
|
46
|
+
@validation_method ||= :equals unless @equal.nil?
|
47
|
+
end
|
48
|
+
|
49
|
+
def call(target)
|
50
|
+
field_value = target.validation_property_value(field_name)
|
51
|
+
return true if @options[:allow_nil] && field_value.nil?
|
52
|
+
|
53
|
+
field_value = '' if field_value.nil?
|
54
|
+
|
55
|
+
# XXX: HACK seems hacky to do this on every validation, probably should
|
56
|
+
# do this elsewhere?
|
57
|
+
field = field_name.to_s.humanize
|
58
|
+
min = @range ? @range.min : @min
|
59
|
+
max = @range ? @range.max : @max
|
60
|
+
equal = @equal
|
61
|
+
|
62
|
+
case @validation_method
|
63
|
+
when :range then
|
64
|
+
unless valid = @range.include?(field_value.size)
|
65
|
+
error_message = ValidationErrors.default_error_message(:length_between, field, min, max)
|
66
|
+
end
|
67
|
+
when :min then
|
68
|
+
unless valid = field_value.size >= min
|
69
|
+
error_message = ValidationErrors.default_error_message(:too_short, field, min)
|
70
|
+
end
|
71
|
+
when :max then
|
72
|
+
unless valid = field_value.size <= max
|
73
|
+
error_message = ValidationErrors.default_error_message(:too_long, field, max)
|
74
|
+
end
|
75
|
+
when :equals then
|
76
|
+
unless valid = field_value.size == equal
|
77
|
+
error_message = ValidationErrors.default_error_message(:wrong_length, field, equal)
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
81
|
+
error_message = @options[:message] || error_message
|
82
|
+
|
83
|
+
add_error(target, error_message, field_name) unless valid
|
84
|
+
|
85
|
+
return valid
|
86
|
+
end
|
87
|
+
|
88
|
+
end # class LengthValidator
|
89
|
+
|
90
|
+
module ValidatesLength
|
91
|
+
|
92
|
+
# Validates that the length of the attribute is equal to, less than,
|
93
|
+
# greater than or within a certain range (depending upon the options
|
94
|
+
# you specify).
|
95
|
+
#
|
96
|
+
# @option :allow_nil<Boolean> true/false (default is true)
|
97
|
+
# @option :minimum ensures that the attribute's length is greater than
|
98
|
+
# or equal to the supplied value
|
99
|
+
# @option :min alias for :minimum
|
100
|
+
# @option :maximum ensures the attribute's length is less than or equal
|
101
|
+
# to the supplied value
|
102
|
+
# @option :max alias for :maximum
|
103
|
+
# @option :equals ensures the attribute's length is equal to the
|
104
|
+
# supplied value
|
105
|
+
# @option :is alias for :equals
|
106
|
+
# @option :in<Range> given a Range, ensures that the attributes length is
|
107
|
+
# include?'ed in the Range
|
108
|
+
# @option :within<Range> alias for :in
|
109
|
+
#
|
110
|
+
# @example [Usage]
|
111
|
+
#
|
112
|
+
# class Page
|
113
|
+
#
|
114
|
+
# property high, Integer
|
115
|
+
# property low, Integer
|
116
|
+
# property just_right, Integer
|
117
|
+
#
|
118
|
+
# validates_length_of :high, :min => 100000000000
|
119
|
+
# validates_length_of :low, :equals => 0
|
120
|
+
# validates_length_of :just_right, :within => 1..10
|
121
|
+
#
|
122
|
+
# # a call to valid? will return false unless:
|
123
|
+
# # high is greater than or equal to 100000000000
|
124
|
+
# # low is equal to 0
|
125
|
+
# # just_right is between 1 and 10 (inclusive of both 1 and 10)
|
126
|
+
#
|
127
|
+
def validates_length_of(*fields)
|
128
|
+
opts = opts_from_validator_args(fields)
|
129
|
+
add_validator_to_context(opts, fields, CouchRest::Validation::LengthValidator)
|
130
|
+
end
|
131
|
+
|
132
|
+
def validates_length(*fields)
|
133
|
+
warn "[DEPRECATION] `validates_length` is deprecated. Please use `validates_length_of` instead."
|
134
|
+
validates_length_of(*fields)
|
135
|
+
end
|
136
|
+
|
137
|
+
end # module ValidatesLength
|
138
|
+
end # module Validation
|
139
|
+
end # module CouchRest
|
@@ -0,0 +1,89 @@
|
|
1
|
+
# Extracted from dm-validations 0.9.10
|
2
|
+
#
|
3
|
+
# Copyright (c) 2007 Guy van den Berg
|
4
|
+
#
|
5
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
# a copy of this software and associated documentation files (the
|
7
|
+
# "Software"), to deal in the Software without restriction, including
|
8
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
# the following conditions:
|
12
|
+
#
|
13
|
+
# The above copyright notice and this permission notice shall be
|
14
|
+
# included in all copies or substantial portions of the Software.
|
15
|
+
#
|
16
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
+
|
24
|
+
module CouchRest
|
25
|
+
module Validation
|
26
|
+
|
27
|
+
##
|
28
|
+
#
|
29
|
+
# @author Guy van den Berg
|
30
|
+
# @since 0.9
|
31
|
+
class MethodValidator < GenericValidator
|
32
|
+
|
33
|
+
def initialize(field_name, options={})
|
34
|
+
super
|
35
|
+
@field_name, @options = field_name, options.clone
|
36
|
+
@options[:method] = @field_name unless @options.has_key?(:method)
|
37
|
+
end
|
38
|
+
|
39
|
+
def call(target)
|
40
|
+
result, message = target.send(@options[:method])
|
41
|
+
add_error(target, message, field_name) unless result
|
42
|
+
result
|
43
|
+
end
|
44
|
+
|
45
|
+
def ==(other)
|
46
|
+
@options[:method] == other.instance_variable_get(:@options)[:method] && super
|
47
|
+
end
|
48
|
+
end # class MethodValidator
|
49
|
+
|
50
|
+
module ValidatesWithMethod
|
51
|
+
|
52
|
+
##
|
53
|
+
# Validate using the given method. The method given needs to return:
|
54
|
+
# [result::<Boolean>, Error Message::<String>]
|
55
|
+
#
|
56
|
+
# @example [Usage]
|
57
|
+
#
|
58
|
+
# class Page
|
59
|
+
#
|
60
|
+
# property :zip_code, String
|
61
|
+
#
|
62
|
+
# validates_with_method :in_the_right_location?
|
63
|
+
#
|
64
|
+
# def in_the_right_location?
|
65
|
+
# if @zip_code == "94301"
|
66
|
+
# return true
|
67
|
+
# else
|
68
|
+
# return [false, "You're in the wrong zip code"]
|
69
|
+
# end
|
70
|
+
# end
|
71
|
+
#
|
72
|
+
# # A call to valid? will return false and
|
73
|
+
# # populate the object's errors with "You're in the
|
74
|
+
# # wrong zip code" unless zip_code == "94301"
|
75
|
+
#
|
76
|
+
# # You can also specify field:
|
77
|
+
#
|
78
|
+
# validates_with_method :zip_code, :in_the_right_location?
|
79
|
+
#
|
80
|
+
# # it will add returned error message to :zip_code field
|
81
|
+
#
|
82
|
+
def validates_with_method(*fields)
|
83
|
+
opts = opts_from_validator_args(fields)
|
84
|
+
add_validator_to_context(opts, fields, CouchRest::Validation::MethodValidator)
|
85
|
+
end
|
86
|
+
|
87
|
+
end # module ValidatesWithMethod
|
88
|
+
end # module Validation
|
89
|
+
end # module CouchRest
|
@@ -0,0 +1,109 @@
|
|
1
|
+
# Extracted from dm-validations 0.9.10
|
2
|
+
#
|
3
|
+
# Copyright (c) 2007 Guy van den Berg
|
4
|
+
#
|
5
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
# a copy of this software and associated documentation files (the
|
7
|
+
# "Software"), to deal in the Software without restriction, including
|
8
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
# the following conditions:
|
12
|
+
#
|
13
|
+
# The above copyright notice and this permission notice shall be
|
14
|
+
# included in all copies or substantial portions of the Software.
|
15
|
+
#
|
16
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
+
|
24
|
+
module CouchRest
|
25
|
+
module Validation
|
26
|
+
|
27
|
+
##
|
28
|
+
#
|
29
|
+
# @author Guy van den Berg
|
30
|
+
# @since 0.9
|
31
|
+
class NumericValidator < GenericValidator
|
32
|
+
|
33
|
+
def initialize(field_name, options={})
|
34
|
+
super
|
35
|
+
@field_name, @options = field_name, options
|
36
|
+
@options[:integer_only] = false unless @options.has_key?(:integer_only)
|
37
|
+
end
|
38
|
+
|
39
|
+
def call(target)
|
40
|
+
value = target.send(field_name)
|
41
|
+
return true if @options[:allow_nil] && value.nil?
|
42
|
+
|
43
|
+
value = (defined?(BigDecimal) && value.kind_of?(BigDecimal)) ? value.to_s('F') : value.to_s
|
44
|
+
|
45
|
+
error_message = @options[:message]
|
46
|
+
precision = @options[:precision]
|
47
|
+
scale = @options[:scale]
|
48
|
+
|
49
|
+
if @options[:integer_only]
|
50
|
+
return true if value =~ /\A[+-]?\d+\z/
|
51
|
+
error_message ||= ValidationErrors.default_error_message(:not_an_integer, field_name)
|
52
|
+
else
|
53
|
+
# FIXME: if precision and scale are not specified, can we assume that it is an integer?
|
54
|
+
# probably not, as floating point numbers don't have hard
|
55
|
+
# defined scale. the scale floats with the length of the
|
56
|
+
# integral and precision. Ie. if precision = 10 and integral
|
57
|
+
# portion of the number is 9834 (4 digits), the max scale will
|
58
|
+
# be 6 (10 - 4). But if the integral length is 1, max scale
|
59
|
+
# will be (10 - 1) = 9, so 1.234567890.
|
60
|
+
if precision && scale
|
61
|
+
#handles both Float when it has scale specified and BigDecimal
|
62
|
+
if precision > scale && scale > 0
|
63
|
+
return true if value =~ /\A[+-]?(?:\d{1,#{precision - scale}}|\d{0,#{precision - scale}}\.\d{1,#{scale}})\z/
|
64
|
+
elsif precision > scale && scale == 0
|
65
|
+
return true if value =~ /\A[+-]?(?:\d{1,#{precision}}(?:\.0)?)\z/
|
66
|
+
elsif precision == scale
|
67
|
+
return true if value =~ /\A[+-]?(?:0(?:\.\d{1,#{scale}})?)\z/
|
68
|
+
else
|
69
|
+
raise ArgumentError, "Invalid precision #{precision.inspect} and scale #{scale.inspect} for #{field_name} (value: #{value.inspect} #{value.class})"
|
70
|
+
end
|
71
|
+
elsif precision && scale.nil?
|
72
|
+
# for floats, if scale is not set
|
73
|
+
|
74
|
+
#total number of digits is less or equal precision
|
75
|
+
return true if value.gsub(/[^\d]/, '').length <= precision
|
76
|
+
|
77
|
+
#number of digits before decimal == precision, and the number is x.0. same as scale = 0
|
78
|
+
return true if value =~ /\A[+-]?(?:\d{1,#{precision}}(?:\.0)?)\z/
|
79
|
+
else
|
80
|
+
return true if value =~ /\A[+-]?(?:\d+|\d*\.\d+)\z/
|
81
|
+
end
|
82
|
+
error_message ||= ValidationErrors.default_error_message(:not_a_number, field_name)
|
83
|
+
end
|
84
|
+
|
85
|
+
add_error(target, error_message, field_name)
|
86
|
+
|
87
|
+
# TODO: check the gt, gte, lt, lte, and eq options
|
88
|
+
|
89
|
+
return false
|
90
|
+
end
|
91
|
+
end # class NumericValidator
|
92
|
+
|
93
|
+
module ValidatesIsNumber
|
94
|
+
|
95
|
+
# Validate whether a field is numeric
|
96
|
+
#
|
97
|
+
def validates_numericality_of(*fields)
|
98
|
+
opts = opts_from_validator_args(fields)
|
99
|
+
add_validator_to_context(opts, fields, CouchRest::Validation::NumericValidator)
|
100
|
+
end
|
101
|
+
|
102
|
+
def validates_is_number(*fields)
|
103
|
+
warn "[DEPRECATION] `validates_is_number` is deprecated. Please use `validates_numericality_of` instead."
|
104
|
+
validates_numericality_of(*fields)
|
105
|
+
end
|
106
|
+
|
107
|
+
end # module ValidatesIsNumber
|
108
|
+
end # module Validation
|
109
|
+
end # module CouchRest
|
@@ -0,0 +1,114 @@
|
|
1
|
+
# Extracted from dm-validations 0.9.10
|
2
|
+
#
|
3
|
+
# Copyright (c) 2007 Guy van den Berg
|
4
|
+
#
|
5
|
+
# Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
# a copy of this software and associated documentation files (the
|
7
|
+
# "Software"), to deal in the Software without restriction, including
|
8
|
+
# without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
# distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
# permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
# the following conditions:
|
12
|
+
#
|
13
|
+
# The above copyright notice and this permission notice shall be
|
14
|
+
# included in all copies or substantial portions of the Software.
|
15
|
+
#
|
16
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
23
|
+
|
24
|
+
module CouchRest
|
25
|
+
module Validation
|
26
|
+
|
27
|
+
##
|
28
|
+
#
|
29
|
+
# @author Guy van den Berg
|
30
|
+
# @since 0.9
|
31
|
+
class RequiredFieldValidator < GenericValidator
|
32
|
+
|
33
|
+
def initialize(field_name, options={})
|
34
|
+
super
|
35
|
+
@field_name, @options = field_name, options
|
36
|
+
end
|
37
|
+
|
38
|
+
def call(target)
|
39
|
+
value = target.validation_property_value(field_name)
|
40
|
+
property = target.validation_property(field_name.to_s)
|
41
|
+
return true if present?(value, property)
|
42
|
+
|
43
|
+
error_message = @options[:message] || default_error(property)
|
44
|
+
add_error(target, error_message, field_name)
|
45
|
+
|
46
|
+
false
|
47
|
+
end
|
48
|
+
|
49
|
+
protected
|
50
|
+
|
51
|
+
# Boolean property types are considered present if non-nil.
|
52
|
+
# Other property types are considered present if non-blank.
|
53
|
+
# Non-properties are considered present if non-blank.
|
54
|
+
def present?(value, property)
|
55
|
+
boolean_type?(property) ? !value.nil? : !value.blank?
|
56
|
+
end
|
57
|
+
|
58
|
+
def default_error(property)
|
59
|
+
actual = boolean_type?(property) ? :nil : :blank
|
60
|
+
ValidationErrors.default_error_message(actual, field_name)
|
61
|
+
end
|
62
|
+
|
63
|
+
# Is +property+ a boolean property?
|
64
|
+
#
|
65
|
+
# Returns true for Boolean, ParanoidBoolean, TrueClass, etc. properties.
|
66
|
+
# Returns false for other property types.
|
67
|
+
# Returns false for non-properties.
|
68
|
+
def boolean_type?(property)
|
69
|
+
property ? property.type == 'Boolean' : false
|
70
|
+
end
|
71
|
+
|
72
|
+
end # class RequiredFieldValidator
|
73
|
+
|
74
|
+
module ValidatesPresent
|
75
|
+
|
76
|
+
##
|
77
|
+
# Validates that the specified attribute is present.
|
78
|
+
#
|
79
|
+
# For most property types "being present" is the same as being "not
|
80
|
+
# blank" as determined by the attribute's #blank? method. However, in
|
81
|
+
# the case of Boolean, "being present" means not nil; i.e. true or
|
82
|
+
# false.
|
83
|
+
#
|
84
|
+
# @note
|
85
|
+
# dm-core's support lib adds the blank? method to many classes,
|
86
|
+
# @see lib/dm-core/support/blank.rb (dm-core) for more information.
|
87
|
+
#
|
88
|
+
# @example [Usage]
|
89
|
+
#
|
90
|
+
# class Page
|
91
|
+
#
|
92
|
+
# property :required_attribute, String
|
93
|
+
# property :another_required, String
|
94
|
+
# property :yet_again, String
|
95
|
+
#
|
96
|
+
# validates_presence_of :required_attribute
|
97
|
+
# validates_presence_of :another_required, :yet_again
|
98
|
+
#
|
99
|
+
# # a call to valid? will return false unless
|
100
|
+
# # all three attributes are !blank?
|
101
|
+
# end
|
102
|
+
def validates_presence_of(*fields)
|
103
|
+
opts = opts_from_validator_args(fields)
|
104
|
+
add_validator_to_context(opts, fields, CouchRest::Validation::RequiredFieldValidator)
|
105
|
+
end
|
106
|
+
|
107
|
+
def validates_present(*fields)
|
108
|
+
warn "[DEPRECATION] `validates_present` is deprecated. Please use `validates_presence_of` instead."
|
109
|
+
validates_presence_of(*fields)
|
110
|
+
end
|
111
|
+
|
112
|
+
end # module ValidatesPresent
|
113
|
+
end # module Validation
|
114
|
+
end # module CouchRest
|