mongoid 7.5.3 → 8.0.1

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 (295) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/README.md +3 -3
  4. data/lib/config/locales/en.yml +46 -30
  5. data/lib/mongoid/association/accessors.rb +32 -3
  6. data/lib/mongoid/association/bindable.rb +48 -0
  7. data/lib/mongoid/association/builders.rb +4 -2
  8. data/lib/mongoid/association/eager_loadable.rb +29 -7
  9. data/lib/mongoid/association/embedded/batchable.rb +28 -5
  10. data/lib/mongoid/association/embedded/embedded_in/binding.rb +24 -2
  11. data/lib/mongoid/association/embedded/embedded_in.rb +2 -1
  12. data/lib/mongoid/association/embedded/embeds_many/binding.rb +1 -0
  13. data/lib/mongoid/association/embedded/embeds_many/buildable.rb +1 -1
  14. data/lib/mongoid/association/embedded/embeds_many/proxy.rb +40 -18
  15. data/lib/mongoid/association/embedded/embeds_one/buildable.rb +18 -4
  16. data/lib/mongoid/association/embedded/embeds_one/proxy.rb +21 -2
  17. data/lib/mongoid/association/macros.rb +2 -1
  18. data/lib/mongoid/association/many.rb +5 -0
  19. data/lib/mongoid/association/nested/many.rb +2 -1
  20. data/lib/mongoid/association/proxy.rb +12 -0
  21. data/lib/mongoid/association/referenced/auto_save.rb +3 -2
  22. data/lib/mongoid/association/referenced/belongs_to/binding.rb +1 -0
  23. data/lib/mongoid/association/referenced/belongs_to/buildable.rb +1 -1
  24. data/lib/mongoid/association/referenced/belongs_to.rb +1 -1
  25. data/lib/mongoid/association/referenced/counter_cache.rb +8 -8
  26. data/lib/mongoid/association/referenced/has_and_belongs_to_many/proxy.rb +64 -11
  27. data/lib/mongoid/association/referenced/has_and_belongs_to_many.rb +4 -1
  28. data/lib/mongoid/association/referenced/has_many/enumerable.rb +10 -18
  29. data/lib/mongoid/association/referenced/has_many/proxy.rb +12 -9
  30. data/lib/mongoid/association/referenced/has_one/buildable.rb +1 -1
  31. data/lib/mongoid/association/referenced/has_one/proxy.rb +8 -11
  32. data/lib/mongoid/association/referenced/syncable.rb +2 -2
  33. data/lib/mongoid/association/relatable.rb +38 -4
  34. data/lib/mongoid/attributes/processing.rb +9 -2
  35. data/lib/mongoid/attributes.rb +30 -27
  36. data/lib/mongoid/cacheable.rb +2 -2
  37. data/lib/mongoid/changeable.rb +37 -2
  38. data/lib/mongoid/clients/options.rb +4 -0
  39. data/lib/mongoid/clients/sessions.rb +2 -14
  40. data/lib/mongoid/config.rb +15 -11
  41. data/lib/mongoid/contextual/aggregable/memory.rb +23 -15
  42. data/lib/mongoid/contextual/aggregable/mongo.rb +1 -1
  43. data/lib/mongoid/contextual/map_reduce.rb +2 -2
  44. data/lib/mongoid/contextual/memory.rb +55 -28
  45. data/lib/mongoid/contextual/mongo.rb +173 -262
  46. data/lib/mongoid/contextual/none.rb +33 -15
  47. data/lib/mongoid/copyable.rb +32 -8
  48. data/lib/mongoid/criteria/includable.rb +24 -20
  49. data/lib/mongoid/criteria/marshalable.rb +10 -2
  50. data/lib/mongoid/criteria/queryable/extensions/array.rb +2 -15
  51. data/lib/mongoid/criteria/queryable/extensions/big_decimal.rb +25 -4
  52. data/lib/mongoid/criteria/queryable/extensions/boolean.rb +1 -1
  53. data/lib/mongoid/criteria/queryable/extensions/date.rb +6 -1
  54. data/lib/mongoid/criteria/queryable/extensions/date_time.rb +6 -1
  55. data/lib/mongoid/criteria/queryable/extensions/hash.rb +0 -16
  56. data/lib/mongoid/criteria/queryable/extensions/numeric.rb +1 -1
  57. data/lib/mongoid/criteria/queryable/extensions/object.rb +2 -1
  58. data/lib/mongoid/criteria/queryable/extensions/range.rb +13 -5
  59. data/lib/mongoid/criteria/queryable/extensions/regexp.rb +1 -1
  60. data/lib/mongoid/criteria/queryable/extensions/symbol.rb +3 -1
  61. data/lib/mongoid/criteria/queryable/extensions/time.rb +6 -1
  62. data/lib/mongoid/criteria/queryable/extensions/time_with_zone.rb +6 -1
  63. data/lib/mongoid/criteria/queryable/optional.rb +3 -9
  64. data/lib/mongoid/criteria/queryable/options.rb +1 -1
  65. data/lib/mongoid/criteria/queryable/selectable.rb +2 -24
  66. data/lib/mongoid/criteria/queryable/selector.rb +89 -4
  67. data/lib/mongoid/criteria/queryable/smash.rb +39 -6
  68. data/lib/mongoid/criteria/queryable.rb +11 -6
  69. data/lib/mongoid/criteria.rb +1 -28
  70. data/lib/mongoid/deprecable.rb +36 -0
  71. data/lib/mongoid/deprecation.rb +25 -0
  72. data/lib/mongoid/document.rb +88 -33
  73. data/lib/mongoid/equality.rb +4 -4
  74. data/lib/mongoid/errors/document_not_found.rb +6 -2
  75. data/lib/mongoid/errors/invalid_dot_dollar_assignment.rb +23 -0
  76. data/lib/mongoid/errors/invalid_field.rb +5 -1
  77. data/lib/mongoid/errors/invalid_field_type.rb +26 -0
  78. data/lib/mongoid/errors/too_many_nested_attribute_records.rb +1 -1
  79. data/lib/mongoid/errors.rb +2 -2
  80. data/lib/mongoid/extensions/array.rb +8 -6
  81. data/lib/mongoid/extensions/big_decimal.rb +29 -10
  82. data/lib/mongoid/extensions/binary.rb +42 -0
  83. data/lib/mongoid/extensions/boolean.rb +8 -2
  84. data/lib/mongoid/extensions/date.rb +26 -20
  85. data/lib/mongoid/extensions/date_time.rb +1 -1
  86. data/lib/mongoid/extensions/float.rb +4 -5
  87. data/lib/mongoid/extensions/hash.rb +12 -5
  88. data/lib/mongoid/extensions/integer.rb +4 -5
  89. data/lib/mongoid/extensions/object.rb +2 -0
  90. data/lib/mongoid/extensions/range.rb +41 -10
  91. data/lib/mongoid/extensions/regexp.rb +11 -4
  92. data/lib/mongoid/extensions/set.rb +11 -4
  93. data/lib/mongoid/extensions/string.rb +2 -13
  94. data/lib/mongoid/extensions/symbol.rb +3 -14
  95. data/lib/mongoid/extensions/time.rb +27 -16
  96. data/lib/mongoid/extensions/time_with_zone.rb +1 -2
  97. data/lib/mongoid/extensions.rb +1 -0
  98. data/lib/mongoid/factory.rb +42 -7
  99. data/lib/mongoid/fields/foreign_key.rb +7 -0
  100. data/lib/mongoid/fields/validators/macro.rb +3 -9
  101. data/lib/mongoid/fields.rb +49 -7
  102. data/lib/mongoid/findable.rb +21 -16
  103. data/lib/mongoid/indexable/specification.rb +1 -1
  104. data/lib/mongoid/indexable/validators/options.rb +4 -1
  105. data/lib/mongoid/interceptable.rb +69 -9
  106. data/lib/mongoid/persistable/creatable.rb +14 -5
  107. data/lib/mongoid/persistable/updatable.rb +12 -5
  108. data/lib/mongoid/persistence_context.rb +8 -42
  109. data/lib/mongoid/query_cache.rb +6 -258
  110. data/lib/mongoid/railties/controller_runtime.rb +1 -1
  111. data/lib/mongoid/reloadable.rb +7 -3
  112. data/lib/mongoid/scopable.rb +9 -11
  113. data/lib/mongoid/selectable.rb +1 -2
  114. data/lib/mongoid/shardable.rb +11 -35
  115. data/lib/mongoid/stateful.rb +27 -1
  116. data/lib/mongoid/timestamps/created.rb +1 -1
  117. data/lib/mongoid/timestamps/updated.rb +1 -1
  118. data/lib/mongoid/touchable.rb +2 -3
  119. data/lib/mongoid/traversable.rb +1 -0
  120. data/lib/mongoid/validatable/uniqueness.rb +2 -1
  121. data/lib/mongoid/version.rb +1 -1
  122. data/lib/mongoid/warnings.rb +3 -4
  123. data/lib/mongoid.rb +1 -0
  124. data/spec/config/mongoid.yml +16 -0
  125. data/spec/integration/app_spec.rb +8 -12
  126. data/spec/integration/associations/belongs_to_spec.rb +18 -0
  127. data/spec/integration/associations/embedded_spec.rb +15 -0
  128. data/spec/integration/associations/embeds_many_spec.rb +15 -2
  129. data/spec/integration/associations/embeds_one_spec.rb +18 -0
  130. data/spec/integration/associations/foreign_key_spec.rb +9 -0
  131. data/spec/integration/associations/has_and_belongs_to_many_spec.rb +21 -0
  132. data/spec/integration/associations/has_one_spec.rb +97 -1
  133. data/spec/integration/associations/scope_option_spec.rb +1 -1
  134. data/spec/integration/callbacks_models.rb +95 -1
  135. data/spec/integration/callbacks_spec.rb +226 -4
  136. data/spec/integration/criteria/range_spec.rb +95 -1
  137. data/spec/integration/discriminator_key_spec.rb +115 -76
  138. data/spec/integration/dots_and_dollars_spec.rb +277 -0
  139. data/spec/integration/i18n_fallbacks_spec.rb +1 -15
  140. data/spec/integration/matcher_examples_spec.rb +20 -13
  141. data/spec/integration/matcher_operator_data/type_decimal.yml +3 -2
  142. data/spec/integration/matcher_operator_spec.rb +3 -5
  143. data/spec/integration/persistence/range_field_spec.rb +350 -0
  144. data/spec/mongoid/association/counter_cache_spec.rb +1 -1
  145. data/spec/mongoid/association/depending_spec.rb +9 -9
  146. data/spec/mongoid/association/eager_spec.rb +2 -1
  147. data/spec/mongoid/association/embedded/embedded_in/binding_spec.rb +2 -1
  148. data/spec/mongoid/association/embedded/embedded_in/buildable_spec.rb +54 -0
  149. data/spec/mongoid/association/embedded/embedded_in/proxy_spec.rb +69 -9
  150. data/spec/mongoid/association/embedded/embeds_many/buildable_spec.rb +112 -0
  151. data/spec/mongoid/association/embedded/embeds_many/proxy_spec.rb +198 -8
  152. data/spec/mongoid/association/embedded/embeds_many_models.rb +36 -0
  153. data/spec/mongoid/association/embedded/embeds_many_query_spec.rb +12 -0
  154. data/spec/mongoid/association/embedded/embeds_many_spec.rb +68 -0
  155. data/spec/mongoid/association/embedded/embeds_one/buildable_spec.rb +25 -0
  156. data/spec/mongoid/association/embedded/embeds_one_models.rb +19 -0
  157. data/spec/mongoid/association/embedded/embeds_one_spec.rb +28 -0
  158. data/spec/mongoid/association/referenced/belongs_to/binding_spec.rb +2 -1
  159. data/spec/mongoid/association/referenced/belongs_to/buildable_spec.rb +54 -0
  160. data/spec/mongoid/association/referenced/belongs_to/proxy_spec.rb +15 -0
  161. data/spec/mongoid/association/referenced/belongs_to_models.rb +11 -0
  162. data/spec/mongoid/association/referenced/belongs_to_spec.rb +2 -2
  163. data/spec/mongoid/association/referenced/has_and_belongs_to_many/proxy_spec.rb +38 -5
  164. data/spec/mongoid/association/referenced/has_and_belongs_to_many_models.rb +25 -0
  165. data/spec/mongoid/association/referenced/has_and_belongs_to_many_spec.rb +35 -2
  166. data/spec/mongoid/association/referenced/has_many/buildable_spec.rb +109 -0
  167. data/spec/mongoid/association/referenced/has_many/enumerable_spec.rb +2 -56
  168. data/spec/mongoid/association/referenced/has_many/proxy_spec.rb +62 -13
  169. data/spec/mongoid/association/referenced/has_many_models.rb +3 -1
  170. data/spec/mongoid/association/referenced/has_many_spec.rb +25 -0
  171. data/spec/mongoid/association/referenced/has_one/buildable_spec.rb +2 -2
  172. data/spec/mongoid/association/referenced/has_one/proxy_spec.rb +107 -1
  173. data/spec/mongoid/association/referenced/has_one_models.rb +16 -0
  174. data/spec/mongoid/association/syncable_spec.rb +14 -0
  175. data/spec/mongoid/atomic/paths_spec.rb +0 -14
  176. data/spec/mongoid/attributes/nested_spec.rb +80 -11
  177. data/spec/mongoid/attributes/nested_spec_models.rb +48 -0
  178. data/spec/mongoid/attributes/projector_spec.rb +1 -5
  179. data/spec/mongoid/attributes_spec.rb +480 -27
  180. data/spec/mongoid/cacheable_spec.rb +3 -3
  181. data/spec/mongoid/changeable_spec.rb +130 -13
  182. data/spec/mongoid/clients/factory_spec.rb +23 -30
  183. data/spec/mongoid/clients/sessions_spec.rb +0 -38
  184. data/spec/mongoid/clients_spec.rb +2 -2
  185. data/spec/mongoid/config_spec.rb +52 -14
  186. data/spec/mongoid/contextual/aggregable/memory_spec.rb +396 -158
  187. data/spec/mongoid/contextual/aggregable/memory_table.yml +88 -0
  188. data/spec/mongoid/contextual/aggregable/memory_table_spec.rb +62 -0
  189. data/spec/mongoid/contextual/map_reduce_spec.rb +2 -16
  190. data/spec/mongoid/contextual/memory_spec.rb +521 -14
  191. data/spec/mongoid/contextual/mongo_spec.rb +566 -416
  192. data/spec/mongoid/contextual/none_spec.rb +11 -19
  193. data/spec/mongoid/copyable_spec.rb +451 -1
  194. data/spec/mongoid/criteria/findable_spec.rb +86 -210
  195. data/spec/mongoid/criteria/includable_spec.rb +1492 -0
  196. data/spec/mongoid/criteria/includable_spec_models.rb +54 -0
  197. data/spec/mongoid/criteria/marshalable_spec.rb +18 -1
  198. data/spec/mongoid/criteria/queryable/extensions/array_spec.rb +7 -19
  199. data/spec/mongoid/criteria/queryable/extensions/big_decimal_spec.rb +134 -26
  200. data/spec/mongoid/criteria/queryable/extensions/date_spec.rb +11 -0
  201. data/spec/mongoid/criteria/queryable/extensions/date_time_spec.rb +11 -0
  202. data/spec/mongoid/criteria/queryable/extensions/hash_spec.rb +0 -15
  203. data/spec/mongoid/criteria/queryable/extensions/numeric_spec.rb +73 -7
  204. data/spec/mongoid/criteria/queryable/extensions/time_spec.rb +11 -0
  205. data/spec/mongoid/criteria/queryable/extensions/time_with_zone_spec.rb +11 -0
  206. data/spec/mongoid/criteria/queryable/optional_spec.rb +0 -484
  207. data/spec/mongoid/criteria/queryable/selectable_logical_spec.rb +50 -0
  208. data/spec/mongoid/criteria/queryable/selectable_spec.rb +77 -85
  209. data/spec/mongoid/criteria/queryable/selector_spec.rb +14 -2
  210. data/spec/mongoid/criteria_spec.rb +469 -1201
  211. data/spec/mongoid/document_fields_spec.rb +173 -24
  212. data/spec/mongoid/document_spec.rb +32 -41
  213. data/spec/mongoid/equality_spec.rb +12 -12
  214. data/spec/mongoid/errors/document_not_found_spec.rb +29 -2
  215. data/spec/mongoid/errors/invalid_field_spec.rb +1 -1
  216. data/spec/mongoid/errors/invalid_field_type_spec.rb +55 -0
  217. data/spec/mongoid/errors/mongoid_error_spec.rb +3 -1
  218. data/spec/mongoid/errors/no_environment_spec.rb +3 -3
  219. data/spec/mongoid/errors/too_many_nested_attribute_records_spec.rb +1 -1
  220. data/spec/mongoid/extensions/array_spec.rb +16 -2
  221. data/spec/mongoid/extensions/big_decimal_spec.rb +697 -212
  222. data/spec/mongoid/extensions/binary_spec.rb +44 -9
  223. data/spec/mongoid/extensions/boolean_spec.rb +68 -82
  224. data/spec/mongoid/extensions/date_class_mongoize_spec.rb +7 -3
  225. data/spec/mongoid/extensions/date_spec.rb +71 -1
  226. data/spec/mongoid/extensions/date_time_spec.rb +15 -9
  227. data/spec/mongoid/extensions/float_spec.rb +48 -76
  228. data/spec/mongoid/extensions/hash_spec.rb +30 -0
  229. data/spec/mongoid/extensions/integer_spec.rb +45 -66
  230. data/spec/mongoid/extensions/range_spec.rb +255 -54
  231. data/spec/mongoid/extensions/regexp_spec.rb +58 -33
  232. data/spec/mongoid/extensions/set_spec.rb +106 -0
  233. data/spec/mongoid/extensions/string_spec.rb +53 -25
  234. data/spec/mongoid/extensions/symbol_spec.rb +18 -25
  235. data/spec/mongoid/extensions/time_spec.rb +634 -66
  236. data/spec/mongoid/extensions/time_with_zone_spec.rb +17 -31
  237. data/spec/mongoid/factory_spec.rb +61 -1
  238. data/spec/mongoid/fields_spec.rb +321 -50
  239. data/spec/mongoid/findable_spec.rb +64 -29
  240. data/spec/mongoid/indexable/specification_spec.rb +2 -2
  241. data/spec/mongoid/indexable_spec.rb +16 -19
  242. data/spec/mongoid/interceptable_spec.rb +584 -5
  243. data/spec/mongoid/interceptable_spec_models.rb +235 -4
  244. data/spec/mongoid/matcher/extract_attribute_spec.rb +1 -5
  245. data/spec/mongoid/mongoizable_spec.rb +285 -0
  246. data/spec/mongoid/persistable/creatable_spec.rb +2 -2
  247. data/spec/mongoid/persistable/deletable_spec.rb +2 -2
  248. data/spec/mongoid/persistable/destroyable_spec.rb +2 -2
  249. data/spec/mongoid/persistable/upsertable_spec.rb +14 -0
  250. data/spec/mongoid/persistence_context_spec.rb +24 -0
  251. data/spec/mongoid/query_cache_middleware_spec.rb +0 -18
  252. data/spec/mongoid/query_cache_spec.rb +0 -154
  253. data/spec/mongoid/reloadable_spec.rb +35 -2
  254. data/spec/mongoid/scopable_spec.rb +36 -34
  255. data/spec/mongoid/shardable_models.rb +0 -14
  256. data/spec/mongoid/shardable_spec.rb +61 -153
  257. data/spec/mongoid/stateful_spec.rb +28 -0
  258. data/spec/mongoid/timestamps_spec.rb +390 -0
  259. data/spec/mongoid/timestamps_spec_models.rb +67 -0
  260. data/spec/mongoid/touchable_spec.rb +116 -0
  261. data/spec/mongoid/touchable_spec_models.rb +12 -8
  262. data/spec/mongoid/traversable_spec.rb +4 -11
  263. data/spec/mongoid/validatable/presence_spec.rb +1 -1
  264. data/spec/mongoid/validatable/uniqueness_spec.rb +60 -31
  265. data/spec/mongoid/warnings_spec.rb +35 -0
  266. data/spec/mongoid_spec.rb +1 -7
  267. data/spec/rails/controller_extension/controller_runtime_spec.rb +2 -2
  268. data/spec/rails/mongoid_spec.rb +4 -16
  269. data/spec/shared/lib/mrss/event_subscriber.rb +5 -15
  270. data/spec/shared/lib/mrss/lite_constraints.rb +0 -8
  271. data/spec/shared/shlib/server.sh +5 -5
  272. data/spec/support/constraints.rb +24 -0
  273. data/spec/support/macros.rb +30 -0
  274. data/spec/support/models/augmentation.rb +12 -0
  275. data/spec/support/models/band.rb +3 -0
  276. data/spec/support/models/catalog.rb +24 -0
  277. data/spec/support/models/circus.rb +3 -0
  278. data/spec/support/models/fanatic.rb +8 -0
  279. data/spec/support/models/implant.rb +9 -0
  280. data/spec/support/models/label.rb +2 -0
  281. data/spec/support/models/passport.rb +9 -0
  282. data/spec/support/models/person.rb +1 -0
  283. data/spec/support/models/player.rb +2 -0
  284. data/spec/support/models/powerup.rb +12 -0
  285. data/spec/support/models/registry.rb +1 -0
  286. data/spec/support/models/school.rb +14 -0
  287. data/spec/support/models/shield.rb +18 -0
  288. data/spec/support/models/student.rb +14 -0
  289. data/spec/support/models/weapon.rb +12 -0
  290. data.tar.gz.sig +0 -0
  291. metadata +669 -638
  292. metadata.gz.sig +0 -0
  293. data/lib/mongoid/errors/eager_load.rb +0 -23
  294. data/lib/mongoid/errors/invalid_value.rb +0 -17
  295. 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
- return object unless object.is_a?(Hash)
29
- if _loading? && base.persisted?
30
- Factory.from_db(klass, object, nil, selected_fields)
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
- Factory.build(klass, object)
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
- return nil unless replacement
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
- doc.destroy unless doc.embedded? || doc.destroyed?
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.with(persistence_context) do |d|
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.after_save save_method, unless: :autosaved?
68
+ klass.after_persist_parent save_method, unless: :autosaved?
68
69
  end
69
70
  end
70
71
  end
@@ -28,6 +28,7 @@ module Mongoid
28
28
  if _base.referenced_many?
29
29
  _target.__send__(inverse).push(_base)
30
30
  else
31
+ remove_associated(_target)
31
32
  _target.set_relation(inverse, _base)
32
33
  end
33
34
  end
@@ -28,7 +28,7 @@ module Mongoid
28
28
  private
29
29
 
30
30
  def execute_query(object, type)
31
- query_criteria(object, type).limit(1).first(id_sort: :none)
31
+ query_criteria(object, type).take
32
32
  end
33
33
 
34
34
  def query_criteria(object, type)
@@ -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
- if record = __send__(name)
101
- foreign_key = association.foreign_key
100
+ foreign_key = association.foreign_key
102
101
 
103
- if attribute_changed?(foreign_key)
104
- original, current = attribute_change(foreign_key)
102
+ if send("#{foreign_key}_previously_changed?")
103
+ original, current = send("#{foreign_key}_previous_change")
105
104
 
106
- unless original.nil?
107
- record.class.with(persistence_context) do |_class|
108
- _class.decrement_counter(cache_column, original)
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
- _base.add_to_set(foreign_key => doc.public_send(_association.primary_key))
33
- if child_persistable?(doc)
34
- doc.save
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.build(type || klass, attributes)
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
- execute_callback :before_add, document
207
- _target.push(document)
208
- characterize_one(document)
209
- bind_one(document)
210
- execute_callback :after_add, document
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
- :save,
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 the option { id_sort: :none }.
244
- # Be aware that #first/#last won't guarantee order in this case.
243
+ # use #take instead.
244
+ # Be aware that #take won't guarantee order.
245
245
  #
246
- # @param [ Integer | Hash ] limit_or_opts The number of documents to
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(limit_or_opts = nil)
249
+ def first(limit = nil)
254
250
  _loaded.try(:values).try(:first) ||
255
- _added[(ul = _unloaded.try(:first, limit_or_opts)).try(:_id)] ||
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 the option { id_sort: :none }.
333
- # Be aware that #first/#last won't guarantee order in this case.
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
- # @option limit_or_opts [ :none ] :id_sort This option is deprecated.
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(limit_or_opts = nil)
334
+ def last(limit = nil)
343
335
  _added.values.try(:last) ||
344
336
  _loaded.try(:values).try(:last) ||
345
- _added[(ul = _unloaded.try(:last, limit_or_opts)).try(:_id)] ||
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.build(type || klass, attributes)
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
- execute_callback :before_remove, document
96
- _target.delete(document) do |doc|
97
- if doc
98
- unbind_one(doc)
99
- cascade!(doc) if !_assigning?
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
- if doc._matches?(selector)
473
- unbind_one(doc) and true
475
+ doc._matches?(selector).tap do |b|
476
+ unbind_one(doc) if b
474
477
  end
475
478
  end
476
479
  removed
@@ -53,7 +53,7 @@ module Mongoid
53
53
  end
54
54
 
55
55
  def execute_query(object, base)
56
- query_criteria(object, base).limit(1).first(id_sort: :none)
56
+ query_criteria(object, base).take
57
57
  end
58
58
 
59
59
  def with_polymorphic_criterion(criteria, base)
@@ -49,17 +49,14 @@ module Mongoid
49
49
  #
50
50
  # @return [ One ] The association.
51
51
  def substitute(replacement)
52
- # If the same object currently associated is being assigned,
53
- # rebind the association and save the target but do not destroy
54
- # the target.
55
-
56
- unbind_one
57
- if persistable?
58
- # TODO can this entire method be skipped if self == replacement?
59
- if _association.destructive? && self != replacement
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 changes.has_key?(association.foreign_key)
71
- old, new = changes[association.foreign_key]
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}_changed?" if (stores_foreign_key? && 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
- mod.name.split('::').each do |part|
408
- parent = parent.const_get(part)
409
- hier << parent
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
- if relations.has_key?(name)
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?(name)
57
+ if nested_attributes.has_key?(aliased)
51
58
  pending_nested[name] = value
52
59
  return true
53
60
  end