mongoid 5.4.0 → 6.4.8
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data/README.md +3 -3
- data/Rakefile +26 -0
- data/lib/config/locales/en.yml +40 -0
- data/lib/mongoid/atomic/modifiers.rb +2 -2
- data/lib/mongoid/atomic.rb +5 -5
- data/lib/mongoid/attributes/readonly.rb +22 -0
- data/lib/mongoid/attributes.rb +22 -21
- data/lib/mongoid/cacheable.rb +36 -0
- data/lib/mongoid/changeable.rb +36 -0
- data/lib/mongoid/clients/options.rb +55 -250
- data/lib/mongoid/clients/sessions.rb +113 -0
- data/lib/mongoid/clients/storage_options.rb +2 -69
- data/lib/mongoid/clients.rb +10 -63
- data/lib/mongoid/composable.rb +29 -2
- data/lib/mongoid/config.rb +1 -0
- data/lib/mongoid/contextual/aggregable/mongo.rb +1 -1
- data/lib/mongoid/contextual/atomic.rb +4 -4
- data/lib/mongoid/contextual/map_reduce.rb +7 -3
- data/lib/mongoid/contextual/memory.rb +9 -4
- data/lib/mongoid/contextual/mongo.rb +65 -30
- data/lib/mongoid/contextual/none.rb +12 -0
- data/lib/mongoid/copyable.rb +13 -6
- data/lib/mongoid/criteria/marshalable.rb +2 -2
- data/lib/mongoid/criteria/modifiable.rb +29 -3
- data/lib/mongoid/criteria/options.rb +25 -0
- data/lib/mongoid/criteria/queryable/aggregable.rb +120 -0
- data/lib/mongoid/criteria/queryable/extensions/array.rb +185 -0
- data/lib/mongoid/criteria/queryable/extensions/big_decimal.rb +37 -0
- data/lib/mongoid/criteria/queryable/extensions/boolean.rb +34 -0
- data/lib/mongoid/criteria/queryable/extensions/date.rb +63 -0
- data/lib/mongoid/criteria/queryable/extensions/date_time.rb +53 -0
- data/lib/mongoid/criteria/queryable/extensions/hash.rb +200 -0
- data/lib/mongoid/criteria/queryable/extensions/nil_class.rb +86 -0
- data/lib/mongoid/criteria/queryable/extensions/numeric.rb +90 -0
- data/lib/mongoid/criteria/queryable/extensions/object.rb +206 -0
- data/lib/mongoid/criteria/queryable/extensions/range.rb +70 -0
- data/lib/mongoid/criteria/queryable/extensions/regexp.rb +79 -0
- data/lib/mongoid/criteria/queryable/extensions/set.rb +34 -0
- data/lib/mongoid/criteria/queryable/extensions/string.rb +137 -0
- data/lib/mongoid/criteria/queryable/extensions/symbol.rb +79 -0
- data/lib/mongoid/criteria/queryable/extensions/time.rb +60 -0
- data/lib/mongoid/criteria/queryable/extensions/time_with_zone.rb +54 -0
- data/lib/mongoid/criteria/queryable/extensions.rb +28 -0
- data/lib/mongoid/criteria/queryable/forwardable.rb +65 -0
- data/lib/mongoid/criteria/queryable/key.rb +103 -0
- data/lib/mongoid/criteria/queryable/macroable.rb +27 -0
- data/lib/mongoid/criteria/queryable/mergeable.rb +273 -0
- data/lib/mongoid/criteria/queryable/optional.rb +429 -0
- data/lib/mongoid/criteria/queryable/options.rb +153 -0
- data/lib/mongoid/criteria/queryable/pipeline.rb +111 -0
- data/lib/mongoid/criteria/queryable/selectable.rb +689 -0
- data/lib/mongoid/criteria/queryable/selector.rb +212 -0
- data/lib/mongoid/criteria/queryable/smash.rb +104 -0
- data/lib/mongoid/criteria/queryable.rb +87 -0
- data/lib/mongoid/criteria.rb +6 -2
- data/lib/mongoid/document.rb +34 -41
- data/lib/mongoid/errors/ambiguous_relationship.rb +1 -1
- data/lib/mongoid/errors/in_memory_collation_not_supported.rb +1 -1
- data/lib/mongoid/errors/invalid_field.rb +2 -2
- data/lib/mongoid/errors/invalid_persistence_option.rb +29 -0
- data/lib/mongoid/errors/invalid_relation.rb +66 -0
- data/lib/mongoid/errors/invalid_session_use.rb +24 -0
- data/lib/mongoid/errors.rb +3 -0
- data/lib/mongoid/evolvable.rb +1 -1
- data/lib/mongoid/extensions/big_decimal.rb +17 -8
- data/lib/mongoid/extensions/date.rb +4 -1
- data/lib/mongoid/extensions/decimal128.rb +3 -3
- data/lib/mongoid/extensions/hash.rb +1 -0
- data/lib/mongoid/extensions/regexp.rb +1 -0
- data/lib/mongoid/extensions/string.rb +6 -3
- data/lib/mongoid/extensions/time.rb +4 -1
- data/lib/mongoid/extensions.rb +0 -4
- data/lib/mongoid/factory.rb +2 -1
- data/lib/mongoid/fields/validators/macro.rb +18 -0
- data/lib/mongoid/findable.rb +2 -2
- data/lib/mongoid/indexable.rb +16 -14
- data/lib/mongoid/interceptable.rb +9 -22
- data/lib/mongoid/matchable/all.rb +2 -2
- data/lib/mongoid/matchable/and.rb +3 -3
- data/lib/mongoid/matchable/default.rb +2 -2
- data/lib/mongoid/matchable/elem_match.rb +28 -0
- data/lib/mongoid/matchable/exists.rb +2 -2
- data/lib/mongoid/matchable/gt.rb +4 -2
- data/lib/mongoid/matchable/gte.rb +4 -2
- data/lib/mongoid/matchable/in.rb +2 -2
- data/lib/mongoid/matchable/lt.rb +4 -2
- data/lib/mongoid/matchable/lte.rb +4 -2
- data/lib/mongoid/matchable/ne.rb +2 -2
- data/lib/mongoid/matchable/nin.rb +2 -2
- data/lib/mongoid/matchable/nor.rb +37 -0
- data/lib/mongoid/matchable/or.rb +3 -3
- data/lib/mongoid/matchable/regexp.rb +3 -3
- data/lib/mongoid/matchable/size.rb +2 -2
- data/lib/mongoid/matchable.rb +16 -7
- data/lib/mongoid/persistable/creatable.rb +5 -3
- data/lib/mongoid/persistable/deletable.rb +5 -3
- data/lib/mongoid/persistable/destroyable.rb +1 -5
- data/lib/mongoid/persistable/settable.rb +5 -5
- data/lib/mongoid/persistable/updatable.rb +7 -14
- data/lib/mongoid/persistable/upsertable.rb +2 -1
- data/lib/mongoid/persistable.rb +4 -6
- data/lib/mongoid/persistence_context.rb +220 -0
- data/lib/mongoid/query_cache.rb +67 -23
- data/lib/mongoid/railtie.rb +17 -1
- data/lib/mongoid/railties/controller_runtime.rb +86 -0
- data/lib/mongoid/relations/accessors.rb +3 -0
- data/lib/mongoid/relations/auto_save.rb +12 -4
- data/lib/mongoid/relations/bindings/referenced/many_to_many.rb +4 -4
- data/lib/mongoid/relations/counter_cache.rb +15 -5
- data/lib/mongoid/relations/eager/base.rb +3 -3
- data/lib/mongoid/relations/eager/has_and_belongs_to_many.rb +2 -2
- data/lib/mongoid/relations/eager/has_many.rb +1 -1
- data/lib/mongoid/relations/eager.rb +6 -11
- data/lib/mongoid/relations/embedded/batchable.rb +20 -18
- data/lib/mongoid/relations/embedded/in.rb +13 -1
- data/lib/mongoid/relations/embedded/many.rb +51 -10
- data/lib/mongoid/relations/embedded/one.rb +14 -1
- data/lib/mongoid/relations/macros.rb +9 -1
- data/lib/mongoid/relations/many.rb +4 -0
- data/lib/mongoid/relations/metadata.rb +3 -3
- data/lib/mongoid/relations/options.rb +2 -2
- data/lib/mongoid/relations/proxy.rb +1 -31
- data/lib/mongoid/relations/referenced/in.rb +19 -10
- data/lib/mongoid/relations/referenced/many.rb +30 -26
- data/lib/mongoid/relations/referenced/many_to_many.rb +20 -13
- data/lib/mongoid/relations/referenced/one.rb +15 -1
- data/lib/mongoid/relations/synchronization.rb +12 -12
- data/lib/mongoid/relations/targets/enumerable.rb +24 -4
- data/lib/mongoid/relations/touchable.rb +7 -4
- data/lib/mongoid/reloadable.rb +2 -2
- data/lib/mongoid/scopable.rb +3 -3
- data/lib/mongoid/serializable.rb +1 -1
- data/lib/mongoid/stateful.rb +1 -0
- data/lib/mongoid/tasks/database.rb +3 -2
- data/lib/mongoid/threaded.rb +74 -0
- data/lib/mongoid/traversable.rb +1 -1
- data/lib/mongoid/validatable/uniqueness.rb +1 -2
- data/lib/mongoid/version.rb +1 -1
- data/lib/mongoid.rb +6 -6
- data/lib/rails/generators/mongoid/config/templates/mongoid.yml +18 -3
- data/spec/app/models/agent.rb +2 -0
- data/spec/app/models/album.rb +5 -1
- data/spec/app/models/array_field.rb +7 -0
- data/spec/app/models/artist.rb +21 -0
- data/spec/app/models/band.rb +3 -0
- data/spec/app/models/book.rb +2 -1
- data/spec/app/models/delegating_patient.rb +16 -0
- data/spec/app/models/dokument.rb +1 -0
- data/spec/app/models/ordered_post.rb +5 -0
- data/spec/app/models/oscar.rb +1 -2
- data/spec/app/models/page.rb +1 -1
- data/spec/app/models/person.rb +3 -3
- data/spec/app/models/princess.rb +2 -0
- data/spec/app/models/record.rb +1 -0
- data/spec/app/models/subscription.rb +1 -0
- data/spec/app/models/thing.rb +1 -1
- data/spec/config/mongoid.yml +15 -0
- data/spec/integration/document_spec.rb +22 -0
- data/spec/mongoid/atomic/modifiers_spec.rb +3 -3
- data/spec/mongoid/atomic_spec.rb +5 -5
- data/spec/mongoid/attributes/nested_spec.rb +18 -14
- data/spec/mongoid/attributes/readonly_spec.rb +87 -44
- data/spec/mongoid/attributes_spec.rb +90 -5
- data/spec/mongoid/cacheable_spec.rb +112 -0
- data/spec/mongoid/changeable_spec.rb +58 -0
- data/spec/mongoid/clients/factory_spec.rb +80 -28
- data/spec/mongoid/clients/options_spec.rb +396 -95
- data/spec/mongoid/clients/sessions_spec.rb +334 -0
- data/spec/mongoid/clients_spec.rb +243 -101
- data/spec/mongoid/composable_spec.rb +7 -0
- data/spec/mongoid/config_spec.rb +67 -11
- data/spec/mongoid/contextual/atomic_spec.rb +3 -3
- data/spec/mongoid/contextual/geo_near_spec.rb +1 -0
- data/spec/mongoid/contextual/mongo_spec.rb +275 -22
- data/spec/mongoid/contextual/none_spec.rb +15 -0
- data/spec/mongoid/copyable_spec.rb +13 -4
- data/spec/mongoid/criteria/modifiable_spec.rb +297 -16
- data/spec/mongoid/criteria/options_spec.rb +29 -0
- data/spec/mongoid/criteria/queryable/aggregable_spec.rb +370 -0
- data/spec/mongoid/criteria/queryable/extensions/array_spec.rb +523 -0
- data/spec/mongoid/criteria/queryable/extensions/big_decimal_spec.rb +59 -0
- data/spec/mongoid/criteria/queryable/extensions/bignum_spec.rb +58 -0
- data/spec/mongoid/criteria/queryable/extensions/boolean_spec.rb +213 -0
- data/spec/mongoid/criteria/queryable/extensions/date_spec.rb +330 -0
- data/spec/mongoid/criteria/queryable/extensions/date_time_spec.rb +405 -0
- data/spec/mongoid/criteria/queryable/extensions/fixnum_spec.rb +58 -0
- data/spec/mongoid/criteria/queryable/extensions/float_spec.rb +65 -0
- data/spec/mongoid/criteria/queryable/extensions/hash_spec.rb +327 -0
- data/spec/mongoid/criteria/queryable/extensions/integer_spec.rb +65 -0
- data/spec/mongoid/criteria/queryable/extensions/nil_class_spec.rb +77 -0
- data/spec/mongoid/criteria/queryable/extensions/object_spec.rb +108 -0
- data/spec/mongoid/criteria/queryable/extensions/range_spec.rb +309 -0
- data/spec/mongoid/{extensions/origin → criteria/queryable/extensions}/regexp_raw_spec.rb +2 -2
- data/spec/mongoid/criteria/queryable/extensions/regexp_spec.rb +90 -0
- data/spec/mongoid/criteria/queryable/extensions/set_spec.rb +39 -0
- data/spec/mongoid/criteria/queryable/extensions/string_spec.rb +302 -0
- data/spec/mongoid/criteria/queryable/extensions/symbol_spec.rb +167 -0
- data/spec/mongoid/criteria/queryable/extensions/time_spec.rb +376 -0
- data/spec/mongoid/criteria/queryable/extensions/time_with_zone_spec.rb +347 -0
- data/spec/mongoid/criteria/queryable/forwardable_spec.rb +87 -0
- data/spec/mongoid/criteria/queryable/key_spec.rb +52 -0
- data/spec/mongoid/criteria/queryable/mergeable_spec.rb +49 -0
- data/spec/mongoid/criteria/queryable/optional_spec.rb +1799 -0
- data/spec/mongoid/criteria/queryable/options_spec.rb +360 -0
- data/spec/mongoid/criteria/queryable/pipeline_spec.rb +200 -0
- data/spec/mongoid/criteria/queryable/queryable_spec.rb +137 -0
- data/spec/mongoid/criteria/queryable/selectable_spec.rb +4242 -0
- data/spec/mongoid/criteria/queryable/selector_spec.rb +844 -0
- data/spec/mongoid/criteria/queryable/smash_spec.rb +30 -0
- data/spec/mongoid/criteria/scopable_spec.rb +81 -0
- data/spec/mongoid/criteria_spec.rb +156 -22
- data/spec/mongoid/document_spec.rb +100 -90
- data/spec/mongoid/errors/invalid_relation_spec.rb +37 -0
- data/spec/mongoid/errors/mongoid_error_spec.rb +6 -3
- data/spec/mongoid/extensions/big_decimal_spec.rb +321 -19
- data/spec/mongoid/extensions/boolean_spec.rb +14 -0
- data/spec/mongoid/extensions/date_spec.rb +2 -6
- data/spec/mongoid/extensions/date_time_spec.rb +2 -6
- data/spec/mongoid/extensions/decimal128_spec.rb +1 -1
- data/spec/mongoid/extensions/float_spec.rb +8 -1
- data/spec/mongoid/extensions/hash_spec.rb +15 -0
- data/spec/mongoid/extensions/integer_spec.rb +8 -1
- data/spec/mongoid/extensions/object_spec.rb +11 -0
- data/spec/mongoid/extensions/regexp_spec.rb +23 -0
- data/spec/mongoid/extensions/string_spec.rb +53 -4
- data/spec/mongoid/extensions/time_spec.rb +2 -6
- data/spec/mongoid/extensions/time_with_zone_spec.rb +2 -6
- data/spec/mongoid/factory_spec.rb +11 -0
- data/spec/mongoid/fields_spec.rb +1 -1
- data/spec/mongoid/findable_spec.rb +47 -2
- data/spec/mongoid/indexable_spec.rb +15 -3
- data/spec/mongoid/interceptable_spec.rb +85 -19
- data/spec/mongoid/matchable/all_spec.rb +4 -4
- data/spec/mongoid/matchable/and_spec.rb +10 -10
- data/spec/mongoid/matchable/default_spec.rb +12 -12
- data/spec/mongoid/matchable/elem_match_spec.rb +86 -0
- data/spec/mongoid/matchable/exists_spec.rb +5 -5
- data/spec/mongoid/matchable/gt_spec.rb +18 -7
- data/spec/mongoid/matchable/gte_spec.rb +17 -7
- data/spec/mongoid/matchable/in_spec.rb +5 -5
- data/spec/mongoid/matchable/lt_spec.rb +18 -7
- data/spec/mongoid/matchable/lte_spec.rb +18 -7
- data/spec/mongoid/matchable/ne_spec.rb +5 -5
- data/spec/mongoid/matchable/nin_spec.rb +5 -5
- data/spec/mongoid/matchable/nor_spec.rb +209 -0
- data/spec/mongoid/matchable/or_spec.rb +7 -7
- data/spec/mongoid/matchable/regexp_spec.rb +5 -5
- data/spec/mongoid/matchable/size_spec.rb +3 -3
- data/spec/mongoid/matchable_spec.rb +199 -54
- data/spec/mongoid/persistable/creatable_spec.rb +7 -2
- data/spec/mongoid/persistable/deletable_spec.rb +35 -1
- data/spec/mongoid/persistable/destroyable_spec.rb +25 -2
- data/spec/mongoid/persistable/incrementable_spec.rb +6 -6
- data/spec/mongoid/persistable/savable_spec.rb +34 -29
- data/spec/mongoid/persistable/settable_spec.rb +77 -27
- data/spec/mongoid/persistable/updatable_spec.rb +182 -3
- data/spec/mongoid/persistable_spec.rb +16 -16
- data/spec/mongoid/persistence_context_spec.rb +694 -0
- data/spec/mongoid/positional_spec.rb +1 -1
- data/spec/mongoid/query_cache_spec.rb +170 -12
- data/spec/mongoid/relations/accessors_spec.rb +1 -1
- data/spec/mongoid/relations/auto_save_spec.rb +39 -6
- data/spec/mongoid/relations/bindings/referenced/many_to_many_spec.rb +4 -4
- data/spec/mongoid/relations/builders_spec.rb +37 -10
- data/spec/mongoid/relations/counter_cache_spec.rb +64 -3
- data/spec/mongoid/relations/eager/has_and_belongs_to_many_spec.rb +16 -0
- data/spec/mongoid/relations/eager_spec.rb +40 -0
- data/spec/mongoid/relations/embedded/many_spec.rb +305 -59
- data/spec/mongoid/relations/embedded/one_spec.rb +2 -1
- data/spec/mongoid/relations/macros_spec.rb +415 -7
- data/spec/mongoid/relations/metadata_spec.rb +15 -1
- data/spec/mongoid/relations/proxy_spec.rb +27 -1
- data/spec/mongoid/relations/referenced/in_spec.rb +41 -1
- data/spec/mongoid/relations/referenced/many_spec.rb +35 -25
- data/spec/mongoid/relations/referenced/many_to_many_spec.rb +14 -26
- data/spec/mongoid/relations/synchronization_spec.rb +48 -2
- data/spec/mongoid/relations/targets/enumerable_spec.rb +108 -0
- data/spec/mongoid/relations/touchable_spec.rb +40 -0
- data/spec/mongoid/reloadable_spec.rb +51 -0
- data/spec/mongoid/scopable_spec.rb +13 -0
- data/spec/mongoid/serializable_spec.rb +0 -50
- data/spec/mongoid/threaded_spec.rb +68 -0
- data/spec/mongoid/validatable/presence_spec.rb +1 -1
- data/spec/mongoid/validatable/uniqueness_spec.rb +18 -9
- data/spec/mongoid/validatable_spec.rb +16 -0
- data/spec/rails/controller_extension/controller_runtime_spec.rb +110 -0
- data/spec/spec_helper.rb +101 -8
- data/spec/support/cluster_config.rb +158 -0
- data/spec/support/constraints.rb +101 -0
- data/spec/support/macros.rb +20 -0
- data/spec/support/session_registry.rb +50 -0
- data/spec/support/spec_config.rb +42 -0
- data.tar.gz.sig +0 -0
- metadata +163 -61
- metadata.gz.sig +0 -0
- data/lib/mongoid/clients/thread_options.rb +0 -19
- data/lib/mongoid/extensions/origin/regexp_raw.rb +0 -43
- data/lib/mongoid/railties/document.rb +0 -12
- data/spec/mongoid/railties/document_spec.rb +0 -24
@@ -0,0 +1,212 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
module Mongoid
|
3
|
+
class Criteria
|
4
|
+
module Queryable
|
5
|
+
|
6
|
+
# The selector is a special kind of hash that knows how to serialize values
|
7
|
+
# coming into it as well as being alias and locale aware for key names.
|
8
|
+
class Selector < Smash
|
9
|
+
|
10
|
+
# Merges another selector into this one.
|
11
|
+
#
|
12
|
+
# @example Merge in another selector.
|
13
|
+
# selector.merge!(name: "test")
|
14
|
+
#
|
15
|
+
# @param [ Hash, Selector ] other The object to merge in.
|
16
|
+
#
|
17
|
+
# @return [ Selector ] The selector.
|
18
|
+
#
|
19
|
+
# @since 1.0.0
|
20
|
+
def merge!(other)
|
21
|
+
other.each_pair do |key, value|
|
22
|
+
if value.is_a?(Hash) && self[key.to_s].is_a?(Hash)
|
23
|
+
value = self[key.to_s].merge(value) do |_key, old_val, new_val|
|
24
|
+
if in?(_key)
|
25
|
+
new_val & old_val
|
26
|
+
elsif nin?(_key)
|
27
|
+
(old_val + new_val).uniq
|
28
|
+
else
|
29
|
+
new_val
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
if multi_selection?(key)
|
34
|
+
value = (self[key.to_s] || []).concat(value)
|
35
|
+
end
|
36
|
+
store(key, value)
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
# Store the value in the selector for the provided key. The selector will
|
41
|
+
# handle all necessary serialization and localization in this step.
|
42
|
+
#
|
43
|
+
# @example Store a value in the selector.
|
44
|
+
# selector.store(:key, "testing")
|
45
|
+
#
|
46
|
+
# @param [ String, Symbol ] key The name of the attribute.
|
47
|
+
# @param [ Object ] value The value to add.
|
48
|
+
#
|
49
|
+
# @return [ Object ] The stored object.
|
50
|
+
#
|
51
|
+
# @since 1.0.0
|
52
|
+
def store(key, value)
|
53
|
+
name, serializer = storage_pair(key)
|
54
|
+
if multi_selection?(name)
|
55
|
+
super(name, evolve_multi(value))
|
56
|
+
else
|
57
|
+
super(localized_key(name, serializer), evolve(serializer, value))
|
58
|
+
end
|
59
|
+
end
|
60
|
+
alias :[]= :store
|
61
|
+
|
62
|
+
# Convert the selector to an aggregation pipeline entry.
|
63
|
+
#
|
64
|
+
# @example Convert the selector to a pipeline.
|
65
|
+
# selector.to_pipeline
|
66
|
+
#
|
67
|
+
# @return [ Array<Hash> ] The pipeline entry for the selector.
|
68
|
+
#
|
69
|
+
# @since 2.0.0
|
70
|
+
def to_pipeline
|
71
|
+
pipeline = []
|
72
|
+
pipeline.push({ "$match" => self }) unless empty?
|
73
|
+
pipeline
|
74
|
+
end
|
75
|
+
|
76
|
+
private
|
77
|
+
|
78
|
+
# Evolves a multi-list selection, like an $and or $or criterion, and
|
79
|
+
# performs the necessary serialization.
|
80
|
+
#
|
81
|
+
# @api private
|
82
|
+
#
|
83
|
+
# @example Evolve the multi-selection.
|
84
|
+
# selector.evolve_multi([{ field: "value" }])
|
85
|
+
#
|
86
|
+
# @param [ Array<Hash> ] value The multi-selection.
|
87
|
+
#
|
88
|
+
# @return [ Array<Hash> ] The serialized values.
|
89
|
+
#
|
90
|
+
# @since 1.0.0
|
91
|
+
def evolve_multi(value)
|
92
|
+
value.map do |val|
|
93
|
+
Hash[val.map do |key, _value|
|
94
|
+
_value = evolve_multi(_value) if multi_selection?(key)
|
95
|
+
name, serializer = storage_pair(key)
|
96
|
+
[ localized_key(name, serializer), evolve(serializer, _value) ]
|
97
|
+
end]
|
98
|
+
end.uniq
|
99
|
+
end
|
100
|
+
|
101
|
+
# Evolve a single key selection with various types of values.
|
102
|
+
#
|
103
|
+
# @api private
|
104
|
+
#
|
105
|
+
# @example Evolve a simple selection.
|
106
|
+
# selector.evolve(field, 5)
|
107
|
+
#
|
108
|
+
# @param [ Object ] serializer The optional serializer for the field.
|
109
|
+
# @param [ Object ] value The value to serialize.
|
110
|
+
#
|
111
|
+
# @return [ Object ] The serialized object.
|
112
|
+
#
|
113
|
+
# @since 1.0.0
|
114
|
+
def evolve(serializer, value)
|
115
|
+
case value
|
116
|
+
when Hash
|
117
|
+
evolve_hash(serializer, value)
|
118
|
+
when Array
|
119
|
+
evolve_array(serializer, value)
|
120
|
+
else
|
121
|
+
(serializer || value.class).evolve(value)
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
# Evolve a single key selection with array values.
|
126
|
+
#
|
127
|
+
# @api private
|
128
|
+
#
|
129
|
+
# @example Evolve a simple selection.
|
130
|
+
# selector.evolve(field, [ 1, 2, 3 ])
|
131
|
+
#
|
132
|
+
# @param [ Object ] serializer The optional serializer for the field.
|
133
|
+
# @param [ Array<Object> ] value The array to serialize.
|
134
|
+
#
|
135
|
+
# @return [ Object ] The serialized array.
|
136
|
+
#
|
137
|
+
# @since 1.0.0
|
138
|
+
def evolve_array(serializer, value)
|
139
|
+
value.map do |_value|
|
140
|
+
evolve(serializer, _value)
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
# Evolve a single key selection with hash values.
|
145
|
+
#
|
146
|
+
# @api private
|
147
|
+
#
|
148
|
+
# @example Evolve a simple selection.
|
149
|
+
# selector.evolve(field, { "$gt" => 5 })
|
150
|
+
#
|
151
|
+
# @param [ Object ] serializer The optional serializer for the field.
|
152
|
+
# @param [ Hash ] value The hash to serialize.
|
153
|
+
#
|
154
|
+
# @return [ Object ] The serialized hash.
|
155
|
+
#
|
156
|
+
# @since 1.0.0
|
157
|
+
def evolve_hash(serializer, value)
|
158
|
+
value.each_pair do |operator, _value|
|
159
|
+
if operator =~ /exists|type|size/
|
160
|
+
value[operator] = _value
|
161
|
+
else
|
162
|
+
value[operator] = evolve(serializer, _value)
|
163
|
+
end
|
164
|
+
end
|
165
|
+
end
|
166
|
+
|
167
|
+
# Determines if the selection is a multi-select, like an $and or $or or $nor
|
168
|
+
# selection.
|
169
|
+
#
|
170
|
+
# @api private
|
171
|
+
#
|
172
|
+
# @example Is the selection a multi-select?
|
173
|
+
# selector.multi_selection?("$and")
|
174
|
+
#
|
175
|
+
# @param [ String ] key The key to check.
|
176
|
+
#
|
177
|
+
# @return [ true, false ] If the key is for a multi-select.
|
178
|
+
#
|
179
|
+
# @since 1.0.0
|
180
|
+
def multi_selection?(key)
|
181
|
+
key =~ /\$and|\$or|\$nor/
|
182
|
+
end
|
183
|
+
|
184
|
+
# Determines if the selection operator takes a list. Returns true for $in and $nin.
|
185
|
+
#
|
186
|
+
# @api private
|
187
|
+
#
|
188
|
+
# @example Does the selection operator take multiple values?
|
189
|
+
# selector.multi_value?("$nin")
|
190
|
+
#
|
191
|
+
# @param [ String ] key The key to check.
|
192
|
+
#
|
193
|
+
# @return [ true, false ] If the key is $in or $nin.
|
194
|
+
#
|
195
|
+
# @since 2.1.1
|
196
|
+
def multi_value?(key)
|
197
|
+
key =~ /\$nin|\$in/
|
198
|
+
end
|
199
|
+
|
200
|
+
private
|
201
|
+
|
202
|
+
def in?(key)
|
203
|
+
key =~ /\$in/
|
204
|
+
end
|
205
|
+
|
206
|
+
def nin?(key)
|
207
|
+
key =~ /\$nin/
|
208
|
+
end
|
209
|
+
end
|
210
|
+
end
|
211
|
+
end
|
212
|
+
end
|
@@ -0,0 +1,104 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
module Mongoid
|
3
|
+
class Criteria
|
4
|
+
module Queryable
|
5
|
+
|
6
|
+
# This is a smart hash for use with options and selectors.
|
7
|
+
class Smash < Hash
|
8
|
+
|
9
|
+
# @attribute [r] aliases The aliases.
|
10
|
+
# @attribute [r] serializers The serializers.
|
11
|
+
attr_reader :aliases, :serializers
|
12
|
+
|
13
|
+
# Perform a deep copy of the smash.
|
14
|
+
#
|
15
|
+
# @example Perform a deep copy.
|
16
|
+
# smash.__deep_copy__
|
17
|
+
#
|
18
|
+
# @return [ Smash ] The copied hash.
|
19
|
+
#
|
20
|
+
# @since 1.0.0
|
21
|
+
def __deep_copy__
|
22
|
+
self.class.new(aliases, serializers) do |copy|
|
23
|
+
each_pair do |key, value|
|
24
|
+
copy.store(key, value.__deep_copy__)
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# Initialize the new selector.
|
30
|
+
#
|
31
|
+
# @example Initialize the new selector.
|
32
|
+
# Queryable::Smash.new(aliases, serializers)
|
33
|
+
#
|
34
|
+
# @param [ Hash ] aliases A hash of mappings from aliases to the actual
|
35
|
+
# field names in the database.
|
36
|
+
# @param [ Hash ] serializers An optional hash of objects that are
|
37
|
+
# responsible for serializing values. The keys of the hash must be
|
38
|
+
# strings that match the field name, and the values must respond to
|
39
|
+
# #localized? and #evolve(object).
|
40
|
+
#
|
41
|
+
# @since 1.0.0
|
42
|
+
def initialize(aliases = {}, serializers = {})
|
43
|
+
@aliases, @serializers = aliases, serializers
|
44
|
+
yield(self) if block_given?
|
45
|
+
end
|
46
|
+
|
47
|
+
# Get an item from the smart hash by the provided key.
|
48
|
+
#
|
49
|
+
# @example Get an item by the key.
|
50
|
+
# smash["test"]
|
51
|
+
#
|
52
|
+
# @param [ String ] key The key.
|
53
|
+
#
|
54
|
+
# @return [ Object ] The found object.
|
55
|
+
#
|
56
|
+
# @since 2.0.0
|
57
|
+
def [](key)
|
58
|
+
fetch(aliases[key]) { super }
|
59
|
+
end
|
60
|
+
|
61
|
+
private
|
62
|
+
|
63
|
+
# Get the localized value for the key if needed. If the field uses
|
64
|
+
# localization the current locale will be appended to the key in
|
65
|
+
# MongoDB dot notation.
|
66
|
+
#
|
67
|
+
# @api private
|
68
|
+
#
|
69
|
+
# @example Get the normalized key name.
|
70
|
+
# smash.localized_key("field", serializer)
|
71
|
+
#
|
72
|
+
# @param [ String ] name The name of the field.
|
73
|
+
# @param [ Object ] serializer The optional field serializer.
|
74
|
+
#
|
75
|
+
# @return [ String ] The normalized key.
|
76
|
+
#
|
77
|
+
# @since 1.0.0
|
78
|
+
def localized_key(name, serializer)
|
79
|
+
serializer && serializer.localized? ? "#{name}.#{::I18n.locale}" : name
|
80
|
+
end
|
81
|
+
|
82
|
+
# Get the pair of objects needed to store the value in a hash by the
|
83
|
+
# provided key. This is the database field name and the serializer.
|
84
|
+
#
|
85
|
+
# @api private
|
86
|
+
#
|
87
|
+
# @example Get the name and serializer.
|
88
|
+
# smash.storage_pair("id")
|
89
|
+
#
|
90
|
+
# @param [ Symbol, String ] key The key provided to the selection.
|
91
|
+
#
|
92
|
+
# @return [ Array<String, Object> ] The name of the db field and
|
93
|
+
# serializer.
|
94
|
+
#
|
95
|
+
# @since 1.0.0
|
96
|
+
def storage_pair(key)
|
97
|
+
field = key.to_s
|
98
|
+
name = aliases[field] || field
|
99
|
+
[ name, serializers[name] ]
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
@@ -0,0 +1,87 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
require "mongoid/criteria/queryable/extensions"
|
3
|
+
require "mongoid/criteria/queryable/forwardable"
|
4
|
+
require "mongoid/criteria/queryable/key"
|
5
|
+
require "mongoid/criteria/queryable/macroable"
|
6
|
+
require "mongoid/criteria/queryable/mergeable"
|
7
|
+
require "mongoid/criteria/queryable/smash"
|
8
|
+
require "mongoid/criteria/queryable/aggregable"
|
9
|
+
require "mongoid/criteria/queryable/pipeline"
|
10
|
+
require "mongoid/criteria/queryable/optional"
|
11
|
+
require "mongoid/criteria/queryable/options"
|
12
|
+
require "mongoid/criteria/queryable/selectable"
|
13
|
+
require "mongoid/criteria/queryable/selector"
|
14
|
+
|
15
|
+
module Mongoid
|
16
|
+
class Criteria
|
17
|
+
|
18
|
+
# A queryable is any object that needs queryable's dsl injected into it to build
|
19
|
+
# MongoDB queries. For example, a Mongoid::Criteria is an Queryable.
|
20
|
+
#
|
21
|
+
# @example Include queryable functionality.
|
22
|
+
# class Criteria
|
23
|
+
# include Queryable
|
24
|
+
# end
|
25
|
+
module Queryable
|
26
|
+
include Mergeable
|
27
|
+
include Aggregable
|
28
|
+
include Selectable
|
29
|
+
include Optional
|
30
|
+
|
31
|
+
# @attribute [r] aliases The aliases.
|
32
|
+
# @attribute [r] driver The Mongo driver being used.
|
33
|
+
# @attribute [r] serializers The serializers.
|
34
|
+
attr_reader :aliases, :driver, :serializers
|
35
|
+
|
36
|
+
# Is this queryable equal to another object? Is true if the selector and
|
37
|
+
# options are equal.
|
38
|
+
#
|
39
|
+
# @example Are the objects equal?
|
40
|
+
# queryable == criteria
|
41
|
+
#
|
42
|
+
# @param [ Object ] other The object to compare against.
|
43
|
+
#
|
44
|
+
# @return [ true, false ] If the objects are equal.
|
45
|
+
#
|
46
|
+
# @since 1.0.0
|
47
|
+
def ==(other)
|
48
|
+
return false unless other.is_a?(Queryable)
|
49
|
+
selector == other.selector && options == other.options
|
50
|
+
end
|
51
|
+
|
52
|
+
# Initialize the new queryable. Will yield itself to the block if a block
|
53
|
+
# is provided for objects that need additional behaviour.
|
54
|
+
#
|
55
|
+
# @example Initialize the queryable.
|
56
|
+
# Queryable.new
|
57
|
+
#
|
58
|
+
# @param [ Hash ] aliases The optional field aliases.
|
59
|
+
# @param [ Hash ] serializers The optional field serializers.
|
60
|
+
# @param [ Symbol ] driver The driver being used.
|
61
|
+
#
|
62
|
+
# @since 1.0.0
|
63
|
+
def initialize(aliases = {}, serializers = {}, driver = :mongo)
|
64
|
+
@aliases, @driver, @serializers = aliases, driver.to_sym, serializers
|
65
|
+
@options = Options.new(aliases, serializers)
|
66
|
+
@selector = Selector.new(aliases, serializers)
|
67
|
+
@pipeline = Pipeline.new(aliases)
|
68
|
+
@aggregating = nil
|
69
|
+
yield(self) if block_given?
|
70
|
+
end
|
71
|
+
|
72
|
+
# Handle the creation of a copy via #clone or #dup.
|
73
|
+
#
|
74
|
+
# @example Handle copy initialization.
|
75
|
+
# queryable.initialize_copy(criteria)
|
76
|
+
#
|
77
|
+
# @param [ Queryable ] other The original copy.
|
78
|
+
#
|
79
|
+
# @since 1.0.0
|
80
|
+
def initialize_copy(other)
|
81
|
+
@options = other.options.__deep_copy__
|
82
|
+
@selector = other.selector.__deep_copy__
|
83
|
+
@pipeline = other.pipeline.__deep_copy__
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
data/lib/mongoid/criteria.rb
CHANGED
@@ -4,7 +4,9 @@ require "mongoid/criteria/includable"
|
|
4
4
|
require "mongoid/criteria/inspectable"
|
5
5
|
require "mongoid/criteria/marshalable"
|
6
6
|
require "mongoid/criteria/modifiable"
|
7
|
+
require "mongoid/criteria/queryable"
|
7
8
|
require "mongoid/criteria/scopable"
|
9
|
+
require "mongoid/criteria/options"
|
8
10
|
|
9
11
|
module Mongoid
|
10
12
|
|
@@ -17,7 +19,7 @@ module Mongoid
|
|
17
19
|
class Criteria
|
18
20
|
include Enumerable
|
19
21
|
include Contextual
|
20
|
-
include
|
22
|
+
include Queryable
|
21
23
|
include Findable
|
22
24
|
include Inspectable
|
23
25
|
include Includable
|
@@ -25,6 +27,8 @@ module Mongoid
|
|
25
27
|
include Modifiable
|
26
28
|
include Scopable
|
27
29
|
include Clients::Options
|
30
|
+
include Options
|
31
|
+
include Clients::Sessions
|
28
32
|
|
29
33
|
# Static array used to check with method missing - we only need to ever
|
30
34
|
# instantiate once.
|
@@ -208,7 +212,7 @@ module Mongoid
|
|
208
212
|
#
|
209
213
|
# @example Merge the criteria with a hash. The hash must contain a klass
|
210
214
|
# key and the key/value pairs correspond to method names/args.
|
211
|
-
|
215
|
+
#
|
212
216
|
# criteria.merge({
|
213
217
|
# klass: Band,
|
214
218
|
# where: { name: "Depeche Mode" },
|
data/lib/mongoid/document.rb
CHANGED
@@ -31,6 +31,15 @@ module Mongoid
|
|
31
31
|
Mongoid.register_model(self)
|
32
32
|
end
|
33
33
|
|
34
|
+
# Regex for matching illegal BSON keys.
|
35
|
+
# Note that bson 4.1 has the constant BSON::String::ILLEGAL_KEY
|
36
|
+
# that should be used instead.
|
37
|
+
# When ruby driver 2.3.0 is released and Mongoid can be updated
|
38
|
+
# to require >= 2.3.0, the BSON constant can be used.
|
39
|
+
#
|
40
|
+
# @since 6.0.0
|
41
|
+
ILLEGAL_KEY = /(\A[$])|(\.)/.freeze
|
42
|
+
|
34
43
|
# Freezes the internal attributes of the document.
|
35
44
|
#
|
36
45
|
# @example Freeze the document
|
@@ -40,7 +49,7 @@ module Mongoid
|
|
40
49
|
#
|
41
50
|
# @since 2.0.0
|
42
51
|
def freeze
|
43
|
-
|
52
|
+
as_attributes.freeze and self
|
44
53
|
end
|
45
54
|
|
46
55
|
# Checks if the document is frozen
|
@@ -105,7 +114,6 @@ module Mongoid
|
|
105
114
|
_building do
|
106
115
|
@new_record = true
|
107
116
|
@attributes ||= {}
|
108
|
-
with(self.class.persistence_options)
|
109
117
|
apply_pre_processed_defaults
|
110
118
|
apply_default_scoping
|
111
119
|
process_attributes(attrs) do
|
@@ -165,20 +173,7 @@ module Mongoid
|
|
165
173
|
#
|
166
174
|
# @since 1.0.0
|
167
175
|
def as_document
|
168
|
-
|
169
|
-
embedded_relations.each_pair do |name, meta|
|
170
|
-
without_autobuild do
|
171
|
-
relation, stored = send(name), meta.store_as
|
172
|
-
if attributes.key?(stored) || !relation.blank?
|
173
|
-
if relation
|
174
|
-
attributes[stored] = relation.as_document
|
175
|
-
else
|
176
|
-
attributes.delete(stored)
|
177
|
-
end
|
178
|
-
end
|
179
|
-
end
|
180
|
-
end
|
181
|
-
attributes
|
176
|
+
BSON::Document.new(as_attributes)
|
182
177
|
end
|
183
178
|
|
184
179
|
# Calls #as_json on the document with additional, Mongoid-specific options.
|
@@ -195,11 +190,11 @@ module Mongoid
|
|
195
190
|
#
|
196
191
|
# @since 5.1.0
|
197
192
|
def as_json(options = nil)
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
super(options)
|
193
|
+
rv = super
|
194
|
+
if options && options[:compact]
|
195
|
+
rv = rv.compact
|
202
196
|
end
|
197
|
+
rv
|
203
198
|
end
|
204
199
|
|
205
200
|
# Returns an instance of the specified class with the attributes,
|
@@ -243,27 +238,6 @@ module Mongoid
|
|
243
238
|
became
|
244
239
|
end
|
245
240
|
|
246
|
-
# Print out the cache key. This will append different values on the
|
247
|
-
# plural model name.
|
248
|
-
#
|
249
|
-
# If new_record? - will append /new
|
250
|
-
# If not - will append /id-updated_at.to_s(:nsec)
|
251
|
-
# Without updated_at - will append /id
|
252
|
-
#
|
253
|
-
# This is usually called insode a cache() block
|
254
|
-
#
|
255
|
-
# @example Returns the cache key
|
256
|
-
# document.cache_key
|
257
|
-
#
|
258
|
-
# @return [ String ] the string with or without updated_at
|
259
|
-
#
|
260
|
-
# @since 2.4.0
|
261
|
-
def cache_key
|
262
|
-
return "#{model_key}/new" if new_record?
|
263
|
-
return "#{model_key}/#{id}-#{updated_at.utc.to_s(:nsec)}" if do_or_do_not(:updated_at)
|
264
|
-
"#{model_key}/#{id}"
|
265
|
-
end
|
266
|
-
|
267
241
|
private
|
268
242
|
|
269
243
|
# Returns the logger
|
@@ -299,6 +273,25 @@ module Mongoid
|
|
299
273
|
nil
|
300
274
|
end
|
301
275
|
|
276
|
+
private
|
277
|
+
|
278
|
+
def as_attributes
|
279
|
+
return attributes if frozen?
|
280
|
+
embedded_relations.each_pair do |name, meta|
|
281
|
+
without_autobuild do
|
282
|
+
relation, stored = send(name), meta.store_as
|
283
|
+
if attributes.key?(stored) || !relation.blank?
|
284
|
+
if relation
|
285
|
+
attributes[stored] = relation.send(:as_attributes)
|
286
|
+
else
|
287
|
+
attributes.delete(stored)
|
288
|
+
end
|
289
|
+
end
|
290
|
+
end
|
291
|
+
end
|
292
|
+
attributes
|
293
|
+
end
|
294
|
+
|
302
295
|
module ClassMethods
|
303
296
|
|
304
297
|
# Performs class equality checking.
|
@@ -29,10 +29,10 @@ module Mongoid
|
|
29
29
|
|
30
30
|
private
|
31
31
|
|
32
|
-
# Get the
|
32
|
+
# Get the queryable of the method.
|
33
33
|
#
|
34
34
|
# @example Get the originating class or module.
|
35
|
-
# error.
|
35
|
+
# error.queryable(Person, :crazy_method_name)
|
36
36
|
#
|
37
37
|
# @param [ Class ] klass The document class.
|
38
38
|
# @param [ Symbol ] name The method name.
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
module Mongoid
|
3
|
+
module Errors
|
4
|
+
|
5
|
+
# Raised when invalid options are used to create a persistence context.
|
6
|
+
#
|
7
|
+
# @since 6.0.0
|
8
|
+
class InvalidPersistenceOption < MongoidError
|
9
|
+
|
10
|
+
# Instantiate the persistence context option error.
|
11
|
+
#
|
12
|
+
# @example Create the error.
|
13
|
+
# InvalidPersistenceOption.new(:invalid_option, [ :connect_timeout, :database ])
|
14
|
+
#
|
15
|
+
# @param [ Symbol ] invalid The invalid option.
|
16
|
+
# @param [ Array<Symbol> ] valid The allowed options.
|
17
|
+
#
|
18
|
+
# @since 6.0.0
|
19
|
+
def initialize(invalid, valid)
|
20
|
+
super(
|
21
|
+
compose_message(
|
22
|
+
"invalid_persistence_option",
|
23
|
+
{ invalid: invalid, valid: valid }
|
24
|
+
)
|
25
|
+
)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,66 @@
|
|
1
|
+
# encoding: utf-8
|
2
|
+
module Mongoid
|
3
|
+
module Errors
|
4
|
+
|
5
|
+
# This error is raised when trying to create a relation that conflicts with
|
6
|
+
# an already defined method.
|
7
|
+
#
|
8
|
+
# @since 6.0.0
|
9
|
+
class InvalidRelation < MongoidError
|
10
|
+
|
11
|
+
# Create the new error.
|
12
|
+
#
|
13
|
+
# @example Create the error.
|
14
|
+
# InvalidRelation.new(person, :crazy_relation_name)
|
15
|
+
#
|
16
|
+
# @param [ Class ] klass The document class.
|
17
|
+
# @param [ Symbol ] name The method name.
|
18
|
+
def initialize(klass, name)
|
19
|
+
super(
|
20
|
+
compose_message(
|
21
|
+
"invalid_relation",
|
22
|
+
{
|
23
|
+
name: name,
|
24
|
+
origin: origin(klass, name),
|
25
|
+
file: location(klass, name)[0],
|
26
|
+
line: location(klass, name)[1]
|
27
|
+
}
|
28
|
+
)
|
29
|
+
)
|
30
|
+
end
|
31
|
+
|
32
|
+
private
|
33
|
+
|
34
|
+
# Get the queryable of the method.
|
35
|
+
#
|
36
|
+
# @example Get the originating class or module.
|
37
|
+
# error.queryable(Person, :crazy_method_name)
|
38
|
+
#
|
39
|
+
# @param [ Class ] klass The document class.
|
40
|
+
# @param [ Symbol ] name The method name.
|
41
|
+
#
|
42
|
+
# @return [ Class, Module ] The originating class or module.
|
43
|
+
#
|
44
|
+
# @since 6.0.0
|
45
|
+
def origin(klass, name)
|
46
|
+
klass.instance_method(name).owner
|
47
|
+
end
|
48
|
+
|
49
|
+
# Get the location of the relation definition.
|
50
|
+
#
|
51
|
+
# @example Get the location of the method on the filesystem.
|
52
|
+
# error.location(Person, :crazy_method_name)
|
53
|
+
#
|
54
|
+
# @param [ Class ] klass The document class.
|
55
|
+
# @param [ Symbol ] name The method name.
|
56
|
+
#
|
57
|
+
# @return [ Array<String, Integer> ] The location of the method.
|
58
|
+
#
|
59
|
+
# @since 6.0.0
|
60
|
+
def location(klass, name)
|
61
|
+
@location ||=
|
62
|
+
(klass.instance_method(name).source_location || [ "Unknown", 0 ])
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|