mongoid 7.5.3 → 8.0.1
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/lib/config/locales/en.yml +46 -30
- data/lib/mongoid/association/accessors.rb +32 -3
- data/lib/mongoid/association/bindable.rb +48 -0
- data/lib/mongoid/association/builders.rb +4 -2
- data/lib/mongoid/association/eager_loadable.rb +29 -7
- data/lib/mongoid/association/embedded/batchable.rb +28 -5
- data/lib/mongoid/association/embedded/embedded_in/binding.rb +24 -2
- data/lib/mongoid/association/embedded/embedded_in.rb +2 -1
- data/lib/mongoid/association/embedded/embeds_many/binding.rb +1 -0
- data/lib/mongoid/association/embedded/embeds_many/buildable.rb +1 -1
- data/lib/mongoid/association/embedded/embeds_many/proxy.rb +40 -18
- data/lib/mongoid/association/embedded/embeds_one/buildable.rb +18 -4
- data/lib/mongoid/association/embedded/embeds_one/proxy.rb +21 -2
- data/lib/mongoid/association/macros.rb +2 -1
- data/lib/mongoid/association/many.rb +5 -0
- data/lib/mongoid/association/nested/many.rb +2 -1
- data/lib/mongoid/association/proxy.rb +12 -0
- data/lib/mongoid/association/referenced/auto_save.rb +3 -2
- data/lib/mongoid/association/referenced/belongs_to/binding.rb +1 -0
- data/lib/mongoid/association/referenced/belongs_to/buildable.rb +1 -1
- data/lib/mongoid/association/referenced/belongs_to.rb +1 -1
- data/lib/mongoid/association/referenced/counter_cache.rb +8 -8
- data/lib/mongoid/association/referenced/has_and_belongs_to_many/proxy.rb +64 -11
- data/lib/mongoid/association/referenced/has_and_belongs_to_many.rb +4 -1
- data/lib/mongoid/association/referenced/has_many/enumerable.rb +10 -18
- data/lib/mongoid/association/referenced/has_many/proxy.rb +12 -9
- data/lib/mongoid/association/referenced/has_one/buildable.rb +1 -1
- data/lib/mongoid/association/referenced/has_one/proxy.rb +8 -11
- data/lib/mongoid/association/referenced/syncable.rb +2 -2
- data/lib/mongoid/association/relatable.rb +38 -4
- data/lib/mongoid/attributes/processing.rb +9 -2
- data/lib/mongoid/attributes.rb +30 -27
- data/lib/mongoid/cacheable.rb +2 -2
- data/lib/mongoid/changeable.rb +37 -2
- data/lib/mongoid/clients/options.rb +4 -0
- data/lib/mongoid/clients/sessions.rb +2 -14
- data/lib/mongoid/config.rb +15 -11
- data/lib/mongoid/contextual/aggregable/memory.rb +23 -15
- data/lib/mongoid/contextual/aggregable/mongo.rb +1 -1
- data/lib/mongoid/contextual/map_reduce.rb +2 -2
- data/lib/mongoid/contextual/memory.rb +55 -28
- data/lib/mongoid/contextual/mongo.rb +173 -262
- data/lib/mongoid/contextual/none.rb +33 -15
- data/lib/mongoid/copyable.rb +32 -8
- data/lib/mongoid/criteria/includable.rb +24 -20
- data/lib/mongoid/criteria/marshalable.rb +10 -2
- data/lib/mongoid/criteria/queryable/extensions/array.rb +2 -15
- data/lib/mongoid/criteria/queryable/extensions/big_decimal.rb +25 -4
- data/lib/mongoid/criteria/queryable/extensions/boolean.rb +1 -1
- data/lib/mongoid/criteria/queryable/extensions/date.rb +6 -1
- data/lib/mongoid/criteria/queryable/extensions/date_time.rb +6 -1
- data/lib/mongoid/criteria/queryable/extensions/hash.rb +0 -16
- data/lib/mongoid/criteria/queryable/extensions/numeric.rb +1 -1
- data/lib/mongoid/criteria/queryable/extensions/object.rb +2 -1
- data/lib/mongoid/criteria/queryable/extensions/range.rb +13 -5
- data/lib/mongoid/criteria/queryable/extensions/regexp.rb +1 -1
- data/lib/mongoid/criteria/queryable/extensions/symbol.rb +3 -1
- data/lib/mongoid/criteria/queryable/extensions/time.rb +6 -1
- data/lib/mongoid/criteria/queryable/extensions/time_with_zone.rb +6 -1
- data/lib/mongoid/criteria/queryable/optional.rb +3 -9
- data/lib/mongoid/criteria/queryable/options.rb +1 -1
- data/lib/mongoid/criteria/queryable/selectable.rb +2 -24
- data/lib/mongoid/criteria/queryable/selector.rb +89 -4
- data/lib/mongoid/criteria/queryable/smash.rb +39 -6
- data/lib/mongoid/criteria/queryable.rb +11 -6
- data/lib/mongoid/criteria.rb +1 -28
- data/lib/mongoid/deprecable.rb +36 -0
- data/lib/mongoid/deprecation.rb +25 -0
- data/lib/mongoid/document.rb +88 -33
- data/lib/mongoid/equality.rb +4 -4
- data/lib/mongoid/errors/document_not_found.rb +6 -2
- data/lib/mongoid/errors/invalid_dot_dollar_assignment.rb +23 -0
- data/lib/mongoid/errors/invalid_field.rb +5 -1
- data/lib/mongoid/errors/invalid_field_type.rb +26 -0
- data/lib/mongoid/errors/too_many_nested_attribute_records.rb +1 -1
- data/lib/mongoid/errors.rb +2 -2
- data/lib/mongoid/extensions/array.rb +8 -6
- data/lib/mongoid/extensions/big_decimal.rb +29 -10
- data/lib/mongoid/extensions/binary.rb +42 -0
- data/lib/mongoid/extensions/boolean.rb +8 -2
- data/lib/mongoid/extensions/date.rb +26 -20
- data/lib/mongoid/extensions/date_time.rb +1 -1
- data/lib/mongoid/extensions/float.rb +4 -5
- data/lib/mongoid/extensions/hash.rb +12 -5
- data/lib/mongoid/extensions/integer.rb +4 -5
- data/lib/mongoid/extensions/object.rb +2 -0
- data/lib/mongoid/extensions/range.rb +41 -10
- data/lib/mongoid/extensions/regexp.rb +11 -4
- data/lib/mongoid/extensions/set.rb +11 -4
- data/lib/mongoid/extensions/string.rb +2 -13
- data/lib/mongoid/extensions/symbol.rb +3 -14
- data/lib/mongoid/extensions/time.rb +27 -16
- data/lib/mongoid/extensions/time_with_zone.rb +1 -2
- data/lib/mongoid/extensions.rb +1 -0
- data/lib/mongoid/factory.rb +42 -7
- data/lib/mongoid/fields/foreign_key.rb +7 -0
- data/lib/mongoid/fields/validators/macro.rb +3 -9
- data/lib/mongoid/fields.rb +49 -7
- data/lib/mongoid/findable.rb +21 -16
- data/lib/mongoid/indexable/specification.rb +1 -1
- data/lib/mongoid/indexable/validators/options.rb +4 -1
- data/lib/mongoid/interceptable.rb +69 -9
- data/lib/mongoid/persistable/creatable.rb +14 -5
- data/lib/mongoid/persistable/updatable.rb +12 -5
- data/lib/mongoid/persistence_context.rb +8 -42
- data/lib/mongoid/query_cache.rb +6 -258
- data/lib/mongoid/railties/controller_runtime.rb +1 -1
- data/lib/mongoid/reloadable.rb +7 -3
- data/lib/mongoid/scopable.rb +9 -11
- data/lib/mongoid/selectable.rb +1 -2
- data/lib/mongoid/shardable.rb +11 -35
- data/lib/mongoid/stateful.rb +27 -1
- data/lib/mongoid/timestamps/created.rb +1 -1
- data/lib/mongoid/timestamps/updated.rb +1 -1
- data/lib/mongoid/touchable.rb +2 -3
- data/lib/mongoid/traversable.rb +1 -0
- data/lib/mongoid/validatable/uniqueness.rb +2 -1
- data/lib/mongoid/version.rb +1 -1
- data/lib/mongoid/warnings.rb +3 -4
- data/lib/mongoid.rb +1 -0
- data/spec/config/mongoid.yml +16 -0
- data/spec/integration/app_spec.rb +8 -12
- data/spec/integration/associations/belongs_to_spec.rb +18 -0
- data/spec/integration/associations/embedded_spec.rb +15 -0
- data/spec/integration/associations/embeds_many_spec.rb +15 -2
- data/spec/integration/associations/embeds_one_spec.rb +18 -0
- data/spec/integration/associations/foreign_key_spec.rb +9 -0
- data/spec/integration/associations/has_and_belongs_to_many_spec.rb +21 -0
- data/spec/integration/associations/has_one_spec.rb +97 -1
- data/spec/integration/associations/scope_option_spec.rb +1 -1
- data/spec/integration/callbacks_models.rb +95 -1
- data/spec/integration/callbacks_spec.rb +226 -4
- data/spec/integration/criteria/range_spec.rb +95 -1
- data/spec/integration/discriminator_key_spec.rb +115 -76
- data/spec/integration/dots_and_dollars_spec.rb +277 -0
- data/spec/integration/i18n_fallbacks_spec.rb +1 -15
- data/spec/integration/matcher_examples_spec.rb +20 -13
- data/spec/integration/matcher_operator_data/type_decimal.yml +3 -2
- data/spec/integration/matcher_operator_spec.rb +3 -5
- data/spec/integration/persistence/range_field_spec.rb +350 -0
- data/spec/mongoid/association/counter_cache_spec.rb +1 -1
- data/spec/mongoid/association/depending_spec.rb +9 -9
- data/spec/mongoid/association/eager_spec.rb +2 -1
- data/spec/mongoid/association/embedded/embedded_in/binding_spec.rb +2 -1
- data/spec/mongoid/association/embedded/embedded_in/buildable_spec.rb +54 -0
- data/spec/mongoid/association/embedded/embedded_in/proxy_spec.rb +69 -9
- data/spec/mongoid/association/embedded/embeds_many/buildable_spec.rb +112 -0
- data/spec/mongoid/association/embedded/embeds_many/proxy_spec.rb +198 -8
- data/spec/mongoid/association/embedded/embeds_many_models.rb +36 -0
- data/spec/mongoid/association/embedded/embeds_many_query_spec.rb +12 -0
- data/spec/mongoid/association/embedded/embeds_many_spec.rb +68 -0
- data/spec/mongoid/association/embedded/embeds_one/buildable_spec.rb +25 -0
- data/spec/mongoid/association/embedded/embeds_one_models.rb +19 -0
- data/spec/mongoid/association/embedded/embeds_one_spec.rb +28 -0
- data/spec/mongoid/association/referenced/belongs_to/binding_spec.rb +2 -1
- data/spec/mongoid/association/referenced/belongs_to/buildable_spec.rb +54 -0
- data/spec/mongoid/association/referenced/belongs_to/proxy_spec.rb +15 -0
- data/spec/mongoid/association/referenced/belongs_to_models.rb +11 -0
- data/spec/mongoid/association/referenced/belongs_to_spec.rb +2 -2
- data/spec/mongoid/association/referenced/has_and_belongs_to_many/proxy_spec.rb +38 -5
- data/spec/mongoid/association/referenced/has_and_belongs_to_many_models.rb +25 -0
- data/spec/mongoid/association/referenced/has_and_belongs_to_many_spec.rb +35 -2
- data/spec/mongoid/association/referenced/has_many/buildable_spec.rb +109 -0
- data/spec/mongoid/association/referenced/has_many/enumerable_spec.rb +2 -56
- data/spec/mongoid/association/referenced/has_many/proxy_spec.rb +62 -13
- data/spec/mongoid/association/referenced/has_many_models.rb +3 -1
- data/spec/mongoid/association/referenced/has_many_spec.rb +25 -0
- data/spec/mongoid/association/referenced/has_one/buildable_spec.rb +2 -2
- data/spec/mongoid/association/referenced/has_one/proxy_spec.rb +107 -1
- data/spec/mongoid/association/referenced/has_one_models.rb +16 -0
- data/spec/mongoid/association/syncable_spec.rb +14 -0
- data/spec/mongoid/atomic/paths_spec.rb +0 -14
- data/spec/mongoid/attributes/nested_spec.rb +80 -11
- data/spec/mongoid/attributes/nested_spec_models.rb +48 -0
- data/spec/mongoid/attributes/projector_spec.rb +1 -5
- data/spec/mongoid/attributes_spec.rb +480 -27
- data/spec/mongoid/cacheable_spec.rb +3 -3
- data/spec/mongoid/changeable_spec.rb +130 -13
- data/spec/mongoid/clients/factory_spec.rb +23 -30
- data/spec/mongoid/clients/sessions_spec.rb +0 -38
- data/spec/mongoid/clients_spec.rb +2 -2
- data/spec/mongoid/config_spec.rb +52 -14
- data/spec/mongoid/contextual/aggregable/memory_spec.rb +396 -158
- data/spec/mongoid/contextual/aggregable/memory_table.yml +88 -0
- data/spec/mongoid/contextual/aggregable/memory_table_spec.rb +62 -0
- data/spec/mongoid/contextual/map_reduce_spec.rb +2 -16
- data/spec/mongoid/contextual/memory_spec.rb +521 -14
- data/spec/mongoid/contextual/mongo_spec.rb +566 -416
- data/spec/mongoid/contextual/none_spec.rb +11 -19
- data/spec/mongoid/copyable_spec.rb +451 -1
- data/spec/mongoid/criteria/findable_spec.rb +86 -210
- data/spec/mongoid/criteria/includable_spec.rb +1492 -0
- data/spec/mongoid/criteria/includable_spec_models.rb +54 -0
- data/spec/mongoid/criteria/marshalable_spec.rb +18 -1
- data/spec/mongoid/criteria/queryable/extensions/array_spec.rb +7 -19
- data/spec/mongoid/criteria/queryable/extensions/big_decimal_spec.rb +134 -26
- data/spec/mongoid/criteria/queryable/extensions/date_spec.rb +11 -0
- data/spec/mongoid/criteria/queryable/extensions/date_time_spec.rb +11 -0
- data/spec/mongoid/criteria/queryable/extensions/hash_spec.rb +0 -15
- data/spec/mongoid/criteria/queryable/extensions/numeric_spec.rb +73 -7
- data/spec/mongoid/criteria/queryable/extensions/time_spec.rb +11 -0
- data/spec/mongoid/criteria/queryable/extensions/time_with_zone_spec.rb +11 -0
- data/spec/mongoid/criteria/queryable/optional_spec.rb +0 -484
- data/spec/mongoid/criteria/queryable/selectable_logical_spec.rb +50 -0
- data/spec/mongoid/criteria/queryable/selectable_spec.rb +77 -85
- data/spec/mongoid/criteria/queryable/selector_spec.rb +14 -2
- data/spec/mongoid/criteria_spec.rb +469 -1201
- data/spec/mongoid/document_fields_spec.rb +173 -24
- data/spec/mongoid/document_spec.rb +32 -41
- data/spec/mongoid/equality_spec.rb +12 -12
- data/spec/mongoid/errors/document_not_found_spec.rb +29 -2
- data/spec/mongoid/errors/invalid_field_spec.rb +1 -1
- data/spec/mongoid/errors/invalid_field_type_spec.rb +55 -0
- data/spec/mongoid/errors/mongoid_error_spec.rb +3 -1
- data/spec/mongoid/errors/no_environment_spec.rb +3 -3
- data/spec/mongoid/errors/too_many_nested_attribute_records_spec.rb +1 -1
- data/spec/mongoid/extensions/array_spec.rb +16 -2
- data/spec/mongoid/extensions/big_decimal_spec.rb +697 -212
- data/spec/mongoid/extensions/binary_spec.rb +44 -9
- data/spec/mongoid/extensions/boolean_spec.rb +68 -82
- data/spec/mongoid/extensions/date_class_mongoize_spec.rb +7 -3
- data/spec/mongoid/extensions/date_spec.rb +71 -1
- data/spec/mongoid/extensions/date_time_spec.rb +15 -9
- data/spec/mongoid/extensions/float_spec.rb +48 -76
- data/spec/mongoid/extensions/hash_spec.rb +30 -0
- data/spec/mongoid/extensions/integer_spec.rb +45 -66
- data/spec/mongoid/extensions/range_spec.rb +255 -54
- data/spec/mongoid/extensions/regexp_spec.rb +58 -33
- data/spec/mongoid/extensions/set_spec.rb +106 -0
- data/spec/mongoid/extensions/string_spec.rb +53 -25
- data/spec/mongoid/extensions/symbol_spec.rb +18 -25
- data/spec/mongoid/extensions/time_spec.rb +634 -66
- data/spec/mongoid/extensions/time_with_zone_spec.rb +17 -31
- data/spec/mongoid/factory_spec.rb +61 -1
- data/spec/mongoid/fields_spec.rb +321 -50
- data/spec/mongoid/findable_spec.rb +64 -29
- data/spec/mongoid/indexable/specification_spec.rb +2 -2
- data/spec/mongoid/indexable_spec.rb +16 -19
- data/spec/mongoid/interceptable_spec.rb +584 -5
- data/spec/mongoid/interceptable_spec_models.rb +235 -4
- data/spec/mongoid/matcher/extract_attribute_spec.rb +1 -5
- data/spec/mongoid/mongoizable_spec.rb +285 -0
- data/spec/mongoid/persistable/creatable_spec.rb +2 -2
- data/spec/mongoid/persistable/deletable_spec.rb +2 -2
- data/spec/mongoid/persistable/destroyable_spec.rb +2 -2
- data/spec/mongoid/persistable/upsertable_spec.rb +14 -0
- data/spec/mongoid/persistence_context_spec.rb +24 -0
- data/spec/mongoid/query_cache_middleware_spec.rb +0 -18
- data/spec/mongoid/query_cache_spec.rb +0 -154
- data/spec/mongoid/reloadable_spec.rb +35 -2
- data/spec/mongoid/scopable_spec.rb +36 -34
- data/spec/mongoid/shardable_models.rb +0 -14
- data/spec/mongoid/shardable_spec.rb +61 -153
- data/spec/mongoid/stateful_spec.rb +28 -0
- data/spec/mongoid/timestamps_spec.rb +390 -0
- data/spec/mongoid/timestamps_spec_models.rb +67 -0
- data/spec/mongoid/touchable_spec.rb +116 -0
- data/spec/mongoid/touchable_spec_models.rb +12 -8
- data/spec/mongoid/traversable_spec.rb +4 -11
- data/spec/mongoid/validatable/presence_spec.rb +1 -1
- data/spec/mongoid/validatable/uniqueness_spec.rb +60 -31
- data/spec/mongoid/warnings_spec.rb +35 -0
- data/spec/mongoid_spec.rb +1 -7
- data/spec/rails/controller_extension/controller_runtime_spec.rb +2 -2
- data/spec/rails/mongoid_spec.rb +4 -16
- data/spec/shared/lib/mrss/event_subscriber.rb +5 -15
- data/spec/shared/lib/mrss/lite_constraints.rb +0 -8
- data/spec/shared/shlib/server.sh +5 -5
- data/spec/support/constraints.rb +24 -0
- data/spec/support/macros.rb +30 -0
- data/spec/support/models/augmentation.rb +12 -0
- data/spec/support/models/band.rb +3 -0
- data/spec/support/models/catalog.rb +24 -0
- data/spec/support/models/circus.rb +3 -0
- data/spec/support/models/fanatic.rb +8 -0
- data/spec/support/models/implant.rb +9 -0
- data/spec/support/models/label.rb +2 -0
- data/spec/support/models/passport.rb +9 -0
- data/spec/support/models/person.rb +1 -0
- data/spec/support/models/player.rb +2 -0
- data/spec/support/models/powerup.rb +12 -0
- data/spec/support/models/registry.rb +1 -0
- data/spec/support/models/school.rb +14 -0
- data/spec/support/models/shield.rb +18 -0
- data/spec/support/models/student.rb +14 -0
- data/spec/support/models/weapon.rb +12 -0
- data.tar.gz.sig +0 -0
- metadata +669 -638
- metadata.gz.sig +0 -0
- data/lib/mongoid/errors/eager_load.rb +0 -23
- data/lib/mongoid/errors/invalid_value.rb +0 -17
- data/spec/mongoid/errors/eager_load_spec.rb +0 -31
@@ -25,11 +25,25 @@ module Mongoid
|
|
25
25
|
#
|
26
26
|
# @return [ Document ] A single document.
|
27
27
|
def build(base, object, _type = nil, selected_fields = nil)
|
28
|
-
|
29
|
-
|
30
|
-
|
28
|
+
if object.is_a?(Hash)
|
29
|
+
if _loading? && base.persisted?
|
30
|
+
Factory.execute_from_db(klass, object, nil, selected_fields, execute_callbacks: false)
|
31
|
+
else
|
32
|
+
Factory.build(klass, object)
|
33
|
+
end
|
31
34
|
else
|
32
|
-
|
35
|
+
clear_associated(object)
|
36
|
+
object
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
private
|
41
|
+
|
42
|
+
def clear_associated(doc)
|
43
|
+
if doc && (inv = inverse(doc))
|
44
|
+
if associated = doc.ivar(inv)
|
45
|
+
associated.substitute(nil)
|
46
|
+
end
|
33
47
|
end
|
34
48
|
end
|
35
49
|
end
|
@@ -31,6 +31,7 @@ module Mongoid
|
|
31
31
|
characterize_one(_target)
|
32
32
|
bind_one
|
33
33
|
characterize_one(_target)
|
34
|
+
update_attributes_hash(_target)
|
34
35
|
_base._reset_memoized_descendants!
|
35
36
|
_target.save if persistable?
|
36
37
|
end
|
@@ -80,11 +81,15 @@ module Mongoid
|
|
80
81
|
end
|
81
82
|
end
|
82
83
|
unbind_one
|
83
|
-
|
84
|
+
unless replacement
|
85
|
+
update_attributes_hash(replacement)
|
86
|
+
return nil
|
87
|
+
end
|
84
88
|
replacement = Factory.build(klass, replacement) if replacement.is_a?(::Hash)
|
85
89
|
self._target = replacement
|
86
|
-
bind_one
|
87
90
|
characterize_one(_target)
|
91
|
+
bind_one
|
92
|
+
update_attributes_hash(_target)
|
88
93
|
_target.save if persistable?
|
89
94
|
end
|
90
95
|
self
|
@@ -112,6 +117,20 @@ module Mongoid
|
|
112
117
|
_base.persisted? && !_binding? && !_building? && !_assigning?
|
113
118
|
end
|
114
119
|
|
120
|
+
# Update the _base's attributes hash with the _target's attributes
|
121
|
+
#
|
122
|
+
# @param replacement [ Document | nil ] The doc to use to update the
|
123
|
+
# attributes hash.
|
124
|
+
#
|
125
|
+
# @api private
|
126
|
+
def update_attributes_hash(replacement)
|
127
|
+
if replacement
|
128
|
+
_base.attributes.merge!(_association.store_as => replacement.attributes)
|
129
|
+
else
|
130
|
+
_base.attributes.delete(_association.store_as)
|
131
|
+
end
|
132
|
+
end
|
133
|
+
|
115
134
|
class << self
|
116
135
|
|
117
136
|
# Returns true if the association is an embedded one. In this case
|
@@ -34,6 +34,7 @@ module Mongoid
|
|
34
34
|
#
|
35
35
|
# @api private
|
36
36
|
class_attribute :aliased_associations
|
37
|
+
|
37
38
|
self.embedded = false
|
38
39
|
self.embedded_relations = BSON::Document.new
|
39
40
|
self.relations = BSON::Document.new
|
@@ -216,7 +217,7 @@ module Mongoid
|
|
216
217
|
Association::MACRO_MAPPING[macro_name].new(self, name, options, &block).tap do |assoc|
|
217
218
|
assoc.setup!
|
218
219
|
self.relations = self.relations.merge(name => assoc)
|
219
|
-
if assoc.respond_to?(:store_as) && assoc.store_as != name
|
220
|
+
if assoc.embedded? && assoc.respond_to?(:store_as) && assoc.store_as != name
|
220
221
|
self.aliased_associations[assoc.store_as] = name
|
221
222
|
end
|
222
223
|
end
|
@@ -61,6 +61,11 @@ module Mongoid
|
|
61
61
|
attributes.map { |attrs| create!(attrs, type, &block) }
|
62
62
|
else
|
63
63
|
doc = build(attributes, type, &block)
|
64
|
+
|
65
|
+
Array(doc).each do |doc|
|
66
|
+
doc.try(:run_pending_callbacks)
|
67
|
+
end
|
68
|
+
|
64
69
|
_base.persisted? ? doc.save! : raise_unsaved(doc)
|
65
70
|
doc
|
66
71
|
end
|
@@ -134,8 +134,9 @@ module Mongoid
|
|
134
134
|
# @param [ Proxy ] relation The association proxy.
|
135
135
|
# @param [ Document ] doc The document to delete.
|
136
136
|
def destroy_document(relation, doc)
|
137
|
+
res = doc.destroy unless doc.embedded? || doc.destroyed?
|
137
138
|
relation.delete(doc)
|
138
|
-
|
139
|
+
res
|
139
140
|
end
|
140
141
|
|
141
142
|
# Update the document.
|
@@ -169,6 +169,18 @@ module Mongoid
|
|
169
169
|
end
|
170
170
|
end
|
171
171
|
|
172
|
+
# Execute the before and after callbacks for the given method.
|
173
|
+
#
|
174
|
+
# @param [ Symbol ] name The name of the callbacks to execute.
|
175
|
+
#
|
176
|
+
# @return [ Object ] The result of the given block
|
177
|
+
def execute_callbacks_around(name, doc)
|
178
|
+
execute_callback :"before_#{name.to_s}", doc
|
179
|
+
yield.tap do
|
180
|
+
execute_callback :"after_#{name.to_s}", doc
|
181
|
+
end
|
182
|
+
end
|
183
|
+
|
172
184
|
class << self
|
173
185
|
|
174
186
|
# Apply ordering to the criteria if it was defined on the association.
|
@@ -56,7 +56,8 @@ module Mongoid
|
|
56
56
|
__autosaving__ do
|
57
57
|
if assoc_value = ivar(association.name)
|
58
58
|
Array(assoc_value).each do |doc|
|
59
|
-
doc.
|
59
|
+
pc = doc.persistence_context? ? doc.persistence_context : persistence_context
|
60
|
+
doc.with(pc) do |d|
|
60
61
|
d.save
|
61
62
|
end
|
62
63
|
end
|
@@ -64,7 +65,7 @@ module Mongoid
|
|
64
65
|
end
|
65
66
|
end
|
66
67
|
end
|
67
|
-
klass.
|
68
|
+
klass.after_persist_parent save_method, unless: :autosaved?
|
68
69
|
end
|
69
70
|
end
|
70
71
|
end
|
@@ -153,7 +153,7 @@ module Mongoid
|
|
153
153
|
create_foreign_key_field!
|
154
154
|
setup_index!
|
155
155
|
define_touchable!
|
156
|
-
@owner_class.validates_associated(name) if validate?
|
156
|
+
@owner_class.validates_associated(name) if validate? || require_association?
|
157
157
|
@owner_class.validates(name, presence: true) if require_association?
|
158
158
|
end
|
159
159
|
|
@@ -97,18 +97,18 @@ module Mongoid
|
|
97
97
|
|
98
98
|
association.inverse_class.tap do |klass|
|
99
99
|
klass.after_update do
|
100
|
-
|
101
|
-
foreign_key = association.foreign_key
|
100
|
+
foreign_key = association.foreign_key
|
102
101
|
|
103
|
-
|
104
|
-
|
102
|
+
if send("#{foreign_key}_previously_changed?")
|
103
|
+
original, current = send("#{foreign_key}_previous_change")
|
105
104
|
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
end
|
105
|
+
unless original.nil?
|
106
|
+
association.klass.with(persistence_context) do |_class|
|
107
|
+
_class.decrement_counter(cache_column, original)
|
110
108
|
end
|
109
|
+
end
|
111
110
|
|
111
|
+
if record = __send__(name)
|
112
112
|
unless current.nil?
|
113
113
|
record[cache_column] = (record[cache_column] || 0) + 1
|
114
114
|
record.class.with(record.persistence_context) do |_class|
|
@@ -28,10 +28,34 @@ module Mongoid
|
|
28
28
|
docs = args.flatten
|
29
29
|
return concat(docs) if docs.size > 1
|
30
30
|
if doc = docs.first
|
31
|
-
append(doc)
|
32
|
-
|
33
|
-
|
34
|
-
|
31
|
+
append(doc) do
|
32
|
+
# We ignore the changes to the value for the foreign key in the
|
33
|
+
# changed_attributes hash in this block of code for two reasons:
|
34
|
+
#
|
35
|
+
# 1) The add_to_set method deletes the value for the foreign
|
36
|
+
# key in the changed_attributes hash, but if we enter this
|
37
|
+
# method with a value for the foreign key in the
|
38
|
+
# changed_attributes hash, then we want it to exist outside
|
39
|
+
# this method as well. It's used later on in the Syncable
|
40
|
+
# module to set the inverse foreign keys.
|
41
|
+
# 2) The reset_unloaded method accesses the value for the foreign
|
42
|
+
# key on _base, which causes it to get added to the
|
43
|
+
# changed_attributes hash. This happens because when reading
|
44
|
+
# a "resizable" attribute, it is automatically added to the
|
45
|
+
# changed_attributes hash. This is true only for the foreign
|
46
|
+
# key value for HABTM associations as the other associations
|
47
|
+
# use strings for their foreign key values. For consistency
|
48
|
+
# with the other associations, we ignore this addition to
|
49
|
+
# the changed_attributes hash.
|
50
|
+
# See MONGOID-4843 for a longer discussion about this.
|
51
|
+
reset_foreign_key_changes do
|
52
|
+
_base.add_to_set(foreign_key => doc.public_send(_association.primary_key))
|
53
|
+
|
54
|
+
if child_persistable?(doc)
|
55
|
+
doc.save
|
56
|
+
end
|
57
|
+
reset_unloaded
|
58
|
+
end
|
35
59
|
end
|
36
60
|
end
|
37
61
|
unsynced(_base, foreign_key) and self
|
@@ -81,12 +105,13 @@ module Mongoid
|
|
81
105
|
#
|
82
106
|
# @return [ Document ] The new document.
|
83
107
|
def build(attributes = {}, type = nil)
|
84
|
-
doc = Factory.
|
85
|
-
_base.public_send(foreign_key).push(doc.public_send(_association.primary_key))
|
108
|
+
doc = Factory.execute_build(type || klass, attributes, execute_callbacks: false)
|
86
109
|
append(doc)
|
87
110
|
doc.apply_post_processed_defaults
|
111
|
+
_base.public_send(foreign_key).push(doc.public_send(_association.primary_key))
|
88
112
|
unsynced(doc, inverse_foreign_key)
|
89
113
|
yield(doc) if block_given?
|
114
|
+
doc.run_pending_callbacks
|
90
115
|
doc
|
91
116
|
end
|
92
117
|
|
@@ -178,6 +203,7 @@ module Mongoid
|
|
178
203
|
push(replacement.compact.uniq)
|
179
204
|
else
|
180
205
|
reset_unloaded
|
206
|
+
clear_foreign_key_changes
|
181
207
|
end
|
182
208
|
self
|
183
209
|
end
|
@@ -195,6 +221,32 @@ module Mongoid
|
|
195
221
|
|
196
222
|
private
|
197
223
|
|
224
|
+
# Clears the foreign key from the changed_attributes hash.
|
225
|
+
#
|
226
|
+
# This is, in general, used to clear the foreign key from the
|
227
|
+
# changed_attributes hash for consistency with the other referenced
|
228
|
+
# associations.
|
229
|
+
#
|
230
|
+
# @api private
|
231
|
+
def clear_foreign_key_changes
|
232
|
+
_base.changed_attributes.delete(foreign_key)
|
233
|
+
end
|
234
|
+
|
235
|
+
# Reset the value in the changed_attributes hash for the foreign key
|
236
|
+
# to its value before executing the given block.
|
237
|
+
#
|
238
|
+
# @api private
|
239
|
+
def reset_foreign_key_changes
|
240
|
+
if _base.changed_attributes.key?(foreign_key)
|
241
|
+
fk = _base.changed_attributes[foreign_key].dup
|
242
|
+
yield if block_given?
|
243
|
+
_base.changed_attributes[foreign_key] = fk
|
244
|
+
else
|
245
|
+
yield if block_given?
|
246
|
+
clear_foreign_key_changes
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
198
250
|
# Appends the document to the target array, updating the index on the
|
199
251
|
# document at the same time.
|
200
252
|
#
|
@@ -203,11 +255,12 @@ module Mongoid
|
|
203
255
|
#
|
204
256
|
# @param [ Document ] document The document to append to the target.
|
205
257
|
def append(document)
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
258
|
+
execute_callbacks_around(:add, document) do
|
259
|
+
_target.push(document)
|
260
|
+
characterize_one(document)
|
261
|
+
bind_one(document)
|
262
|
+
yield if block_given?
|
263
|
+
end
|
211
264
|
end
|
212
265
|
|
213
266
|
# Instantiate the binding associated with this association.
|
@@ -119,6 +119,8 @@ module Mongoid
|
|
119
119
|
@options[:inverse_foreign_key]
|
120
120
|
elsif @options.key?(:inverse_of)
|
121
121
|
inverse_of ? "#{inverse_of.to_s.singularize}#{FOREIGN_KEY_SUFFIX}" : nil
|
122
|
+
elsif inv = inverse_association&.foreign_key
|
123
|
+
inv
|
122
124
|
else
|
123
125
|
"#{inverse_class_name.demodulize.underscore}#{FOREIGN_KEY_SUFFIX}"
|
124
126
|
end
|
@@ -219,7 +221,7 @@ module Mongoid
|
|
219
221
|
def synced_save
|
220
222
|
assoc = self
|
221
223
|
inverse_class.set_callback(
|
222
|
-
:
|
224
|
+
:persist_parent,
|
223
225
|
:after,
|
224
226
|
if: ->(doc){ doc._syncable?(assoc) }
|
225
227
|
) do |doc|
|
@@ -228,6 +230,7 @@ module Mongoid
|
|
228
230
|
end
|
229
231
|
|
230
232
|
def create_foreign_key_field!
|
233
|
+
inverse_class.aliased_associations[foreign_key] = name.to_s
|
231
234
|
@owner_class.field(
|
232
235
|
foreign_key,
|
233
236
|
type: FOREIGN_KEY_FIELD_TYPE,
|
@@ -240,19 +240,15 @@ module Mongoid
|
|
240
240
|
# @note Automatically adding a sort on _id when no other sort is
|
241
241
|
# defined on the criteria has the potential to cause bad performance issues.
|
242
242
|
# If you experience unexpected poor performance when using #first or #last,
|
243
|
-
# use
|
244
|
-
# Be aware that #
|
243
|
+
# use #take instead.
|
244
|
+
# Be aware that #take won't guarantee order.
|
245
245
|
#
|
246
|
-
# @param [ Integer
|
247
|
-
# return, or a hash of options.
|
248
|
-
#
|
249
|
-
# @option limit_or_opts [ :none ] :id_sort This option is deprecated.
|
250
|
-
# Don't apply a sort on _id if no other sort is defined on the criteria.
|
246
|
+
# @param [ Integer ] limit The number of documents to return.
|
251
247
|
#
|
252
248
|
# @return [ Document ] The first document found.
|
253
|
-
def first(
|
249
|
+
def first(limit = nil)
|
254
250
|
_loaded.try(:values).try(:first) ||
|
255
|
-
_added[(ul = _unloaded.try(:first,
|
251
|
+
_added[(ul = _unloaded.try(:first, limit)).try(:_id)] ||
|
256
252
|
ul ||
|
257
253
|
_added.values.try(:first)
|
258
254
|
end
|
@@ -329,20 +325,16 @@ module Mongoid
|
|
329
325
|
# @note Automatically adding a sort on _id when no other sort is
|
330
326
|
# defined on the criteria has the potential to cause bad performance issues.
|
331
327
|
# If you experience unexpected poor performance when using #first or #last,
|
332
|
-
# use
|
333
|
-
# Be aware that #
|
334
|
-
#
|
335
|
-
# @param [ Integer | Hash ] limit_or_opts The number of documents to
|
336
|
-
# return, or a hash of options.
|
328
|
+
# use #take instead.
|
329
|
+
# Be aware that #take won't guarantee order.
|
337
330
|
#
|
338
|
-
# @
|
339
|
-
# Don't apply a sort on _id if no other sort is defined on the criteria.
|
331
|
+
# @param [ Integer ] limit The number of documents to return.
|
340
332
|
#
|
341
333
|
# @return [ Document ] The last document found.
|
342
|
-
def last(
|
334
|
+
def last(limit = nil)
|
343
335
|
_added.values.try(:last) ||
|
344
336
|
_loaded.try(:values).try(:last) ||
|
345
|
-
_added[(ul = _unloaded.try(:last,
|
337
|
+
_added[(ul = _unloaded.try(:last, limit)).try(:_id)] ||
|
346
338
|
ul
|
347
339
|
end
|
348
340
|
|
@@ -71,10 +71,11 @@ module Mongoid
|
|
71
71
|
#
|
72
72
|
# @return [ Document ] The new document.
|
73
73
|
def build(attributes = {}, type = nil)
|
74
|
-
doc = Factory.
|
74
|
+
doc = Factory.execute_build(type || klass, attributes, execute_callbacks: false)
|
75
75
|
append(doc)
|
76
76
|
doc.apply_post_processed_defaults
|
77
77
|
yield(doc) if block_given?
|
78
|
+
doc.run_pending_callbacks
|
78
79
|
doc.run_callbacks(:build) { doc }
|
79
80
|
doc
|
80
81
|
end
|
@@ -92,13 +93,15 @@ module Mongoid
|
|
92
93
|
#
|
93
94
|
# @return [ Document ] The matching document.
|
94
95
|
def delete(document)
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
96
|
+
execute_callbacks_around(:remove, document) do
|
97
|
+
_target.delete(document) do |doc|
|
98
|
+
if doc
|
99
|
+
unbind_one(doc)
|
100
|
+
cascade!(doc) if !_assigning?
|
101
|
+
end
|
102
|
+
end.tap do
|
103
|
+
reset_unloaded
|
100
104
|
end
|
101
|
-
execute_callback :after_remove, doc
|
102
105
|
end
|
103
106
|
end
|
104
107
|
|
@@ -469,8 +472,8 @@ module Mongoid
|
|
469
472
|
selector = conditions || {}
|
470
473
|
removed = klass.send(method, selector.merge!(criteria.selector))
|
471
474
|
_target.delete_if do |doc|
|
472
|
-
|
473
|
-
unbind_one(doc)
|
475
|
+
doc._matches?(selector).tap do |b|
|
476
|
+
unbind_one(doc) if b
|
474
477
|
end
|
475
478
|
end
|
476
479
|
removed
|
@@ -49,17 +49,14 @@ module Mongoid
|
|
49
49
|
#
|
50
50
|
# @return [ One ] The association.
|
51
51
|
def substitute(replacement)
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
send(_association.dependent)
|
61
|
-
else
|
62
|
-
save if persisted?
|
52
|
+
if self != replacement
|
53
|
+
unbind_one
|
54
|
+
if persistable?
|
55
|
+
if _association.destructive?
|
56
|
+
send(_association.dependent)
|
57
|
+
else
|
58
|
+
save if persisted?
|
59
|
+
end
|
63
60
|
end
|
64
61
|
end
|
65
62
|
HasOne::Proxy.new(_base, replacement, _association) if replacement
|
@@ -67,8 +67,8 @@ module Mongoid
|
|
67
67
|
#
|
68
68
|
# @return [ Object ] The updated values.
|
69
69
|
def update_inverse_keys(association)
|
70
|
-
if
|
71
|
-
old, new =
|
70
|
+
if previous_changes.has_key?(association.foreign_key)
|
71
|
+
old, new = previous_changes[association.foreign_key]
|
72
72
|
adds, subs = new - (old || []), (old || []) - new
|
73
73
|
|
74
74
|
# If we are autosaving we don't want a duplicate to get added - the
|
@@ -242,7 +242,7 @@ module Mongoid
|
|
242
242
|
#
|
243
243
|
# @return [ String ] The foreign key check.
|
244
244
|
def foreign_key_check
|
245
|
-
@foreign_key_check ||= "#{foreign_key}
|
245
|
+
@foreign_key_check ||= "#{foreign_key}_previously_changed?" if (stores_foreign_key? && foreign_key)
|
246
246
|
end
|
247
247
|
|
248
248
|
# Create an association proxy object using the owner and target.
|
@@ -299,6 +299,35 @@ module Mongoid
|
|
299
299
|
end
|
300
300
|
end
|
301
301
|
|
302
|
+
# @return [ Array<String> ] The associations above this one in the inclusion tree.
|
303
|
+
attr_accessor :parent_inclusions
|
304
|
+
|
305
|
+
def parent_inclusions
|
306
|
+
@parent_inclusions ||= []
|
307
|
+
end
|
308
|
+
|
309
|
+
# Is this association an embeds_many or has_many association?
|
310
|
+
#
|
311
|
+
# @return [ true | false ] true if it is a *_many association, false if not.
|
312
|
+
def many?
|
313
|
+
[Referenced::HasMany, Embedded::EmbedsMany].any? { |a| self.is_a?(a) }
|
314
|
+
end
|
315
|
+
|
316
|
+
# Is this association an embeds_one or has_one association?
|
317
|
+
#
|
318
|
+
# @return [ true | false ] true if it is a *_one association, false if not.
|
319
|
+
def one?
|
320
|
+
[Referenced::HasOne, Embedded::EmbedsOne].any? { |a| self.is_a?(a) }
|
321
|
+
end
|
322
|
+
|
323
|
+
# Is this association an embedded_in or belongs_to association?
|
324
|
+
#
|
325
|
+
# @return [ true | false ] true if it is an embedded_in or belongs_to
|
326
|
+
# association, false if not.
|
327
|
+
def in_to?
|
328
|
+
[Referenced::BelongsTo, Embedded::EmbeddedIn].any? { |a| self.is_a?(a) }
|
329
|
+
end
|
330
|
+
|
302
331
|
private
|
303
332
|
|
304
333
|
# Gets the model classes with inverse associations of this model. This is used to determine
|
@@ -404,10 +433,15 @@ module Mongoid
|
|
404
433
|
def namespace_hierarchy(mod)
|
405
434
|
parent = Object
|
406
435
|
hier = [parent]
|
407
|
-
|
408
|
-
|
409
|
-
|
436
|
+
|
437
|
+
# name is not present on anonymous modules
|
438
|
+
if mod.name
|
439
|
+
mod.name.split('::').each do |part|
|
440
|
+
parent = parent.const_get(part)
|
441
|
+
hier << parent
|
442
|
+
end
|
410
443
|
end
|
444
|
+
|
411
445
|
hier.reverse
|
412
446
|
end
|
413
447
|
|
@@ -43,11 +43,18 @@ module Mongoid
|
|
43
43
|
# @return [ true, false ] True if pending, false if not.
|
44
44
|
def pending_attribute?(key, value)
|
45
45
|
name = key.to_s
|
46
|
-
|
46
|
+
|
47
|
+
aliased = if aliased_associations.key?(name)
|
48
|
+
aliased_associations[name]
|
49
|
+
else
|
50
|
+
name
|
51
|
+
end
|
52
|
+
|
53
|
+
if relations.has_key?(aliased)
|
47
54
|
pending_relations[name] = value
|
48
55
|
return true
|
49
56
|
end
|
50
|
-
if nested_attributes.has_key?(
|
57
|
+
if nested_attributes.has_key?(aliased)
|
51
58
|
pending_nested[name] = value
|
52
59
|
return true
|
53
60
|
end
|