mongoid 7.5.4 → 8.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (298) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/README.md +3 -3
  4. data/Rakefile +0 -25
  5. data/lib/config/locales/en.yml +46 -30
  6. data/lib/mongoid/association/accessors.rb +32 -3
  7. data/lib/mongoid/association/bindable.rb +48 -0
  8. data/lib/mongoid/association/builders.rb +4 -2
  9. data/lib/mongoid/association/eager_loadable.rb +29 -7
  10. data/lib/mongoid/association/embedded/batchable.rb +28 -5
  11. data/lib/mongoid/association/embedded/embedded_in/binding.rb +24 -2
  12. data/lib/mongoid/association/embedded/embedded_in.rb +2 -1
  13. data/lib/mongoid/association/embedded/embeds_many/binding.rb +1 -0
  14. data/lib/mongoid/association/embedded/embeds_many/buildable.rb +1 -1
  15. data/lib/mongoid/association/embedded/embeds_many/proxy.rb +40 -18
  16. data/lib/mongoid/association/embedded/embeds_one/buildable.rb +18 -4
  17. data/lib/mongoid/association/embedded/embeds_one/proxy.rb +21 -2
  18. data/lib/mongoid/association/macros.rb +2 -1
  19. data/lib/mongoid/association/many.rb +5 -0
  20. data/lib/mongoid/association/nested/many.rb +2 -1
  21. data/lib/mongoid/association/proxy.rb +12 -0
  22. data/lib/mongoid/association/referenced/auto_save.rb +3 -2
  23. data/lib/mongoid/association/referenced/belongs_to/binding.rb +1 -0
  24. data/lib/mongoid/association/referenced/belongs_to/buildable.rb +1 -1
  25. data/lib/mongoid/association/referenced/belongs_to.rb +1 -1
  26. data/lib/mongoid/association/referenced/counter_cache.rb +8 -8
  27. data/lib/mongoid/association/referenced/has_and_belongs_to_many/proxy.rb +64 -11
  28. data/lib/mongoid/association/referenced/has_and_belongs_to_many.rb +4 -1
  29. data/lib/mongoid/association/referenced/has_many/enumerable.rb +10 -18
  30. data/lib/mongoid/association/referenced/has_many/proxy.rb +12 -9
  31. data/lib/mongoid/association/referenced/has_one/buildable.rb +1 -1
  32. data/lib/mongoid/association/referenced/has_one/proxy.rb +8 -11
  33. data/lib/mongoid/association/referenced/syncable.rb +2 -2
  34. data/lib/mongoid/association/relatable.rb +38 -4
  35. data/lib/mongoid/attributes/processing.rb +9 -2
  36. data/lib/mongoid/attributes.rb +30 -27
  37. data/lib/mongoid/cacheable.rb +2 -2
  38. data/lib/mongoid/changeable.rb +37 -2
  39. data/lib/mongoid/clients/options.rb +4 -0
  40. data/lib/mongoid/clients/sessions.rb +2 -14
  41. data/lib/mongoid/config.rb +15 -11
  42. data/lib/mongoid/contextual/aggregable/memory.rb +23 -15
  43. data/lib/mongoid/contextual/aggregable/mongo.rb +1 -1
  44. data/lib/mongoid/contextual/map_reduce.rb +2 -2
  45. data/lib/mongoid/contextual/memory.rb +55 -28
  46. data/lib/mongoid/contextual/mongo.rb +173 -262
  47. data/lib/mongoid/contextual/none.rb +33 -15
  48. data/lib/mongoid/copyable.rb +32 -8
  49. data/lib/mongoid/criteria/includable.rb +24 -20
  50. data/lib/mongoid/criteria/marshalable.rb +10 -2
  51. data/lib/mongoid/criteria/queryable/extensions/array.rb +2 -15
  52. data/lib/mongoid/criteria/queryable/extensions/big_decimal.rb +25 -4
  53. data/lib/mongoid/criteria/queryable/extensions/boolean.rb +1 -1
  54. data/lib/mongoid/criteria/queryable/extensions/date.rb +6 -1
  55. data/lib/mongoid/criteria/queryable/extensions/date_time.rb +6 -1
  56. data/lib/mongoid/criteria/queryable/extensions/hash.rb +0 -16
  57. data/lib/mongoid/criteria/queryable/extensions/numeric.rb +1 -1
  58. data/lib/mongoid/criteria/queryable/extensions/object.rb +2 -1
  59. data/lib/mongoid/criteria/queryable/extensions/range.rb +13 -5
  60. data/lib/mongoid/criteria/queryable/extensions/regexp.rb +1 -1
  61. data/lib/mongoid/criteria/queryable/extensions/symbol.rb +3 -1
  62. data/lib/mongoid/criteria/queryable/extensions/time.rb +6 -1
  63. data/lib/mongoid/criteria/queryable/extensions/time_with_zone.rb +6 -1
  64. data/lib/mongoid/criteria/queryable/optional.rb +3 -9
  65. data/lib/mongoid/criteria/queryable/options.rb +1 -1
  66. data/lib/mongoid/criteria/queryable/selectable.rb +2 -24
  67. data/lib/mongoid/criteria/queryable/selector.rb +90 -5
  68. data/lib/mongoid/criteria/queryable/smash.rb +39 -6
  69. data/lib/mongoid/criteria/queryable/storable.rb +1 -1
  70. data/lib/mongoid/criteria/queryable.rb +11 -6
  71. data/lib/mongoid/criteria.rb +1 -28
  72. data/lib/mongoid/deprecable.rb +36 -0
  73. data/lib/mongoid/deprecation.rb +25 -0
  74. data/lib/mongoid/document.rb +88 -33
  75. data/lib/mongoid/equality.rb +4 -4
  76. data/lib/mongoid/errors/document_not_found.rb +6 -2
  77. data/lib/mongoid/errors/invalid_dot_dollar_assignment.rb +23 -0
  78. data/lib/mongoid/errors/invalid_field.rb +5 -1
  79. data/lib/mongoid/errors/invalid_field_type.rb +26 -0
  80. data/lib/mongoid/errors/too_many_nested_attribute_records.rb +1 -1
  81. data/lib/mongoid/errors.rb +2 -2
  82. data/lib/mongoid/extensions/array.rb +8 -6
  83. data/lib/mongoid/extensions/big_decimal.rb +29 -10
  84. data/lib/mongoid/extensions/binary.rb +42 -0
  85. data/lib/mongoid/extensions/boolean.rb +8 -2
  86. data/lib/mongoid/extensions/date.rb +26 -20
  87. data/lib/mongoid/extensions/date_time.rb +1 -1
  88. data/lib/mongoid/extensions/float.rb +4 -5
  89. data/lib/mongoid/extensions/hash.rb +12 -5
  90. data/lib/mongoid/extensions/integer.rb +4 -5
  91. data/lib/mongoid/extensions/object.rb +2 -0
  92. data/lib/mongoid/extensions/range.rb +41 -10
  93. data/lib/mongoid/extensions/regexp.rb +11 -4
  94. data/lib/mongoid/extensions/set.rb +11 -4
  95. data/lib/mongoid/extensions/string.rb +2 -13
  96. data/lib/mongoid/extensions/symbol.rb +3 -14
  97. data/lib/mongoid/extensions/time.rb +27 -16
  98. data/lib/mongoid/extensions/time_with_zone.rb +1 -2
  99. data/lib/mongoid/extensions.rb +1 -0
  100. data/lib/mongoid/factory.rb +42 -7
  101. data/lib/mongoid/fields/foreign_key.rb +7 -0
  102. data/lib/mongoid/fields/validators/macro.rb +3 -9
  103. data/lib/mongoid/fields.rb +49 -7
  104. data/lib/mongoid/findable.rb +21 -16
  105. data/lib/mongoid/indexable/specification.rb +1 -1
  106. data/lib/mongoid/indexable/validators/options.rb +4 -1
  107. data/lib/mongoid/interceptable.rb +69 -9
  108. data/lib/mongoid/persistable/creatable.rb +14 -5
  109. data/lib/mongoid/persistable/updatable.rb +12 -5
  110. data/lib/mongoid/persistence_context.rb +8 -42
  111. data/lib/mongoid/query_cache.rb +6 -258
  112. data/lib/mongoid/railties/controller_runtime.rb +1 -1
  113. data/lib/mongoid/reloadable.rb +7 -3
  114. data/lib/mongoid/scopable.rb +9 -11
  115. data/lib/mongoid/selectable.rb +1 -2
  116. data/lib/mongoid/shardable.rb +11 -35
  117. data/lib/mongoid/stateful.rb +27 -1
  118. data/lib/mongoid/timestamps/created.rb +1 -1
  119. data/lib/mongoid/timestamps/updated.rb +1 -1
  120. data/lib/mongoid/touchable.rb +2 -3
  121. data/lib/mongoid/traversable.rb +1 -0
  122. data/lib/mongoid/validatable/uniqueness.rb +2 -1
  123. data/lib/mongoid/version.rb +1 -1
  124. data/lib/mongoid/warnings.rb +3 -4
  125. data/lib/mongoid.rb +1 -0
  126. data/spec/config/mongoid.yml +16 -0
  127. data/spec/integration/app_spec.rb +8 -12
  128. data/spec/integration/associations/belongs_to_spec.rb +18 -0
  129. data/spec/integration/associations/embedded_spec.rb +15 -0
  130. data/spec/integration/associations/embeds_many_spec.rb +15 -2
  131. data/spec/integration/associations/embeds_one_spec.rb +18 -0
  132. data/spec/integration/associations/foreign_key_spec.rb +9 -0
  133. data/spec/integration/associations/has_and_belongs_to_many_spec.rb +21 -0
  134. data/spec/integration/associations/has_one_spec.rb +97 -1
  135. data/spec/integration/associations/scope_option_spec.rb +1 -1
  136. data/spec/integration/callbacks_models.rb +95 -1
  137. data/spec/integration/callbacks_spec.rb +226 -4
  138. data/spec/integration/criteria/range_spec.rb +95 -1
  139. data/spec/integration/discriminator_key_spec.rb +115 -76
  140. data/spec/integration/dots_and_dollars_spec.rb +277 -0
  141. data/spec/integration/i18n_fallbacks_spec.rb +1 -15
  142. data/spec/integration/matcher_examples_spec.rb +20 -13
  143. data/spec/integration/matcher_operator_data/type_decimal.yml +3 -2
  144. data/spec/integration/matcher_operator_spec.rb +3 -5
  145. data/spec/integration/persistence/range_field_spec.rb +350 -0
  146. data/spec/mongoid/association/counter_cache_spec.rb +1 -1
  147. data/spec/mongoid/association/depending_spec.rb +9 -9
  148. data/spec/mongoid/association/eager_spec.rb +2 -1
  149. data/spec/mongoid/association/embedded/embedded_in/binding_spec.rb +2 -1
  150. data/spec/mongoid/association/embedded/embedded_in/buildable_spec.rb +54 -0
  151. data/spec/mongoid/association/embedded/embedded_in/proxy_spec.rb +69 -9
  152. data/spec/mongoid/association/embedded/embeds_many/buildable_spec.rb +112 -0
  153. data/spec/mongoid/association/embedded/embeds_many/proxy_spec.rb +198 -8
  154. data/spec/mongoid/association/embedded/embeds_many_models.rb +36 -0
  155. data/spec/mongoid/association/embedded/embeds_many_query_spec.rb +12 -0
  156. data/spec/mongoid/association/embedded/embeds_many_spec.rb +68 -0
  157. data/spec/mongoid/association/embedded/embeds_one/buildable_spec.rb +25 -0
  158. data/spec/mongoid/association/embedded/embeds_one_models.rb +19 -0
  159. data/spec/mongoid/association/embedded/embeds_one_spec.rb +28 -0
  160. data/spec/mongoid/association/referenced/belongs_to/binding_spec.rb +2 -1
  161. data/spec/mongoid/association/referenced/belongs_to/buildable_spec.rb +54 -0
  162. data/spec/mongoid/association/referenced/belongs_to/proxy_spec.rb +15 -0
  163. data/spec/mongoid/association/referenced/belongs_to_models.rb +11 -0
  164. data/spec/mongoid/association/referenced/belongs_to_spec.rb +2 -2
  165. data/spec/mongoid/association/referenced/has_and_belongs_to_many/proxy_spec.rb +38 -5
  166. data/spec/mongoid/association/referenced/has_and_belongs_to_many_models.rb +25 -0
  167. data/spec/mongoid/association/referenced/has_and_belongs_to_many_spec.rb +35 -2
  168. data/spec/mongoid/association/referenced/has_many/buildable_spec.rb +109 -0
  169. data/spec/mongoid/association/referenced/has_many/enumerable_spec.rb +2 -56
  170. data/spec/mongoid/association/referenced/has_many/proxy_spec.rb +62 -13
  171. data/spec/mongoid/association/referenced/has_many_models.rb +3 -1
  172. data/spec/mongoid/association/referenced/has_many_spec.rb +25 -0
  173. data/spec/mongoid/association/referenced/has_one/buildable_spec.rb +2 -2
  174. data/spec/mongoid/association/referenced/has_one/proxy_spec.rb +107 -1
  175. data/spec/mongoid/association/referenced/has_one_models.rb +16 -0
  176. data/spec/mongoid/association/syncable_spec.rb +14 -0
  177. data/spec/mongoid/atomic/paths_spec.rb +0 -14
  178. data/spec/mongoid/attributes/nested_spec.rb +80 -11
  179. data/spec/mongoid/attributes/nested_spec_models.rb +48 -0
  180. data/spec/mongoid/attributes/projector_spec.rb +1 -5
  181. data/spec/mongoid/attributes_spec.rb +480 -27
  182. data/spec/mongoid/cacheable_spec.rb +3 -3
  183. data/spec/mongoid/changeable_spec.rb +130 -13
  184. data/spec/mongoid/clients/factory_spec.rb +23 -30
  185. data/spec/mongoid/clients/sessions_spec.rb +0 -38
  186. data/spec/mongoid/clients_spec.rb +2 -2
  187. data/spec/mongoid/config_spec.rb +52 -14
  188. data/spec/mongoid/contextual/aggregable/memory_spec.rb +396 -158
  189. data/spec/mongoid/contextual/aggregable/memory_table.yml +88 -0
  190. data/spec/mongoid/contextual/aggregable/memory_table_spec.rb +62 -0
  191. data/spec/mongoid/contextual/map_reduce_spec.rb +2 -16
  192. data/spec/mongoid/contextual/memory_spec.rb +521 -14
  193. data/spec/mongoid/contextual/mongo_spec.rb +566 -416
  194. data/spec/mongoid/contextual/none_spec.rb +11 -19
  195. data/spec/mongoid/copyable_spec.rb +451 -1
  196. data/spec/mongoid/criteria/findable_spec.rb +86 -210
  197. data/spec/mongoid/criteria/includable_spec.rb +1492 -0
  198. data/spec/mongoid/criteria/includable_spec_models.rb +54 -0
  199. data/spec/mongoid/criteria/marshalable_spec.rb +18 -1
  200. data/spec/mongoid/criteria/queryable/extensions/array_spec.rb +7 -19
  201. data/spec/mongoid/criteria/queryable/extensions/big_decimal_spec.rb +134 -26
  202. data/spec/mongoid/criteria/queryable/extensions/date_spec.rb +11 -0
  203. data/spec/mongoid/criteria/queryable/extensions/date_time_spec.rb +11 -0
  204. data/spec/mongoid/criteria/queryable/extensions/hash_spec.rb +0 -15
  205. data/spec/mongoid/criteria/queryable/extensions/numeric_spec.rb +73 -7
  206. data/spec/mongoid/criteria/queryable/extensions/time_spec.rb +11 -0
  207. data/spec/mongoid/criteria/queryable/extensions/time_with_zone_spec.rb +11 -0
  208. data/spec/mongoid/criteria/queryable/optional_spec.rb +0 -484
  209. data/spec/mongoid/criteria/queryable/selectable_logical_spec.rb +50 -0
  210. data/spec/mongoid/criteria/queryable/selectable_spec.rb +77 -85
  211. data/spec/mongoid/criteria/queryable/selector_spec.rb +16 -77
  212. data/spec/mongoid/criteria/queryable/storable_spec.rb +0 -72
  213. data/spec/mongoid/criteria_spec.rb +469 -1201
  214. data/spec/mongoid/document_fields_spec.rb +173 -24
  215. data/spec/mongoid/document_spec.rb +32 -41
  216. data/spec/mongoid/equality_spec.rb +12 -12
  217. data/spec/mongoid/errors/document_not_found_spec.rb +29 -2
  218. data/spec/mongoid/errors/invalid_field_spec.rb +1 -1
  219. data/spec/mongoid/errors/invalid_field_type_spec.rb +55 -0
  220. data/spec/mongoid/errors/mongoid_error_spec.rb +3 -1
  221. data/spec/mongoid/errors/no_environment_spec.rb +3 -3
  222. data/spec/mongoid/errors/too_many_nested_attribute_records_spec.rb +1 -1
  223. data/spec/mongoid/extensions/array_spec.rb +16 -2
  224. data/spec/mongoid/extensions/big_decimal_spec.rb +697 -212
  225. data/spec/mongoid/extensions/binary_spec.rb +44 -9
  226. data/spec/mongoid/extensions/boolean_spec.rb +68 -82
  227. data/spec/mongoid/extensions/date_class_mongoize_spec.rb +7 -3
  228. data/spec/mongoid/extensions/date_spec.rb +71 -1
  229. data/spec/mongoid/extensions/date_time_spec.rb +15 -9
  230. data/spec/mongoid/extensions/float_spec.rb +48 -76
  231. data/spec/mongoid/extensions/hash_spec.rb +30 -0
  232. data/spec/mongoid/extensions/integer_spec.rb +45 -66
  233. data/spec/mongoid/extensions/range_spec.rb +255 -54
  234. data/spec/mongoid/extensions/regexp_spec.rb +58 -33
  235. data/spec/mongoid/extensions/set_spec.rb +106 -0
  236. data/spec/mongoid/extensions/string_spec.rb +53 -25
  237. data/spec/mongoid/extensions/symbol_spec.rb +18 -25
  238. data/spec/mongoid/extensions/time_spec.rb +634 -66
  239. data/spec/mongoid/extensions/time_with_zone_spec.rb +17 -31
  240. data/spec/mongoid/factory_spec.rb +61 -1
  241. data/spec/mongoid/fields_spec.rb +321 -50
  242. data/spec/mongoid/findable_spec.rb +64 -29
  243. data/spec/mongoid/indexable/specification_spec.rb +2 -2
  244. data/spec/mongoid/indexable_spec.rb +16 -19
  245. data/spec/mongoid/interceptable_spec.rb +584 -5
  246. data/spec/mongoid/interceptable_spec_models.rb +235 -4
  247. data/spec/mongoid/matcher/extract_attribute_spec.rb +1 -5
  248. data/spec/mongoid/mongoizable_spec.rb +285 -0
  249. data/spec/mongoid/persistable/creatable_spec.rb +2 -2
  250. data/spec/mongoid/persistable/deletable_spec.rb +2 -2
  251. data/spec/mongoid/persistable/destroyable_spec.rb +2 -2
  252. data/spec/mongoid/persistable/upsertable_spec.rb +14 -0
  253. data/spec/mongoid/persistence_context_spec.rb +24 -0
  254. data/spec/mongoid/query_cache_middleware_spec.rb +0 -18
  255. data/spec/mongoid/query_cache_spec.rb +0 -154
  256. data/spec/mongoid/reloadable_spec.rb +35 -2
  257. data/spec/mongoid/scopable_spec.rb +36 -34
  258. data/spec/mongoid/shardable_models.rb +0 -14
  259. data/spec/mongoid/shardable_spec.rb +61 -153
  260. data/spec/mongoid/stateful_spec.rb +28 -0
  261. data/spec/mongoid/timestamps_spec.rb +390 -0
  262. data/spec/mongoid/timestamps_spec_models.rb +67 -0
  263. data/spec/mongoid/touchable_spec.rb +116 -0
  264. data/spec/mongoid/touchable_spec_models.rb +12 -8
  265. data/spec/mongoid/traversable_spec.rb +4 -11
  266. data/spec/mongoid/validatable/presence_spec.rb +1 -1
  267. data/spec/mongoid/validatable/uniqueness_spec.rb +60 -31
  268. data/spec/mongoid/warnings_spec.rb +35 -0
  269. data/spec/mongoid_spec.rb +1 -7
  270. data/spec/rails/controller_extension/controller_runtime_spec.rb +2 -2
  271. data/spec/rails/mongoid_spec.rb +4 -16
  272. data/spec/shared/lib/mrss/event_subscriber.rb +5 -15
  273. data/spec/shared/lib/mrss/lite_constraints.rb +0 -8
  274. data/spec/shared/shlib/server.sh +5 -5
  275. data/spec/support/constraints.rb +24 -0
  276. data/spec/support/macros.rb +30 -0
  277. data/spec/support/models/augmentation.rb +12 -0
  278. data/spec/support/models/band.rb +3 -0
  279. data/spec/support/models/catalog.rb +24 -0
  280. data/spec/support/models/circus.rb +3 -0
  281. data/spec/support/models/fanatic.rb +8 -0
  282. data/spec/support/models/implant.rb +9 -0
  283. data/spec/support/models/label.rb +2 -0
  284. data/spec/support/models/passport.rb +9 -0
  285. data/spec/support/models/person.rb +1 -0
  286. data/spec/support/models/player.rb +2 -0
  287. data/spec/support/models/powerup.rb +12 -0
  288. data/spec/support/models/registry.rb +1 -0
  289. data/spec/support/models/school.rb +14 -0
  290. data/spec/support/models/shield.rb +18 -0
  291. data/spec/support/models/student.rb +14 -0
  292. data/spec/support/models/weapon.rb +12 -0
  293. data.tar.gz.sig +0 -0
  294. metadata +689 -657
  295. metadata.gz.sig +0 -0
  296. data/lib/mongoid/errors/eager_load.rb +0 -23
  297. data/lib/mongoid/errors/invalid_value.rb +0 -17
  298. data/spec/mongoid/errors/eager_load_spec.rb +0 -31
@@ -67,10 +67,11 @@ module Mongoid
67
67
  #
68
68
  # @return [ Document ] The new document.
69
69
  def build(attributes = {}, type = nil)
70
- doc = Factory.build(type || _association.klass, attributes)
70
+ doc = Factory.execute_build(type || _association.klass, attributes, execute_callbacks: false)
71
71
  append(doc)
72
72
  doc.apply_post_processed_defaults
73
73
  yield(doc) if block_given?
74
+ doc.run_pending_callbacks
74
75
  doc.run_callbacks(:build) { doc }
75
76
  _base._reset_memoized_descendants!
76
77
  doc
@@ -94,6 +95,7 @@ module Mongoid
94
95
  # @return [ self ] The empty association.
95
96
  def clear
96
97
  batch_clear(_target.dup)
98
+ update_attributes_hash
97
99
  self
98
100
  end
99
101
 
@@ -135,20 +137,21 @@ module Mongoid
135
137
  #
136
138
  # @return [ Document, nil ] The deleted document or nil if nothing deleted.
137
139
  def delete(document)
138
- execute_callback :before_remove, document
139
- doc = _target.delete_one(document)
140
- if doc && !_binding?
141
- _unscoped.delete_one(doc)
142
- if _assigning?
143
- _base.add_atomic_pull(doc)
144
- else
145
- doc.delete(suppress: true)
146
- unbind_one(doc)
140
+ execute_callbacks_around(:remove, document) do
141
+ doc = _target.delete_one(document)
142
+ if doc && !_binding?
143
+ _unscoped.delete_one(doc)
144
+ if _assigning?
145
+ _base.add_atomic_pull(doc)
146
+ else
147
+ doc.delete(suppress: true)
148
+ unbind_one(doc)
149
+ end
150
+ update_attributes_hash
147
151
  end
152
+ reindex
153
+ doc
148
154
  end
149
- reindex
150
- execute_callback :after_remove, document
151
- doc
152
155
  end
153
156
 
154
157
  # Delete all the documents in the association without running callbacks.
@@ -209,7 +212,7 @@ module Mongoid
209
212
  #
210
213
  # @return [ true, false ] True is persisted documents exist, false if not.
211
214
  def exists?
212
- count > 0
215
+ _target.any? { |doc| doc.persisted? }
213
216
  end
214
217
 
215
218
  # Finds a document in this association through several different
@@ -256,6 +259,7 @@ module Mongoid
256
259
  integrate(doc)
257
260
  doc._index = index
258
261
  end
262
+ update_attributes_hash
259
263
  @_unscoped = _target.dup
260
264
  @_target = scope(_target)
261
265
  end
@@ -291,6 +295,8 @@ module Mongoid
291
295
  end
292
296
  else
293
297
  delete(_target[-1])
298
+ end.tap do
299
+ update_attributes_hash
294
300
  end
295
301
  end
296
302
 
@@ -314,6 +320,8 @@ module Mongoid
314
320
  end
315
321
  else
316
322
  delete(_target[0])
323
+ end.tap do
324
+ update_attributes_hash
317
325
  end
318
326
  end
319
327
 
@@ -328,6 +336,7 @@ module Mongoid
328
336
  # @return [ Many ] The proxied association.
329
337
  def substitute(docs)
330
338
  batch_replace(docs)
339
+ update_attributes_hash
331
340
  self
332
341
  end
333
342
 
@@ -365,6 +374,7 @@ module Mongoid
365
374
  end
366
375
  _unscoped.push(document)
367
376
  integrate(document)
377
+ update_attributes_hash
368
378
  document._index = _unscoped.size - 1
369
379
  execute_callback :after_add, document
370
380
  end
@@ -398,6 +408,7 @@ module Mongoid
398
408
  def delete_one(document)
399
409
  _target.delete_one(document)
400
410
  _unscoped.delete_one(document)
411
+ update_attributes_hash
401
412
  reindex
402
413
  end
403
414
 
@@ -486,6 +497,7 @@ module Mongoid
486
497
  criteria = where(conditions || {})
487
498
  removed = criteria.size
488
499
  batch_remove(criteria, method)
500
+ update_attributes_hash
489
501
  removed
490
502
  end
491
503
 
@@ -511,12 +523,22 @@ module Mongoid
511
523
  @_unscoped = docs
512
524
  end
513
525
 
526
+ # Returns a list of attributes hashes for each document.
527
+ #
528
+ # @return [ Array<Hash> ] The list of attributes hashes
514
529
  def as_attributes
515
- attributes = []
516
- _unscoped.each do |doc|
517
- attributes.push(doc.as_document)
530
+ _unscoped.map { |doc| doc.send(:as_attributes) }
531
+ end
532
+
533
+ # Update the _base's attributes hash with the _target's attributes
534
+ #
535
+ # @api private
536
+ def update_attributes_hash
537
+ if !_target.empty?
538
+ _base.attributes.merge!(_association.store_as => _target.map(&:attributes))
539
+ else
540
+ _base.attributes.delete(_association.store_as)
518
541
  end
519
- attributes
520
542
  end
521
543
 
522
544
  class << self
@@ -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