mongoid-with-auth 1.9.4

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.
Files changed (104) hide show
  1. data/MIT_LICENSE +20 -0
  2. data/README.rdoc +49 -0
  3. data/lib/mongoid.rb +122 -0
  4. data/lib/mongoid/associations.rb +300 -0
  5. data/lib/mongoid/associations/belongs_to_related.rb +58 -0
  6. data/lib/mongoid/associations/embedded_in.rb +72 -0
  7. data/lib/mongoid/associations/embeds_many.rb +254 -0
  8. data/lib/mongoid/associations/embeds_one.rb +96 -0
  9. data/lib/mongoid/associations/has_many_related.rb +181 -0
  10. data/lib/mongoid/associations/has_one_related.rb +85 -0
  11. data/lib/mongoid/associations/meta_data.rb +29 -0
  12. data/lib/mongoid/associations/options.rb +57 -0
  13. data/lib/mongoid/associations/proxy.rb +24 -0
  14. data/lib/mongoid/attributes.rb +204 -0
  15. data/lib/mongoid/callbacks.rb +23 -0
  16. data/lib/mongoid/collection.rb +120 -0
  17. data/lib/mongoid/collections.rb +41 -0
  18. data/lib/mongoid/collections/cyclic_iterator.rb +34 -0
  19. data/lib/mongoid/collections/master.rb +29 -0
  20. data/lib/mongoid/collections/operations.rb +41 -0
  21. data/lib/mongoid/collections/slaves.rb +45 -0
  22. data/lib/mongoid/components.rb +27 -0
  23. data/lib/mongoid/concern.rb +31 -0
  24. data/lib/mongoid/config.rb +205 -0
  25. data/lib/mongoid/contexts.rb +25 -0
  26. data/lib/mongoid/contexts/enumerable.rb +151 -0
  27. data/lib/mongoid/contexts/ids.rb +25 -0
  28. data/lib/mongoid/contexts/mongo.rb +285 -0
  29. data/lib/mongoid/contexts/paging.rb +50 -0
  30. data/lib/mongoid/criteria.rb +239 -0
  31. data/lib/mongoid/criterion/complex.rb +21 -0
  32. data/lib/mongoid/criterion/exclusion.rb +65 -0
  33. data/lib/mongoid/criterion/inclusion.rb +110 -0
  34. data/lib/mongoid/criterion/optional.rb +136 -0
  35. data/lib/mongoid/cursor.rb +81 -0
  36. data/lib/mongoid/deprecation.rb +22 -0
  37. data/lib/mongoid/dirty.rb +253 -0
  38. data/lib/mongoid/document.rb +311 -0
  39. data/lib/mongoid/errors.rb +108 -0
  40. data/lib/mongoid/extensions.rb +101 -0
  41. data/lib/mongoid/extensions/array/accessors.rb +17 -0
  42. data/lib/mongoid/extensions/array/aliasing.rb +4 -0
  43. data/lib/mongoid/extensions/array/assimilation.rb +26 -0
  44. data/lib/mongoid/extensions/array/conversions.rb +29 -0
  45. data/lib/mongoid/extensions/array/parentization.rb +13 -0
  46. data/lib/mongoid/extensions/big_decimal/conversions.rb +19 -0
  47. data/lib/mongoid/extensions/binary/conversions.rb +17 -0
  48. data/lib/mongoid/extensions/boolean/conversions.rb +22 -0
  49. data/lib/mongoid/extensions/date/conversions.rb +24 -0
  50. data/lib/mongoid/extensions/datetime/conversions.rb +12 -0
  51. data/lib/mongoid/extensions/float/conversions.rb +20 -0
  52. data/lib/mongoid/extensions/hash/accessors.rb +38 -0
  53. data/lib/mongoid/extensions/hash/assimilation.rb +39 -0
  54. data/lib/mongoid/extensions/hash/conversions.rb +45 -0
  55. data/lib/mongoid/extensions/hash/criteria_helpers.rb +20 -0
  56. data/lib/mongoid/extensions/hash/scoping.rb +12 -0
  57. data/lib/mongoid/extensions/integer/conversions.rb +20 -0
  58. data/lib/mongoid/extensions/nil/assimilation.rb +17 -0
  59. data/lib/mongoid/extensions/object/conversions.rb +33 -0
  60. data/lib/mongoid/extensions/objectid/conversions.rb +15 -0
  61. data/lib/mongoid/extensions/proc/scoping.rb +12 -0
  62. data/lib/mongoid/extensions/string/conversions.rb +15 -0
  63. data/lib/mongoid/extensions/string/inflections.rb +97 -0
  64. data/lib/mongoid/extensions/symbol/inflections.rb +36 -0
  65. data/lib/mongoid/extensions/time_conversions.rb +35 -0
  66. data/lib/mongoid/extras.rb +61 -0
  67. data/lib/mongoid/factory.rb +20 -0
  68. data/lib/mongoid/field.rb +59 -0
  69. data/lib/mongoid/fields.rb +65 -0
  70. data/lib/mongoid/finders.rb +136 -0
  71. data/lib/mongoid/identity.rb +39 -0
  72. data/lib/mongoid/indexes.rb +30 -0
  73. data/lib/mongoid/javascript.rb +21 -0
  74. data/lib/mongoid/javascript/functions.yml +37 -0
  75. data/lib/mongoid/matchers.rb +36 -0
  76. data/lib/mongoid/matchers/all.rb +11 -0
  77. data/lib/mongoid/matchers/default.rb +26 -0
  78. data/lib/mongoid/matchers/exists.rb +13 -0
  79. data/lib/mongoid/matchers/gt.rb +11 -0
  80. data/lib/mongoid/matchers/gte.rb +11 -0
  81. data/lib/mongoid/matchers/in.rb +11 -0
  82. data/lib/mongoid/matchers/lt.rb +11 -0
  83. data/lib/mongoid/matchers/lte.rb +11 -0
  84. data/lib/mongoid/matchers/ne.rb +11 -0
  85. data/lib/mongoid/matchers/nin.rb +11 -0
  86. data/lib/mongoid/matchers/size.rb +11 -0
  87. data/lib/mongoid/memoization.rb +33 -0
  88. data/lib/mongoid/named_scope.rb +37 -0
  89. data/lib/mongoid/observable.rb +30 -0
  90. data/lib/mongoid/paths.rb +62 -0
  91. data/lib/mongoid/persistence.rb +222 -0
  92. data/lib/mongoid/persistence/command.rb +39 -0
  93. data/lib/mongoid/persistence/insert.rb +50 -0
  94. data/lib/mongoid/persistence/insert_embedded.rb +38 -0
  95. data/lib/mongoid/persistence/remove.rb +39 -0
  96. data/lib/mongoid/persistence/remove_all.rb +37 -0
  97. data/lib/mongoid/persistence/remove_embedded.rb +50 -0
  98. data/lib/mongoid/persistence/update.rb +63 -0
  99. data/lib/mongoid/scope.rb +75 -0
  100. data/lib/mongoid/state.rb +39 -0
  101. data/lib/mongoid/timestamps.rb +27 -0
  102. data/lib/mongoid/version.rb +4 -0
  103. data/lib/mongoid/versioning.rb +27 -0
  104. metadata +284 -0
@@ -0,0 +1,50 @@
1
+ # encoding: utf-8
2
+ module Mongoid #:nodoc:
3
+ module Contexts #:nodoc:
4
+ module Paging
5
+ # Paginates the documents.
6
+ #
7
+ # Example:
8
+ #
9
+ # <tt>context.paginate(:page => 6, :per_page => 25)</tt>
10
+ #
11
+ # Returns:
12
+ #
13
+ # A collection of documents paginated.
14
+ # All previous <tt>limit</tt> and <tt>skip</tt> call will be ignored.
15
+ def paginate(pager_options={})
16
+ if pager_options[:per_page]
17
+ options[:limit] = pager_options[:per_page].to_i
18
+ if pager_options[:page]
19
+ options[:skip] = (pager_options[:page].to_i - 1) * pager_options[:per_page].to_i
20
+ end
21
+ end
22
+
23
+ @collection ||= execute(true)
24
+ WillPaginate::Collection.create(page, per_page, count) do |pager|
25
+ pager.replace(@collection.to_a)
26
+ end
27
+ end
28
+
29
+ # Either returns the page option and removes it from the options, or
30
+ # returns a default value of 1.
31
+ #
32
+ # Returns:
33
+ #
34
+ # An +Integer+ page number.
35
+ def page
36
+ skips, limits = options[:skip], options[:limit]
37
+ (skips && limits) ? (skips + limits) / limits : 1
38
+ end
39
+
40
+ # Get the number of results per page or the default of 20.
41
+ #
42
+ # Returns:
43
+ #
44
+ # The +Integer+ number of documents in each page.
45
+ def per_page
46
+ (options[:limit] || 20).to_i
47
+ end
48
+ end
49
+ end
50
+ end
@@ -0,0 +1,239 @@
1
+ # encoding: utf-8
2
+ require "mongoid/criterion/complex"
3
+ require "mongoid/criterion/exclusion"
4
+ require "mongoid/criterion/inclusion"
5
+ require "mongoid/criterion/optional"
6
+
7
+ module Mongoid #:nodoc:
8
+ # The +Criteria+ class is the core object needed in Mongoid to retrieve
9
+ # objects from the database. It is a DSL that essentially sets up the
10
+ # selector and options arguments that get passed on to a <tt>Mongo::Collection</tt>
11
+ # in the Ruby driver. Each method on the +Criteria+ returns self to they
12
+ # can be chained in order to create a readable criterion to be executed
13
+ # against the database.
14
+ #
15
+ # Example setup:
16
+ #
17
+ # <tt>criteria = Criteria.new</tt>
18
+ #
19
+ # <tt>criteria.only(:field).where(:field => "value").skip(20).limit(20)</tt>
20
+ #
21
+ # <tt>criteria.execute</tt>
22
+ class Criteria
23
+ include Criterion::Exclusion
24
+ include Criterion::Inclusion
25
+ include Criterion::Optional
26
+ include Enumerable
27
+
28
+ attr_reader :collection, :ids, :klass, :options, :selector
29
+ attr_accessor :documents
30
+
31
+ delegate \
32
+ :aggregate,
33
+ :avg,
34
+ :blank?,
35
+ :count,
36
+ :distinct,
37
+ :empty?,
38
+ :execute,
39
+ :first,
40
+ :group,
41
+ :id_criteria,
42
+ :last,
43
+ :max,
44
+ :min,
45
+ :one,
46
+ :page,
47
+ :paginate,
48
+ :per_page,
49
+ :sum, :to => :context
50
+
51
+ # Concatinate the criteria with another enumerable. If the other is a
52
+ # +Criteria+ then it needs to get the collection from it.
53
+ def +(other)
54
+ entries + comparable(other)
55
+ end
56
+
57
+ # Returns the difference between the criteria and another enumerable. If
58
+ # the other is a +Criteria+ then it needs to get the collection from it.
59
+ def -(other)
60
+ entries - comparable(other)
61
+ end
62
+
63
+ # Returns true if the supplied +Enumerable+ or +Criteria+ is equal to the results
64
+ # of this +Criteria+ or the criteria itself.
65
+ #
66
+ # This will force a database load when called if an enumerable is passed.
67
+ #
68
+ # Options:
69
+ #
70
+ # other: The other +Enumerable+ or +Criteria+ to compare to.
71
+ def ==(other)
72
+ case other
73
+ when Criteria
74
+ self.selector == other.selector && self.options == other.options
75
+ when Enumerable
76
+ return (execute.entries == other)
77
+ else
78
+ return false
79
+ end
80
+ end
81
+
82
+ # Return or create the context in which this criteria should be executed.
83
+ #
84
+ # This will return an Enumerable context if the class is embedded,
85
+ # otherwise it will return a Mongo context for root classes.
86
+ def context
87
+ @context ||= Contexts.context_for(self)
88
+ end
89
+
90
+ # Iterate over each +Document+ in the results. This can take an optional
91
+ # block to pass to each argument in the results.
92
+ #
93
+ # Example:
94
+ #
95
+ # <tt>criteria.each { |doc| p doc }</tt>
96
+ def each(&block)
97
+ context.iterate(&block)
98
+ self
99
+ end
100
+
101
+ # Merges the supplied argument hash into a single criteria
102
+ #
103
+ # Options:
104
+ #
105
+ # criteria_conditions: Hash of criteria keys, and parameter values
106
+ #
107
+ # Example:
108
+ #
109
+ # <tt>criteria.fuse(:where => { :field => "value"}, :limit => 20)</tt>
110
+ #
111
+ # Returns <tt>self</tt>
112
+ def fuse(criteria_conditions = {})
113
+ criteria_conditions.inject(self) do |criteria, (key, value)|
114
+ criteria.send(key, value)
115
+ end
116
+ end
117
+
118
+ # Create the new +Criteria+ object. This will initialize the selector
119
+ # and options hashes, as well as the type of criteria.
120
+ #
121
+ # Options:
122
+ #
123
+ # type: One of :all, :first:, or :last
124
+ # klass: The class to execute on.
125
+ def initialize(klass)
126
+ @selector, @options, @klass, @documents = {}, {}, klass, []
127
+ end
128
+
129
+ # Merges another object into this +Criteria+. The other object may be a
130
+ # +Criteria+ or a +Hash+. This is used to combine multiple scopes together,
131
+ # where a chained scope situation may be desired.
132
+ #
133
+ # Options:
134
+ #
135
+ # other: The +Criteria+ or +Hash+ to merge with.
136
+ #
137
+ # Example:
138
+ #
139
+ # <tt>criteria.merge({ :conditions => { :title => "Sir" } })</tt>
140
+ def merge(other)
141
+ @selector.update(other.selector)
142
+ @options.update(other.options)
143
+ @documents = other.documents
144
+ end
145
+
146
+ # Used for chaining +Criteria+ scopes together in the for of class methods
147
+ # on the +Document+ the criteria is for.
148
+ #
149
+ # Options:
150
+ #
151
+ # name: The name of the class method on the +Document+ to chain.
152
+ # args: The arguments passed to the method.
153
+ #
154
+ # Returns: <tt>Criteria</tt>
155
+ def method_missing(name, *args)
156
+ if @klass.respond_to?(name)
157
+ new_scope = @klass.send(name, *args)
158
+ new_scope.merge(self)
159
+ return new_scope
160
+ else
161
+ return entries.send(name, *args)
162
+ end
163
+ end
164
+
165
+ alias :to_ary :to_a
166
+
167
+ # Returns the selector and options as a +Hash+ that would be passed to a
168
+ # scope for use with named scopes.
169
+ def scoped
170
+ { :where => @selector }.merge(@options)
171
+ end
172
+
173
+ # Translate the supplied arguments into a +Criteria+ object.
174
+ #
175
+ # If the passed in args is a single +String+, then it will
176
+ # construct an id +Criteria+ from it.
177
+ #
178
+ # If the passed in args are a type and a hash, then it will construct
179
+ # the +Criteria+ with the proper selector, options, and type.
180
+ #
181
+ # Options:
182
+ #
183
+ # args: either a +String+ or a +Symbol+, +Hash combination.
184
+ #
185
+ # Example:
186
+ #
187
+ # <tt>Criteria.translate(Person, "4ab2bc4b8ad548971900005c")</tt>
188
+ # <tt>Criteria.translate(Person, :conditions => { :field => "value"}, :limit => 20)</tt>
189
+ def self.translate(*args)
190
+ klass = args[0]
191
+ params = args[1] || {}
192
+ unless params.is_a?(Hash)
193
+ return new(klass).id_criteria(params)
194
+ end
195
+ conditions = params.delete(:conditions) || {}
196
+ if conditions.include?(:id)
197
+ conditions[:_id] = conditions[:id]
198
+ conditions.delete(:id)
199
+ end
200
+ return new(klass).where(conditions).extras(params)
201
+ end
202
+
203
+ protected
204
+
205
+ # Filters the unused options out of the options +Hash+. Currently this
206
+ # takes into account the "page" and "per_page" options that would be passed
207
+ # in if using will_paginate.
208
+ #
209
+ # Example:
210
+ #
211
+ # Given a criteria with a selector of { :page => 1, :per_page => 40 }
212
+ #
213
+ # <tt>criteria.filter_options</tt> # selector: { :skip => 0, :limit => 40 }
214
+ def filter_options
215
+ page_num = @options.delete(:page)
216
+ per_page_num = @options.delete(:per_page)
217
+ if (page_num || per_page_num)
218
+ @options[:limit] = limits = (per_page_num || 20).to_i
219
+ @options[:skip] = (page_num || 1).to_i * limits - limits
220
+ end
221
+ end
222
+
223
+ # Return the entries of the other criteria or the object. Used for
224
+ # comparing criteria or an enumerable.
225
+ def comparable(other)
226
+ other.is_a?(Criteria) ? other.entries : other
227
+ end
228
+
229
+ # Update the selector setting the operator on the value for each key in the
230
+ # supplied attributes +Hash+.
231
+ #
232
+ # Example:
233
+ #
234
+ # <tt>criteria.update_selector({ :field => "value" }, "$in")</tt>
235
+ def update_selector(attributes, operator)
236
+ attributes.each { |key, value| @selector[key] = { operator => value } }; self
237
+ end
238
+ end
239
+ end
@@ -0,0 +1,21 @@
1
+ # encoding: utf-8
2
+ module Mongoid #:nodoc:
3
+ module Criterion #:nodoc:
4
+ # Complex criterion are used when performing operations on symbols to get
5
+ # get a shorthand syntax for where clauses.
6
+ #
7
+ # Example:
8
+ #
9
+ # <tt>{ :field => { "$lt" => "value" } }</tt>
10
+ # becomes:
11
+ # <tt> { :field.lt => "value }</tt>
12
+ class Complex
13
+ attr_accessor :key, :operator
14
+
15
+ # Create the new complex criterion.
16
+ def initialize(opts = {})
17
+ @key, @operator = opts[:key], opts[:operator]
18
+ end
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,65 @@
1
+ # encoding: utf-8
2
+ module Mongoid #:nodoc:
3
+ module Criterion #:nodoc:
4
+ module Exclusion
5
+ # Adds a criterion to the +Criteria+ that specifies values that are not allowed
6
+ # to match any document in the database. The MongoDB conditional operator that
7
+ # will be used is "$ne".
8
+ #
9
+ # Options:
10
+ #
11
+ # attributes: A +Hash+ where the key is the field name and the value is a
12
+ # value that must not be equal to the corresponding field value in the database.
13
+ #
14
+ # Example:
15
+ #
16
+ # <tt>criteria.excludes(:field => "value1")</tt>
17
+ #
18
+ # <tt>criteria.excludes(:field1 => "value1", :field2 => "value1")</tt>
19
+ #
20
+ # Returns: <tt>self</tt>
21
+ def excludes(attributes = {})
22
+ mongo_id = attributes.delete(:id)
23
+ attributes = attributes.merge(:_id => mongo_id) if mongo_id
24
+ update_selector(attributes, "$ne")
25
+ end
26
+
27
+ # Adds a criterion to the +Criteria+ that specifies values where none
28
+ # should match in order to return results. This is similar to an SQL "NOT IN"
29
+ # clause. The MongoDB conditional operator that will be used is "$nin".
30
+ #
31
+ # Options:
32
+ #
33
+ # exclusions: A +Hash+ where the key is the field name and the value is an
34
+ # +Array+ of values that none can match.
35
+ #
36
+ # Example:
37
+ #
38
+ # <tt>criteria.not_in(:field => ["value1", "value2"])</tt>
39
+ #
40
+ # <tt>criteria.not_in(:field1 => ["value1", "value2"], :field2 => ["value1"])</tt>
41
+ #
42
+ # Returns: <tt>self</tt>
43
+ def not_in(exclusions)
44
+ exclusions.each { |key, value| @selector[key] = { "$nin" => value } }; self
45
+ end
46
+
47
+ # Adds a criterion to the +Criteria+ that specifies the fields that will
48
+ # get returned from the Document. Used mainly for list views that do not
49
+ # require all fields to be present. This is similar to SQL "SELECT" values.
50
+ #
51
+ # Options:
52
+ #
53
+ # args: A list of field names to retrict the returned fields to.
54
+ #
55
+ # Example:
56
+ #
57
+ # <tt>criteria.only(:field1, :field2, :field3)</tt>
58
+ #
59
+ # Returns: <tt>self</tt>
60
+ def only(*args)
61
+ @options[:fields] = args.flatten if args.any?; self
62
+ end
63
+ end
64
+ end
65
+ end
@@ -0,0 +1,110 @@
1
+ # encoding: utf-8
2
+ module Mongoid #:nodoc:
3
+ module Criterion #:nodoc:
4
+ module Inclusion
5
+ # Adds a criterion to the +Criteria+ that specifies values that must all
6
+ # be matched in order to return results. Similar to an "in" clause but the
7
+ # underlying conditional logic is an "AND" and not an "OR". The MongoDB
8
+ # conditional operator that will be used is "$all".
9
+ #
10
+ # Options:
11
+ #
12
+ # attributes: A +Hash+ where the key is the field name and the value is an
13
+ # +Array+ of values that must all match.
14
+ #
15
+ # Example:
16
+ #
17
+ # <tt>criteria.all(:field => ["value1", "value2"])</tt>
18
+ #
19
+ # <tt>criteria.all(:field1 => ["value1", "value2"], :field2 => ["value1"])</tt>
20
+ #
21
+ # Returns: <tt>self</tt>
22
+ def all(attributes = {})
23
+ update_selector(attributes, "$all")
24
+ end
25
+ alias :all_in :all
26
+
27
+ # Adds a criterion to the +Criteria+ that specifies values that must
28
+ # be matched in order to return results. This is similar to a SQL "WHERE"
29
+ # clause. This is the actual selector that will be provided to MongoDB,
30
+ # similar to the Javascript object that is used when performing a find()
31
+ # in the MongoDB console.
32
+ #
33
+ # Options:
34
+ #
35
+ # selectior: A +Hash+ that must match the attributes of the +Document+.
36
+ #
37
+ # Example:
38
+ #
39
+ # <tt>criteria.and(:field1 => "value1", :field2 => 15)</tt>
40
+ #
41
+ # Returns: <tt>self</tt>
42
+ def and(selector = nil)
43
+ where(selector)
44
+ end
45
+
46
+ # Adds a criterion to the +Criteria+ that specifies values where any can
47
+ # be matched in order to return results. This is similar to an SQL "IN"
48
+ # clause. The MongoDB conditional operator that will be used is "$in".
49
+ #
50
+ # Options:
51
+ #
52
+ # attributes: A +Hash+ where the key is the field name and the value is an
53
+ # +Array+ of values that any can match.
54
+ #
55
+ # Example:
56
+ #
57
+ # <tt>criteria.in(:field => ["value1", "value2"])</tt>
58
+ #
59
+ # <tt>criteria.in(:field1 => ["value1", "value2"], :field2 => ["value1"])</tt>
60
+ #
61
+ # Returns: <tt>self</tt>
62
+ def in(attributes = {})
63
+ update_selector(attributes, "$in")
64
+ end
65
+ alias :any_in :in
66
+
67
+ # Adds a criterion to the +Criteria+ that specifies values to do
68
+ # geospacial searches by. The field must be indexed with the "2d" option.
69
+ #
70
+ # Options:
71
+ #
72
+ # attributes: A +Hash+ where the keys are the field names and the values are
73
+ # +Arrays+ of [latitude, longitude] pairs.
74
+ #
75
+ # Example:
76
+ #
77
+ # <tt>criteria.near(:field1 => [30, -44])</tt>
78
+ #
79
+ # Returns: <tt>self</tt>
80
+ def near(attributes = {})
81
+ update_selector(attributes, "$near")
82
+ end
83
+
84
+ # Adds a criterion to the +Criteria+ that specifies values that must
85
+ # be matched in order to return results. This is similar to a SQL "WHERE"
86
+ # clause. This is the actual selector that will be provided to MongoDB,
87
+ # similar to the Javascript object that is used when performing a find()
88
+ # in the MongoDB console.
89
+ #
90
+ # Options:
91
+ #
92
+ # selectior: A +Hash+ that must match the attributes of the +Document+.
93
+ #
94
+ # Example:
95
+ #
96
+ # <tt>criteria.where(:field1 => "value1", :field2 => 15)</tt>
97
+ #
98
+ # Returns: <tt>self</tt>
99
+ def where(selector = nil)
100
+ case selector
101
+ when String
102
+ @selector.update("$where" => selector)
103
+ else
104
+ @selector.update(selector ? selector.expand_complex_criteria : {})
105
+ end
106
+ self
107
+ end
108
+ end
109
+ end
110
+ end