mongoid 6.4.2 → 7.0.3

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 (318) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +1 -1
  3. data/lib/config/locales/en.yml +17 -0
  4. data/lib/mongoid/association/accessors.rb +339 -0
  5. data/lib/mongoid/{relations/binding.rb → association/bindable.rb} +32 -52
  6. data/lib/mongoid/association/builders.rb +92 -0
  7. data/lib/mongoid/{relations/constraint.rb → association/constrainable.rb} +11 -22
  8. data/lib/mongoid/association/depending.rb +137 -0
  9. data/lib/mongoid/{relations/eager.rb → association/eager_loadable.rb} +11 -11
  10. data/lib/mongoid/{relations → association}/embedded/batchable.rb +19 -19
  11. data/lib/mongoid/association/embedded/cyclic.rb +109 -0
  12. data/lib/mongoid/association/embedded/embedded_in/binding.rb +56 -0
  13. data/lib/mongoid/{relations/builders/embedded/in.rb → association/embedded/embedded_in/buildable.rb} +12 -6
  14. data/lib/mongoid/association/embedded/embedded_in/proxy.rb +121 -0
  15. data/lib/mongoid/association/embedded/embedded_in.rb +154 -0
  16. data/lib/mongoid/{relations/bindings/embedded/many.rb → association/embedded/embeds_many/binding.rb} +11 -9
  17. data/lib/mongoid/{relations/builders/embedded/many.rb → association/embedded/embeds_many/buildable.rb} +13 -7
  18. data/lib/mongoid/association/embedded/embeds_many/proxy.rb +552 -0
  19. data/lib/mongoid/association/embedded/embeds_many.rb +210 -0
  20. data/lib/mongoid/{relations/bindings/embedded/one.rb → association/embedded/embeds_one/binding.rb} +12 -10
  21. data/lib/mongoid/{relations/builders/embedded/one.rb → association/embedded/embeds_one/buildable.rb} +13 -7
  22. data/lib/mongoid/association/embedded/embeds_one/proxy.rb +130 -0
  23. data/lib/mongoid/association/embedded/embeds_one.rb +173 -0
  24. data/lib/mongoid/association/embedded.rb +4 -0
  25. data/lib/mongoid/association/macros.rb +204 -0
  26. data/lib/mongoid/{relations → association}/many.rb +19 -49
  27. data/lib/mongoid/{relations → association}/marshalable.rb +6 -4
  28. data/lib/mongoid/association/nested/many.rb +200 -0
  29. data/lib/mongoid/association/nested/nested_buildable.rb +72 -0
  30. data/lib/mongoid/association/nested/one.rb +127 -0
  31. data/lib/mongoid/association/nested.rb +15 -0
  32. data/lib/mongoid/{relations → association}/one.rb +6 -6
  33. data/lib/mongoid/association/options.rb +152 -0
  34. data/lib/mongoid/{relations → association}/proxy.rb +31 -58
  35. data/lib/mongoid/association/referenced/auto_save.rb +79 -0
  36. data/lib/mongoid/association/referenced/belongs_to/binding.rb +87 -0
  37. data/lib/mongoid/association/referenced/belongs_to/buildable.rb +46 -0
  38. data/lib/mongoid/association/referenced/belongs_to/eager.rb +36 -0
  39. data/lib/mongoid/association/referenced/belongs_to/proxy.rb +136 -0
  40. data/lib/mongoid/association/referenced/belongs_to.rb +248 -0
  41. data/lib/mongoid/association/referenced/counter_cache.rb +163 -0
  42. data/lib/mongoid/association/referenced/eager.rb +162 -0
  43. data/lib/mongoid/association/referenced/has_and_belongs_to_many/binding.rb +71 -0
  44. data/lib/mongoid/association/referenced/has_and_belongs_to_many/buildable.rb +40 -0
  45. data/lib/mongoid/association/referenced/has_and_belongs_to_many/eager.rb +52 -0
  46. data/lib/mongoid/association/referenced/has_and_belongs_to_many/proxy.rb +312 -0
  47. data/lib/mongoid/association/referenced/has_and_belongs_to_many.rb +290 -0
  48. data/lib/mongoid/{relations/bindings/referenced/many.rb → association/referenced/has_many/binding.rb} +6 -5
  49. data/lib/mongoid/association/referenced/has_many/buildable.rb +38 -0
  50. data/lib/mongoid/association/referenced/has_many/eager.rb +43 -0
  51. data/lib/mongoid/association/referenced/has_many/enumerable.rb +510 -0
  52. data/lib/mongoid/association/referenced/has_many/proxy.rb +578 -0
  53. data/lib/mongoid/association/referenced/has_many.rb +275 -0
  54. data/lib/mongoid/{relations/bindings/referenced/one.rb → association/referenced/has_one/binding.rb} +11 -8
  55. data/lib/mongoid/association/referenced/has_one/buildable.rb +60 -0
  56. data/lib/mongoid/association/referenced/has_one/eager.rb +35 -0
  57. data/lib/mongoid/{relations/builders/nested_attributes/one.rb → association/referenced/has_one/nested_builder.rb} +9 -9
  58. data/lib/mongoid/association/referenced/has_one/proxy.rb +113 -0
  59. data/lib/mongoid/association/referenced/has_one.rb +204 -0
  60. data/lib/mongoid/association/referenced/syncable.rb +170 -0
  61. data/lib/mongoid/association/referenced.rb +7 -0
  62. data/lib/mongoid/{relations → association}/reflections.rb +21 -17
  63. data/lib/mongoid/association/relatable.rb +509 -0
  64. data/lib/mongoid/{relations.rb → association.rb} +57 -56
  65. data/lib/mongoid/atomic/paths/embedded/many.rb +1 -1
  66. data/lib/mongoid/atomic/paths/embedded/one.rb +1 -1
  67. data/lib/mongoid/atomic.rb +4 -4
  68. data/lib/mongoid/attributes/dynamic.rb +2 -2
  69. data/lib/mongoid/attributes/nested.rb +22 -11
  70. data/lib/mongoid/attributes/processing.rb +2 -2
  71. data/lib/mongoid/attributes/readonly.rb +2 -4
  72. data/lib/mongoid/attributes.rb +22 -13
  73. data/lib/mongoid/changeable.rb +1 -1
  74. data/lib/mongoid/clients/options.rb +7 -5
  75. data/lib/mongoid/composable.rb +4 -4
  76. data/lib/mongoid/config.rb +1 -0
  77. data/lib/mongoid/contextual/atomic.rb +1 -1
  78. data/lib/mongoid/contextual/geo_near.rb +1 -1
  79. data/lib/mongoid/contextual/memory.rb +21 -3
  80. data/lib/mongoid/contextual/mongo.rb +10 -8
  81. data/lib/mongoid/copyable.rb +7 -6
  82. data/lib/mongoid/criteria/includable.rb +14 -14
  83. data/lib/mongoid/criteria/modifiable.rb +8 -14
  84. data/lib/mongoid/criteria/options.rb +2 -2
  85. data/lib/mongoid/criteria/queryable/extensions/string.rb +1 -1
  86. data/lib/mongoid/criteria/queryable/pipeline.rb +10 -5
  87. data/lib/mongoid/criteria/queryable/selectable.rb +37 -7
  88. data/lib/mongoid/criteria.rb +2 -2
  89. data/lib/mongoid/document.rb +15 -6
  90. data/lib/mongoid/errors/invalid_dependent_strategy.rb +32 -0
  91. data/lib/mongoid/errors/invalid_relation_option.rb +29 -0
  92. data/lib/mongoid/errors/unknown_model.rb +25 -0
  93. data/lib/mongoid/errors.rb +3 -0
  94. data/lib/mongoid/extensions/array.rb +5 -5
  95. data/lib/mongoid/extensions/big_decimal.rb +1 -1
  96. data/lib/mongoid/extensions/hash.rb +5 -2
  97. data/lib/mongoid/extensions/object.rb +4 -4
  98. data/lib/mongoid/extensions/range.rb +1 -0
  99. data/lib/mongoid/extensions/regexp.rb +1 -0
  100. data/lib/mongoid/extensions.rb +0 -4
  101. data/lib/mongoid/factory.rb +13 -3
  102. data/lib/mongoid/fields/foreign_key.rb +5 -5
  103. data/lib/mongoid/fields/standard.rb +2 -14
  104. data/lib/mongoid/fields/validators/macro.rb +1 -1
  105. data/lib/mongoid/fields.rb +3 -3
  106. data/lib/mongoid/indexable.rb +4 -1
  107. data/lib/mongoid/interceptable.rb +5 -5
  108. data/lib/mongoid/matchable/and.rb +1 -1
  109. data/lib/mongoid/matchable/elem_match.rb +9 -3
  110. data/lib/mongoid/matchable/eq.rb +22 -0
  111. data/lib/mongoid/matchable/ne.rb +1 -1
  112. data/lib/mongoid/matchable.rb +3 -1
  113. data/lib/mongoid/persistable/deletable.rb +7 -6
  114. data/lib/mongoid/persistable/incrementable.rb +1 -1
  115. data/lib/mongoid/persistable/logical.rb +1 -1
  116. data/lib/mongoid/persistable/settable.rb +57 -12
  117. data/lib/mongoid/persistable.rb +4 -5
  118. data/lib/mongoid/persistence_context.rb +20 -5
  119. data/lib/mongoid/query_cache.rb +8 -4
  120. data/lib/mongoid/railtie.rb +17 -0
  121. data/lib/mongoid/railties/controller_runtime.rb +86 -0
  122. data/lib/mongoid/scopable.rb +3 -3
  123. data/lib/mongoid/serializable.rb +4 -4
  124. data/lib/mongoid/shardable.rb +1 -1
  125. data/lib/mongoid/threaded.rb +36 -0
  126. data/lib/mongoid/touchable.rb +102 -0
  127. data/lib/mongoid/traversable.rb +3 -3
  128. data/lib/mongoid/validatable/presence.rb +2 -2
  129. data/lib/mongoid/validatable/uniqueness.rb +4 -4
  130. data/lib/mongoid/validatable.rb +8 -8
  131. data/lib/mongoid/version.rb +1 -1
  132. data/lib/rails/generators/mongoid/config/templates/mongoid.yml +3 -0
  133. data/spec/app/models/animal.rb +2 -1
  134. data/spec/app/models/bomb.rb +1 -1
  135. data/spec/app/models/message.rb +1 -1
  136. data/spec/app/models/minim.rb +7 -0
  137. data/spec/app/models/person.rb +5 -2
  138. data/spec/app/models/shipment_address.rb +1 -0
  139. data/spec/app/models/store_as_dup_test3.rb +7 -0
  140. data/spec/app/models/store_as_dup_test4.rb +7 -0
  141. data/spec/app/models/updatable.rb +7 -0
  142. data/spec/app/models/vertex.rb +6 -0
  143. data/spec/app/models/wiki_page.rb +1 -1
  144. data/spec/config/mongoid.yml +13 -3
  145. data/spec/integration/associations/belongs_to_spec.rb +13 -0
  146. data/spec/lite_spec_helper.rb +56 -0
  147. data/spec/mongoid/{relations → association}/accessors_spec.rb +40 -1
  148. data/spec/mongoid/{relations → association}/auto_save_spec.rb +60 -12
  149. data/spec/mongoid/{relations → association}/builders_spec.rb +1 -1
  150. data/spec/mongoid/association/constrainable_spec.rb +115 -0
  151. data/spec/mongoid/{relations → association}/counter_cache_spec.rb +1 -1
  152. data/spec/mongoid/association/depending_spec.rb +866 -0
  153. data/spec/mongoid/{relations → association}/eager_spec.rb +12 -12
  154. data/spec/mongoid/{relations → association/embedded}/cyclic_spec.rb +1 -1
  155. data/spec/mongoid/{relations/bindings/embedded/in_spec.rb → association/embedded/embedded_in/binding_spec.rb} +13 -13
  156. data/spec/mongoid/{relations/builders/embedded/in_spec.rb → association/embedded/embedded_in/buildable_spec.rb} +9 -9
  157. data/spec/mongoid/{relations/embedded/in_spec.rb → association/embedded/embedded_in/proxy_spec.rb} +5 -77
  158. data/spec/mongoid/association/embedded/embedded_in_spec.rb +843 -0
  159. data/spec/mongoid/{relations/bindings/embedded/many_spec.rb → association/embedded/embeds_many/binding_spec.rb} +3 -3
  160. data/spec/mongoid/{relations/builders/embedded/many_spec.rb → association/embedded/embeds_many/buildable_spec.rb} +17 -45
  161. data/spec/mongoid/{relations/embedded/many_spec.rb → association/embedded/embeds_many/proxy_spec.rb} +124 -182
  162. data/spec/mongoid/association/embedded/embeds_many_spec.rb +852 -0
  163. data/spec/mongoid/{relations/bindings/embedded/one_spec.rb → association/embedded/embeds_one/binding_spec.rb} +4 -4
  164. data/spec/mongoid/{relations/builders/embedded/one_spec.rb → association/embedded/embeds_one/buildable_spec.rb} +14 -34
  165. data/spec/mongoid/{relations/embedded/one_spec.rb → association/embedded/embeds_one/proxy_spec.rb} +39 -84
  166. data/spec/mongoid/association/embedded/embeds_one_spec.rb +908 -0
  167. data/spec/mongoid/{relations → association}/macros_spec.rb +148 -73
  168. data/spec/mongoid/{relations/builders/nested_attributes → association/nested}/many_spec.rb +16 -19
  169. data/spec/mongoid/{relations/builders/nested_attributes → association/nested}/one_spec.rb +17 -20
  170. data/spec/mongoid/association/options_spec.rb +1321 -0
  171. data/spec/mongoid/{relations → association}/polymorphic_spec.rb +66 -34
  172. data/spec/mongoid/{relations/bindings/referenced/in_spec.rb → association/referenced/belongs_to/binding_spec.rb} +7 -7
  173. data/spec/mongoid/{relations/builders/referenced/in_spec.rb → association/referenced/belongs_to/buildable_spec.rb} +46 -79
  174. data/spec/mongoid/{relations/eager/belongs_to_spec.rb → association/referenced/belongs_to/eager_spec.rb} +33 -14
  175. data/spec/mongoid/{relations/referenced/in_spec.rb → association/referenced/belongs_to/proxy_spec.rb} +57 -91
  176. data/spec/mongoid/association/referenced/belongs_to_spec.rb +2006 -0
  177. data/spec/mongoid/{relations/bindings/referenced/many_to_many_spec.rb → association/referenced/has_and_belongs_to_many/binding_spec.rb} +5 -5
  178. data/spec/mongoid/association/referenced/has_and_belongs_to_many/buildable_spec.rb +121 -0
  179. data/spec/mongoid/{relations/eager/has_and_belongs_to_many_spec.rb → association/referenced/has_and_belongs_to_many/eager_spec.rb} +26 -7
  180. data/spec/mongoid/association/referenced/has_and_belongs_to_many/proxy_persistence_spec.rb +56 -0
  181. data/spec/mongoid/{relations/referenced/many_to_many_spec.rb → association/referenced/has_and_belongs_to_many/proxy_spec.rb} +107 -98
  182. data/spec/mongoid/association/referenced/has_and_belongs_to_many_models.rb +26 -0
  183. data/spec/mongoid/association/referenced/has_and_belongs_to_many_spec.rb +1027 -0
  184. data/spec/mongoid/{relations/bindings/referenced/many_spec.rb → association/referenced/has_many/binding_spec.rb} +5 -5
  185. data/spec/mongoid/association/referenced/has_many/buildable_spec.rb +119 -0
  186. data/spec/mongoid/{relations/eager/has_many_spec.rb → association/referenced/has_many/eager_spec.rb} +26 -11
  187. data/spec/mongoid/{relations/targets → association/referenced/has_many}/enumerable_spec.rb +122 -1
  188. data/spec/mongoid/association/referenced/has_many/proxy_query_spec.rb +23 -0
  189. data/spec/mongoid/{relations/referenced/many_spec.rb → association/referenced/has_many/proxy_spec.rb} +28 -93
  190. data/spec/mongoid/association/referenced/has_many_models.rb +37 -0
  191. data/spec/mongoid/association/referenced/has_many_spec.rb +1225 -0
  192. data/spec/mongoid/{relations/bindings/referenced/one_spec.rb → association/referenced/has_one/binding_spec.rb} +4 -4
  193. data/spec/mongoid/association/referenced/has_one/buildable_spec.rb +113 -0
  194. data/spec/mongoid/{relations/eager/has_one_spec.rb → association/referenced/has_one/eager_spec.rb} +10 -10
  195. data/spec/mongoid/{relations/referenced/one_spec.rb → association/referenced/has_one/proxy_spec.rb} +9 -109
  196. data/spec/mongoid/association/referenced/has_one_models.rb +48 -0
  197. data/spec/mongoid/association/referenced/has_one_spec.rb +1350 -0
  198. data/spec/mongoid/{relations → association}/reflections_spec.rb +1 -12
  199. data/spec/mongoid/{relations/synchronization_spec.rb → association/syncable_spec.rb} +4 -2
  200. data/spec/mongoid/{relations_spec.rb → association_spec.rb} +1 -1
  201. data/spec/mongoid/atomic/modifiers_spec.rb +2 -2
  202. data/spec/mongoid/atomic_spec.rb +4 -4
  203. data/spec/mongoid/attributes/nested_spec.rb +29 -11
  204. data/spec/mongoid/attributes/readonly_spec.rb +80 -125
  205. data/spec/mongoid/attributes_spec.rb +38 -2
  206. data/spec/mongoid/clients/factory_spec.rb +24 -18
  207. data/spec/mongoid/clients/options_spec.rb +58 -44
  208. data/spec/mongoid/clients/sessions_spec.rb +1 -1
  209. data/spec/mongoid/clients/transactions_spec.rb +369 -0
  210. data/spec/mongoid/clients_spec.rb +68 -8
  211. data/spec/mongoid/config_spec.rb +27 -1
  212. data/spec/mongoid/contextual/memory_spec.rb +19 -0
  213. data/spec/mongoid/contextual/mongo_spec.rb +33 -5
  214. data/spec/mongoid/copyable_spec.rb +90 -6
  215. data/spec/mongoid/copyable_spec_models.rb +17 -0
  216. data/spec/mongoid/criteria/modifiable_spec.rb +183 -60
  217. data/spec/mongoid/criteria/queryable/extensions/big_decimal_spec.rb +3 -3
  218. data/spec/mongoid/criteria/queryable/extensions/string_spec.rb +43 -0
  219. data/spec/mongoid/criteria/queryable/pipeline_spec.rb +12 -0
  220. data/spec/mongoid/criteria/queryable/selectable_spec.rb +42 -3
  221. data/spec/mongoid/criteria/queryable/selector_spec.rb +2 -2
  222. data/spec/mongoid/criteria/scopable_spec.rb +81 -0
  223. data/spec/mongoid/criteria_spec.rb +33 -18
  224. data/spec/mongoid/document_spec.rb +83 -4
  225. data/spec/mongoid/extensions/array_spec.rb +11 -15
  226. data/spec/mongoid/extensions/big_decimal_spec.rb +9 -9
  227. data/spec/mongoid/extensions/hash_spec.rb +18 -1
  228. data/spec/mongoid/extensions/object_spec.rb +7 -11
  229. data/spec/mongoid/extensions/range_spec.rb +7 -0
  230. data/spec/mongoid/extensions/regexp_spec.rb +23 -0
  231. data/spec/mongoid/factory_spec.rb +19 -1
  232. data/spec/mongoid/fields/foreign_key_spec.rb +24 -32
  233. data/spec/mongoid/fields_spec.rb +2 -2
  234. data/spec/mongoid/findable_spec.rb +1 -1
  235. data/spec/mongoid/indexable_spec.rb +18 -8
  236. data/spec/mongoid/interceptable_spec.rb +22 -1
  237. data/spec/mongoid/matchable/elem_match_spec.rb +20 -0
  238. data/spec/mongoid/matchable/eq_spec.rb +48 -0
  239. data/spec/mongoid/persistable/incrementable_spec.rb +6 -6
  240. data/spec/mongoid/persistable/savable_spec.rb +2 -2
  241. data/spec/mongoid/persistable/settable_spec.rb +95 -10
  242. data/spec/mongoid/persistable_spec.rb +21 -6
  243. data/spec/mongoid/persistence_context_spec.rb +1 -1
  244. data/spec/mongoid/query_cache_spec.rb +61 -22
  245. data/spec/mongoid/relations/proxy_spec.rb +124 -124
  246. data/spec/mongoid/scopable_spec.rb +13 -0
  247. data/spec/mongoid/shardable_spec.rb +32 -12
  248. data/spec/mongoid/threaded_spec.rb +68 -0
  249. data/spec/mongoid/{relations/touchable_spec.rb → touchable_spec.rb} +40 -1
  250. data/spec/mongoid/validatable/associated_spec.rb +1 -1
  251. data/spec/mongoid/validatable/presence_spec.rb +7 -6
  252. data/spec/mongoid/validatable_spec.rb +1 -1
  253. data/spec/rails/controller_extension/controller_runtime_spec.rb +110 -0
  254. data/spec/spec_helper.rb +51 -25
  255. data/spec/support/constraints.rb +101 -0
  256. data/spec/support/macros.rb +20 -0
  257. data/spec/support/spec_config.rb +39 -0
  258. metadata +602 -582
  259. checksums.yaml.gz.sig +0 -2
  260. data/lib/mongoid/relations/accessors.rb +0 -267
  261. data/lib/mongoid/relations/auto_save.rb +0 -94
  262. data/lib/mongoid/relations/bindings/embedded/in.rb +0 -59
  263. data/lib/mongoid/relations/bindings/referenced/in.rb +0 -65
  264. data/lib/mongoid/relations/bindings/referenced/many_to_many.rb +0 -70
  265. data/lib/mongoid/relations/bindings.rb +0 -9
  266. data/lib/mongoid/relations/builder.rb +0 -57
  267. data/lib/mongoid/relations/builders/nested_attributes/many.rb +0 -199
  268. data/lib/mongoid/relations/builders/referenced/in.rb +0 -26
  269. data/lib/mongoid/relations/builders/referenced/many.rb +0 -26
  270. data/lib/mongoid/relations/builders/referenced/many_to_many.rb +0 -39
  271. data/lib/mongoid/relations/builders/referenced/one.rb +0 -26
  272. data/lib/mongoid/relations/builders.rb +0 -106
  273. data/lib/mongoid/relations/cascading/delete.rb +0 -44
  274. data/lib/mongoid/relations/cascading/destroy.rb +0 -43
  275. data/lib/mongoid/relations/cascading/nullify.rb +0 -35
  276. data/lib/mongoid/relations/cascading/restrict.rb +0 -39
  277. data/lib/mongoid/relations/cascading.rb +0 -56
  278. data/lib/mongoid/relations/conversions.rb +0 -34
  279. data/lib/mongoid/relations/counter_cache.rb +0 -160
  280. data/lib/mongoid/relations/cyclic.rb +0 -107
  281. data/lib/mongoid/relations/eager/base.rb +0 -153
  282. data/lib/mongoid/relations/eager/belongs_to.rb +0 -31
  283. data/lib/mongoid/relations/eager/has_and_belongs_to_many.rb +0 -47
  284. data/lib/mongoid/relations/eager/has_many.rb +0 -38
  285. data/lib/mongoid/relations/eager/has_one.rb +0 -30
  286. data/lib/mongoid/relations/embedded/in.rb +0 -241
  287. data/lib/mongoid/relations/embedded/many.rb +0 -683
  288. data/lib/mongoid/relations/embedded/one.rb +0 -235
  289. data/lib/mongoid/relations/macros.rb +0 -367
  290. data/lib/mongoid/relations/metadata.rb +0 -1179
  291. data/lib/mongoid/relations/nested_builder.rb +0 -74
  292. data/lib/mongoid/relations/options.rb +0 -49
  293. data/lib/mongoid/relations/polymorphic.rb +0 -39
  294. data/lib/mongoid/relations/referenced/in.rb +0 -304
  295. data/lib/mongoid/relations/referenced/many.rb +0 -812
  296. data/lib/mongoid/relations/referenced/many_to_many.rb +0 -479
  297. data/lib/mongoid/relations/referenced/one.rb +0 -290
  298. data/lib/mongoid/relations/synchronization.rb +0 -169
  299. data/lib/mongoid/relations/targets/enumerable.rb +0 -493
  300. data/lib/mongoid/relations/targets.rb +0 -2
  301. data/lib/mongoid/relations/touchable.rb +0 -97
  302. data/spec/mongoid/fields/internal/foreign_keys/array_spec.rb +0 -184
  303. data/spec/mongoid/fields/internal/foreign_keys/object_spec.rb +0 -201
  304. data/spec/mongoid/relations/builders/referenced/many_spec.rb +0 -137
  305. data/spec/mongoid/relations/builders/referenced/many_to_many_spec.rb +0 -178
  306. data/spec/mongoid/relations/builders/referenced/one_spec.rb +0 -111
  307. data/spec/mongoid/relations/cascading/delete_spec.rb +0 -101
  308. data/spec/mongoid/relations/cascading/destroy_spec.rb +0 -47
  309. data/spec/mongoid/relations/cascading/nullify_spec.rb +0 -32
  310. data/spec/mongoid/relations/cascading/restrict_spec.rb +0 -68
  311. data/spec/mongoid/relations/cascading_spec.rb +0 -355
  312. data/spec/mongoid/relations/constraint_spec.rb +0 -75
  313. data/spec/mongoid/relations/conversions_spec.rb +0 -128
  314. data/spec/mongoid/relations/metadata_spec.rb +0 -1985
  315. data/spec/mongoid/relations/options_spec.rb +0 -35
  316. data.tar.gz.sig +0 -0
  317. metadata.gz.sig +0 -0
  318. /data/spec/mongoid/{relations → association}/embedded/dirty_spec.rb +0 -0
@@ -1,1179 +0,0 @@
1
- # encoding: utf-8
2
- module Mongoid
3
- module Relations
4
-
5
- # The "Grand Poobah" of information about any relation is this class. It
6
- # contains everything you could ever possible want to know.
7
- class Metadata < Hash
8
-
9
- delegate :foreign_key_default, :stores_foreign_key?, to: :relation
10
-
11
- # Returns the as option of the relation.
12
- #
13
- # @example Get the as option.
14
- # metadata.as
15
- #
16
- # @return [ true, false ] The as option.
17
- #
18
- # @since 2.1.0
19
- def as
20
- self[:as]
21
- end
22
-
23
- # Tells whether an as option exists.
24
- #
25
- # @example Is the as option set?
26
- # metadata.as?
27
- #
28
- # @return [ true, false ] True if an as exists, false if not.
29
- #
30
- # @since 2.0.0.rc.1
31
- def as?
32
- !!as
33
- end
34
-
35
- # Is the relation autobuilding if accessed via the getter and the
36
- # document is new.
37
- #
38
- # @example Is the relation autobuilding?
39
- # metadata.autobuilding?
40
- #
41
- # @return [ true, false ] If the relation autobuilds.
42
- #
43
- # @since 3.0.0
44
- def autobuilding?
45
- !!self[:autobuild]
46
- end
47
-
48
- # Returns the autosave option of the relation.
49
- #
50
- # @example Get the autosave option.
51
- # metadata.autosave
52
- #
53
- # @return [ true, false ] The autosave option.
54
- #
55
- # @since 2.1.0
56
- def autosave
57
- self[:autosave]
58
- end
59
-
60
- # Does the metadata have a autosave option?
61
- #
62
- # @example Is the relation autosaving?
63
- # metadata.autosave?
64
- #
65
- # @return [ true, false ] If the relation autosaves.
66
- #
67
- # @since 2.1.0
68
- def autosave?
69
- !!autosave
70
- end
71
-
72
- # Gets a relation builder associated with the relation this metadata is
73
- # for.
74
- #
75
- # @example Get the builder.
76
- # metadata.builder(document)
77
- #
78
- # @param [ Document ] base The base document.
79
- # @param [ Object ] object A document or attributes to give the builder.
80
- #
81
- # @return [ Builder ] The builder for the relation.
82
- #
83
- # @since 2.0.0.rc.1
84
- def builder(base, object)
85
- relation.builder(base, self, object)
86
- end
87
-
88
- # Returns the name of the strategy used for handling dependent relations.
89
- #
90
- # @example Get the strategy.
91
- # metadata.cascade_strategy
92
- #
93
- # @return [ Object ] The cascading strategy to use.
94
- #
95
- # @since 2.0.0.rc.1
96
- def cascade_strategy
97
- if dependent?
98
- "Mongoid::Relations::Cascading::#{dependent.to_s.classify}".constantize
99
- end
100
- end
101
-
102
- # Is this an embedded relations that allows callbacks to cascade down to
103
- # it?
104
- #
105
- # @example Does the relation have cascading callbacks?
106
- # metadata.cascading_callbacks?
107
- #
108
- # @return [ true, false ] If the relation cascades callbacks.
109
- #
110
- # @since 2.3.0
111
- def cascading_callbacks?
112
- !!self[:cascade_callbacks]
113
- end
114
-
115
- # Returns the name of the class that this relation contains. If the
116
- # class_name was provided as an option this will return that, otherwise
117
- # it will determine the name from the name property.
118
- #
119
- # @example Get the class name.
120
- # metadata.class_name
121
- #
122
- # @return [ String ] The name of the relation's proxied class.
123
- #
124
- # @since 2.0.0.rc.1
125
- def class_name
126
- @class_name ||= (self[:class_name] || classify).sub(/\A::/,"")
127
- end
128
-
129
- # Get the foreign key contraint for the metadata.
130
- #
131
- # @example Get the constaint.
132
- # metadata.constraint
133
- #
134
- # @return [ Constraint ] The constraint.
135
- #
136
- # @since 2.0.0.rc.1
137
- def constraint
138
- @constraint ||= Constraint.new(self)
139
- end
140
-
141
- # Does the metadata have a counter cache?
142
- #
143
- # @example Is the metadata counter_cached?
144
- # metadata.counter_cached?
145
- #
146
- # @return [ true, false ] If the metadata has counter_cache
147
- #
148
- # @since 3.1.0
149
- def counter_cached?
150
- !!self[:counter_cache]
151
- end
152
-
153
- # Returns the counter cache column name
154
- #
155
- # @example Get the counter cache column.
156
- # metadata.counter_cache_column_name
157
- #
158
- # @return [ String ] The counter cache column
159
- #
160
- # @since 3.1.0
161
- def counter_cache_column_name
162
- if self[:counter_cache] == true
163
- "#{inverse || inverse_class_name.demodulize.underscore.pluralize}_count"
164
- else
165
- self[:counter_cache].to_s
166
- end
167
- end
168
-
169
- # Get the criteria that is used to query for this metadata's relation.
170
- #
171
- # @example Get the criteria.
172
- # metadata.criteria([ id_one, id_two ], Person)
173
- #
174
- # @param [ Object ] object The foreign key used for the query.
175
- # @param [ Class ] type The base class.
176
- #
177
- # @return [ Criteria ] The criteria.
178
- #
179
- # @since 2.1.0
180
- def criteria(object, type)
181
- relation.criteria(self, object, type)
182
- end
183
-
184
- # Returns the cyclic option of the relation.
185
- #
186
- # @example Get the cyclic option.
187
- # metadata.cyclic
188
- #
189
- # @return [ true, false ] The cyclic option.
190
- #
191
- # @since 2.1.0
192
- def cyclic
193
- self[:cyclic]
194
- end
195
-
196
- # Does the metadata have a cyclic option?
197
- #
198
- # @example Is the metadata cyclic?
199
- # metadata.cyclic?
200
- #
201
- # @return [ true, false ] If the metadata is cyclic.
202
- #
203
- # @since 2.1.0
204
- def cyclic?
205
- !!cyclic
206
- end
207
-
208
- # Returns the dependent option of the relation.
209
- #
210
- # @example Get the dependent option.
211
- # metadata.dependent
212
- #
213
- # @return [ Symbol ] The dependent option.
214
- #
215
- # @since 2.1.0
216
- def dependent
217
- self[:dependent]
218
- end
219
-
220
- # Does the metadata have a dependent option?
221
- #
222
- # @example Is the metadata performing cascades?
223
- # metadata.dependent?
224
- #
225
- # @return [ true, false ] If the metadata cascades.
226
- #
227
- # @since 2.1.0
228
- def dependent?
229
- !!dependent
230
- end
231
-
232
- # Will determine if the relation is an embedded one or not. Currently
233
- # only checks against embeds one and many.
234
- #
235
- # @example Is the document embedded.
236
- # metadata.embedded?
237
- #
238
- # @return [ true, false ] True if embedded, false if not.
239
- #
240
- # @since 2.0.0.rc.1
241
- def embedded?
242
- @embedded ||= (macro == :embeds_one || macro == :embeds_many)
243
- end
244
-
245
- # Returns the extension of the relation.
246
- #
247
- # @example Get the relation extension.
248
- # metadata.extension
249
- #
250
- # @return [ Module ] The extension or nil.
251
- #
252
- # @since 2.0.0.rc.1
253
- def extension
254
- self[:extend]
255
- end
256
-
257
- # Tells whether an extension definition exist for this relation.
258
- #
259
- # @example Is an extension defined?
260
- # metadata.extension?
261
- #
262
- # @return [ true, false ] True if an extension exists, false if not.
263
- #
264
- # @since 2.0.0.rc.1
265
- def extension?
266
- !!extension
267
- end
268
-
269
- # Does this metadata have a forced nil inverse_of defined. (Used in many
270
- # to manies)
271
- #
272
- # @example Is this a forced nil inverse?
273
- # metadata.forced_nil_inverse?
274
- #
275
- # @return [ true, false ] If inverse_of has been explicitly set to nil.
276
- #
277
- # @since 2.3.3
278
- def forced_nil_inverse?
279
- @forced_nil_inverse ||= has_key?(:inverse_of) && inverse_of.nil?
280
- end
281
-
282
- # Handles all the logic for figuring out what the foreign_key is for each
283
- # relations query. The logic is as follows:
284
- #
285
- # 1. If the developer defined a custom key, use that.
286
- # 2. If the relation stores a foreign key,
287
- # use the class_name_id strategy.
288
- # 3. If the relation does not store the key,
289
- # use the inverse_class_name_id strategy.
290
- #
291
- # @example Get the foreign key.
292
- # metadata.foreign_key
293
- #
294
- # @return [ String ] The foreign key for the relation.
295
- #
296
- # @since 2.0.0.rc.1
297
- def foreign_key
298
- @foreign_key ||= determine_foreign_key
299
- end
300
-
301
- # Get the name of the method to check if the foreign key has changed.
302
- #
303
- # @example Get the foreign key check method.
304
- # metadata.foreign_key_check
305
- #
306
- # @return [ String ] The foreign key check.
307
- #
308
- # @since 2.1.0
309
- def foreign_key_check
310
- @foreign_key_check ||= "#{foreign_key}_changed?"
311
- end
312
-
313
- # Returns the name of the method used to set the foreign key on a
314
- # document.
315
- #
316
- # @example Get the setter for the foreign key.
317
- # metadata.foreign_key_setter
318
- #
319
- # @return [ String ] The foreign_key plus =.
320
- #
321
- # @since 2.0.0.rc.1
322
- def foreign_key_setter
323
- @foreign_key_setter ||= "#{foreign_key}="
324
- end
325
-
326
- # Returns the index option of the relation.
327
- #
328
- # @example Get the index option.
329
- # metadata.index
330
- #
331
- # @return [ true, false ] The index option.
332
- #
333
- # @since 2.1.0
334
- def index
335
- self[:index]
336
- end
337
-
338
- # Tells whether a foreign key index exists on the relation.
339
- #
340
- # @example Is the key indexed?
341
- # metadata.indexed?
342
- #
343
- # @return [ true, false ] True if an index exists, false if not.
344
- #
345
- # @since 2.0.0.rc.1
346
- def indexed?
347
- !!index
348
- end
349
-
350
- # Instantiate new metadata for a relation.
351
- #
352
- # @example Create the new metadata.
353
- # Metadata.new(:name => :addresses)
354
- #
355
- # @param [ Hash ] properties The relation options.
356
- #
357
- # @since 2.0.0.rc.1
358
- def initialize(properties = {})
359
- Options.validate!(properties)
360
- merge!(properties)
361
- end
362
-
363
- # Since a lot of the information from the metadata is inferred and not
364
- # explicitly stored in the hash, the inspection needs to be much more
365
- # detailed.
366
- #
367
- # @example Inspect the metadata.
368
- # metadata.inspect
369
- #
370
- # @return [ String ] Oodles of information in a nice format.
371
- #
372
- # @since 2.0.0.rc.1
373
- def inspect
374
- %Q{#<Mongoid::Relations::Metadata
375
- autobuild: #{autobuilding?}
376
- class_name: #{class_name}
377
- cyclic: #{cyclic.inspect}
378
- counter_cache:#{counter_cached?}
379
- dependent: #{dependent.inspect}
380
- inverse_of: #{inverse_of.inspect}
381
- key: #{key}
382
- macro: #{macro}
383
- name: #{name}
384
- order: #{order.inspect}
385
- polymorphic: #{polymorphic?}
386
- relation: #{relation}
387
- setter: #{setter}>
388
- }
389
- end
390
-
391
- # Get the name of the inverse relations if they exists. If this is a
392
- # polymorphic relation then just return the :as option that was defined.
393
- #
394
- # @example Get the names of the inverses.
395
- # metadata.inverses
396
- #
397
- # @param [ Document ] other The document to aid in the discovery.
398
- #
399
- # @return [ Array<Symbol> ] The inverse name.
400
- def inverses(other = nil)
401
- if self[:polymorphic]
402
- lookup_inverses(other)
403
- else
404
- @inverses ||= determine_inverses
405
- end
406
- end
407
-
408
- # Get the name of the inverse relation if it exists. If this is a
409
- # polymorphic relation then just return the :as option that was defined.
410
- #
411
- # @example Get the name of the inverse.
412
- # metadata.inverse
413
- #
414
- # @param [ Document ] other The document to aid in the discovery.
415
- #
416
- # @return [ Symbol ] The inverse name.
417
- #
418
- # @since 2.0.0.rc.1
419
- def inverse(other = nil)
420
- invs = inverses(other)
421
- invs.first if invs.count == 1
422
- end
423
-
424
- # Returns the inverse_class_name option of the relation.
425
- #
426
- # @example Get the inverse_class_name option.
427
- # metadata.inverse_class_name
428
- #
429
- # @return [ true, false ] The inverse_class_name option.
430
- #
431
- # @since 2.1.0
432
- def inverse_class_name
433
- self[:inverse_class_name]
434
- end
435
-
436
- # Returns the if the inverse class name option exists.
437
- #
438
- # @example Is an inverse class name defined?
439
- # metadata.inverse_class_name?
440
- #
441
- # @return [ true, false ] If the inverse if defined.
442
- #
443
- # @since 2.1.0
444
- def inverse_class_name?
445
- !!inverse_class_name
446
- end
447
-
448
- # Used for relational many to many only. This determines the name of the
449
- # foreign key field on the inverse side of the relation, since in this
450
- # case there are keys on both sides.
451
- #
452
- # @example Find the inverse foreign key
453
- # metadata.inverse_foreign_key
454
- #
455
- # @return [ String ] The foreign key on the inverse.
456
- #
457
- # @since 2.0.0.rc.1
458
- def inverse_foreign_key
459
- @inverse_foreign_key ||= determine_inverse_foreign_key
460
- end
461
-
462
- # Used for relational many to many only. This determines the name of the
463
- # foreign key field setter on the inverse side of the relation, since in this
464
- # case there are keys on both sides.
465
- #
466
- # @example Find the inverse foreign key setter.
467
- # metadata.inverse_foreign_key_setter
468
- #
469
- # @return [ String ] The foreign key setter on the inverse.
470
- #
471
- # @since 2.0.0.rc.1
472
- def inverse_foreign_key_setter
473
- @inverse_foreign_key_setter ||= "#{inverse_foreign_key}="
474
- end
475
-
476
- # Returns the inverse class of the proxied relation.
477
- #
478
- # @example Get the inverse class.
479
- # metadata.inverse_klass
480
- #
481
- # @return [ Class ] The class of the inverse of the relation.
482
- #
483
- # @since 2.0.0.rc.1
484
- def inverse_klass
485
- @inverse_klass ||= inverse_class_name.constantize
486
- end
487
-
488
- # Get the metadata for the inverse relation.
489
- #
490
- # @example Get the inverse metadata.
491
- # metadata.inverse_metadata(doc)
492
- #
493
- # @param [ Document, Class ] object The document or class.
494
- #
495
- # @return [ Metadata ] The inverse metadata.
496
- #
497
- # @since 2.1.0
498
- def inverse_metadata(object)
499
- object.reflect_on_association(inverse(object))
500
- end
501
-
502
- # Returns the inverse_of option of the relation.
503
- #
504
- # @example Get the inverse_of option.
505
- # metadata.inverse_of
506
- #
507
- # @return [ true, false ] The inverse_of option.
508
- #
509
- # @since 2.1.0
510
- def inverse_of
511
- self[:inverse_of]
512
- end
513
-
514
- # Does the metadata have a inverse_of option?
515
- #
516
- # @example Is an inverse_of defined?
517
- # metadata.inverse_of?
518
- #
519
- # @return [ true, false ] If the relation has an inverse_of defined.
520
- #
521
- # @since 2.1.0
522
- def inverse_of?
523
- !!inverse_of
524
- end
525
-
526
- # Returns the setter for the inverse side of the relation.
527
- #
528
- # @example Get the inverse setter.
529
- # metadata.inverse_setter
530
- #
531
- # @param [ Document ] other A document to aid in the discovery.
532
- #
533
- # @return [ String ] The inverse setter name.
534
- #
535
- # @since 2.0.0.rc.1
536
- def inverse_setter(other = nil)
537
- inverse(other).__setter__
538
- end
539
-
540
- # Returns the name of the field in which to store the name of the class
541
- # for the polymorphic relation.
542
- #
543
- # @example Get the name of the field.
544
- # metadata.inverse_type
545
- #
546
- # @return [ String ] The name of the field for storing the type.
547
- #
548
- # @since 2.0.0.rc.1
549
- def inverse_type
550
- @inverse_type ||= determine_inverse_for(:type)
551
- end
552
-
553
- # Gets the setter for the field that sets the type of document on a
554
- # polymorphic relation.
555
- #
556
- # @example Get the inverse type setter.
557
- # metadata.inverse_type_setter
558
- #
559
- # @return [ String ] The name of the setter.
560
- #
561
- # @since 2.0.0.rc.1
562
- def inverse_type_setter
563
- @inverse_type_setter ||= inverse_type.__setter__
564
- end
565
-
566
- # This returns the key that is to be used to grab the attributes for the
567
- # relation or the foreign key or id that a referenced relation will use
568
- # to query for the object.
569
- #
570
- # @example Get the lookup key.
571
- # metadata.key
572
- #
573
- # @return [ String ] The association name, foreign key name, or _id.
574
- #
575
- # @since 2.0.0.rc.1
576
- def key
577
- @key ||= determine_key
578
- end
579
-
580
- # Returns the class of the proxied relation.
581
- #
582
- # @example Get the class.
583
- # metadata.klass
584
- #
585
- # @return [ Class ] The class of the relation.
586
- #
587
- # @since 2.0.0.rc.1
588
- def klass
589
- @klass ||= class_name.constantize
590
- end
591
-
592
- # Is this metadata representing a one to many or many to many relation?
593
- #
594
- # @example Is the relation a many?
595
- # metadata.many?
596
- #
597
- # @return [ true, false ] If the relation is a many.
598
- #
599
- # @since 2.1.6
600
- def many?
601
- @many ||= (relation.macro.to_s =~ /many/)
602
- end
603
-
604
- # Returns the macro for the relation of this metadata.
605
- #
606
- # @example Get the macro.
607
- # metadata.macro
608
- #
609
- # @return [ Symbol ] The macro.
610
- #
611
- # @since 2.0.0.rc.1
612
- def macro
613
- relation.macro
614
- end
615
-
616
- # Get the name associated with this metadata.
617
- #
618
- # @example Get the name.
619
- # metadata.name
620
- #
621
- # @return [ Symbol ] The name.
622
- #
623
- # @since 2.1.0
624
- def name
625
- self[:name]
626
- end
627
-
628
- # Is the name defined?
629
- #
630
- # @example Is the name defined?
631
- # metadata.name?
632
- #
633
- # @return [ true, false ] If the name is defined.
634
- #
635
- # @since 2.1.0
636
- def name?
637
- !!name
638
- end
639
-
640
- # Does the relation have a destructive dependent option specified. This
641
- # is true for :dependent => :delete and :dependent => :destroy.
642
- #
643
- # @example Is the relation destructive?
644
- # metadata.destructive?
645
- #
646
- # @return [ true, false ] If the relation is destructive.
647
- #
648
- # @since 2.1.0
649
- def destructive?
650
- @destructive ||= (dependent == :delete || dependent == :destroy)
651
- end
652
-
653
- # Gets a relation nested builder associated with the relation this metadata
654
- # is for. Nested builders are used in conjunction with nested attributes.
655
- #
656
- # @example Get the nested builder.
657
- # metadata.nested_builder(attributes, options)
658
- #
659
- # @param [ Hash ] attributes The attributes to build the relation with.
660
- # @param [ Hash ] options Options for the nested builder.
661
- #
662
- # @return [ NestedBuilder ] The nested builder for the relation.
663
- #
664
- # @since 2.0.0.rc.1
665
- def nested_builder(attributes, options)
666
- if polymorphic? && options[:class_name]
667
- self[:class_name] = options[:class_name]
668
- end
669
- relation.nested_builder(self, attributes, options)
670
- end
671
-
672
- # Get the path calculator for the supplied document.
673
- #
674
- # @example Get the path calculator.
675
- # metadata.path(document)
676
- #
677
- # @param [ Document ] document The document to calculate on.
678
- #
679
- # @return [ Object ] The atomic path calculator.
680
- #
681
- # @since 2.1.0
682
- def path(document)
683
- relation.path(document)
684
- end
685
-
686
- # Returns true if the relation is polymorphic.
687
- #
688
- # @example Is the relation polymorphic?
689
- # metadata.polymorphic?
690
- #
691
- # @return [ true, false ] True if the relation is polymorphic, false if not.
692
- #
693
- # @since 2.0.0.rc.1
694
- def polymorphic?
695
- @polymorphic ||= (!!self[:as] || !!self[:polymorphic])
696
- end
697
-
698
- # Get the primary key field for finding the related document.
699
- #
700
- # @example Get the primary key.
701
- # metadata.primary_key
702
- #
703
- # @return [ String ] The primary key field.
704
- #
705
- # @since 3.1.0
706
- def primary_key
707
- @primary_key ||= (self[:primary_key] || "_id").to_s
708
- end
709
-
710
- # Get the relation associated with this metadata.
711
- #
712
- # @example Get the relation.
713
- # metadata.relation
714
- #
715
- # @return [ Proxy ] The relation proxy class.
716
- #
717
- # @since 2.1.0
718
- def relation
719
- self[:relation]
720
- end
721
-
722
- # Gets the method name used to set this relation.
723
- #
724
- # @example Get the setter.
725
- # metadata = Metadata.new(:name => :person)
726
- # metadata.setter # => "person="
727
- #
728
- # @return [ String ] The name plus "=".
729
- #
730
- # @since 2.0.0.rc.1
731
- def setter
732
- @setter ||= "#{name}="
733
- end
734
-
735
- # Returns the name of the field in which to store the name of the class
736
- # for the polymorphic relation.
737
- #
738
- # @example Get the name of the field.
739
- # metadata.inverse_type
740
- #
741
- # @return [ String ] The name of the field for storing the type.
742
- #
743
- # @since 2.0.0.rc.1
744
- def type
745
- @type ||= polymorphic? ? "#{as}_type" : nil
746
- end
747
-
748
- # Gets the setter for the field that sets the type of document on a
749
- # polymorphic relation.
750
- #
751
- # @example Get the inverse type setter.
752
- # metadata.inverse_type_setter
753
- #
754
- # @return [ String ] The name of the setter.
755
- #
756
- # @since 2.0.0.rc.1
757
- def type_setter
758
- @type_setter ||= type.__setter__
759
- end
760
-
761
-
762
- # Key where embedded document is save.
763
- # By default is the name of relation
764
- #
765
- # @return [ String ] the name of key where save
766
- #
767
- # @since 3.0.0
768
- def store_as
769
- @store_as ||= (self[:store_as].try(:to_s) || name.to_s)
770
- end
771
-
772
- # Are we validating this relation automatically?
773
- #
774
- # @example Is automatic validation on?
775
- # metadata.validate?
776
- #
777
- # @return [ true, false ] True unless explictly set to false.
778
- #
779
- # @since 2.0.0.rc.1
780
- def validate?
781
- if self[:validate].nil?
782
- self[:validate] = relation.validation_default
783
- else
784
- self[:validate]
785
- end
786
- end
787
-
788
- # Returns the metadata itself. Here for compatibility with Rails
789
- # association metadata.
790
- #
791
- # @example Get the options.
792
- # metadata.options
793
- #
794
- # @return [ Metadata ] self.
795
- #
796
- # @since 2.4.6
797
- def options
798
- self
799
- end
800
-
801
- # Returns default order for this association.
802
- #
803
- # @example Get default order
804
- # metadata.order
805
- #
806
- # @return [ Criterion::Complex, nil] nil if doesn't set
807
- #
808
- # @since 2.1.0
809
- def order
810
- self[:order]
811
- end
812
-
813
- # Is a default order set?
814
- #
815
- # @example Is the order set?
816
- # metadata.order?
817
- #
818
- # @return [ true, false ] If the order is set.
819
- #
820
- # @since 2.1.0
821
- def order?
822
- !!order
823
- end
824
-
825
- # Is this relation touchable?
826
- #
827
- # @example Is the relation touchable?
828
- # metadata.touchable?
829
- #
830
- # @return [ true, false ] If the relation can be touched.
831
- #
832
- # @since 3.0.0
833
- def touchable?
834
- !!self[:touch]
835
- end
836
-
837
- # Returns the metadata class types.
838
- #
839
- # @example Get the relation class types.
840
- # metadata.type_relation
841
- #
842
- # @return [ Hash ] The hash with relation class types.
843
- #
844
- # @since 3.1.0
845
- def type_relation
846
- { _type: { "$in" => klass._types }}
847
- end
848
-
849
- private
850
-
851
- # Returns the class name for the relation.
852
- #
853
- # @example Get the class name.
854
- # metadata.classify
855
- #
856
- # @return [ String ] The classified name.
857
- #
858
- # @since 2.0.0.rc.1
859
- def classify
860
- @classify ||= "#{find_module}::#{name.to_s.classify}"
861
- end
862
-
863
- # Get the name for the inverse field.
864
- #
865
- # @api private
866
- #
867
- # @example Get the inverse field name.
868
- # metadata.determine_inverse_for(:type)
869
- #
870
- # @param [ Symbol ] field The inverse field name.
871
- #
872
- # @return [ String ] The name of the field.
873
- #
874
- # @since 3.0.0
875
- def determine_inverse_for(field)
876
- relation.stores_foreign_key? && polymorphic? ? "#{name}_#{field}" : nil
877
- end
878
-
879
- # Deterimene the inverses that can be memoized.
880
- #
881
- # @api private
882
- #
883
- # @example Determin the inverses.
884
- # metadata.determine_inverses
885
- #
886
- # @return [ Array<Symbol> ] The inverses.
887
- #
888
- # @since 3.0.0
889
- def determine_inverses
890
- return [ inverse_of ] if has_key?(:inverse_of)
891
- return [ as ] if has_key?(:as)
892
- return [ cyclic_inverse ] if self[:cyclic]
893
- [ inverse_relation ]
894
- end
895
-
896
- # Find the module the class with the specific name is in.
897
- # This is done by starting at the inverse_class_name's
898
- # module and stepping down to see where it is defined.
899
- #
900
- # @api private
901
- #
902
- # @example Find the module.
903
- # metadata.find_module
904
- #
905
- # @return [ String ] The module.
906
- #
907
- # @since 3.0.0
908
- def find_module
909
- if inverse_class_name.present?
910
- parts = inverse_class_name.split('::')
911
- modules = parts.size.times.map { |i| parts.first(i).join('::') }.reverse
912
- find_from_parts(modules)
913
- end
914
- end
915
-
916
- # Find the modules from a reversed list.
917
- #
918
- # @api private
919
- #
920
- # @example Find the module from the parts.
921
- # metadata.find_from_parts([ "Namespace", "Module" ])
922
- #
923
- # @param [ Array<String> ] modules The modules.
924
- #
925
- # @return [ String ] The matching module.
926
- #
927
- # @since 3.0.0
928
- def find_from_parts(modules)
929
- modules.find do |mod|
930
- if mod.blank?
931
- false
932
- else
933
- ActiveSupport::Inflector.constantize(mod).constants.include?(
934
- name.to_s.classify.to_sym
935
- )
936
- end
937
- end
938
- end
939
-
940
- # Get the name of the inverse relation in a cyclic relation.
941
- #
942
- # @example Get the cyclic inverse name.
943
- #
944
- # class Role
945
- # include Mongoid::Document
946
- # embedded_in :parent_role, :cyclic => true
947
- # embeds_many :child_roles, :cyclic => true
948
- # end
949
- #
950
- # metadata = Metadata.new(:name => :parent_role)
951
- # metadata.cyclic_inverse # => "child_roles"
952
- #
953
- # @return [ String ] The cyclic inverse name.
954
- #
955
- # @since 2.0.0.rc.1
956
- def cyclic_inverse
957
- @cyclic_inverse ||= determine_cyclic_inverse
958
- end
959
-
960
- # Determine the cyclic inverse. Performance improvement with the
961
- # memoization.
962
- #
963
- # @example Determine the inverse.
964
- # metadata.determine_cyclic_inverse
965
- #
966
- # @return [ String ] The cyclic inverse name.
967
- #
968
- # @since 2.0.0.rc.1
969
- def determine_cyclic_inverse
970
- underscored = class_name.demodulize.underscore
971
- klass.relations.each_pair do |key, meta|
972
- if key =~ /#{underscored.singularize}|#{underscored.pluralize}/ &&
973
- meta.relation != relation
974
- return key.to_sym
975
- end
976
- end
977
- end
978
-
979
- # Determine the value for the relation's foreign key. Performance
980
- # improvement.
981
- #
982
- # @example Determine the foreign key.
983
- # metadata.determine_foreign_key
984
- #
985
- # @return [ String ] The foreign key.
986
- #
987
- # @since 2.0.0.rc.1
988
- def determine_foreign_key
989
- return self[:foreign_key].to_s if self[:foreign_key]
990
- return nil if relation.embedded?
991
-
992
- if relation.stores_foreign_key?
993
- relation.foreign_key(name)
994
- else
995
- suffix = relation.foreign_key_suffix
996
- "#{inverse}#{suffix}"
997
- end
998
- end
999
-
1000
- # Determine the inverse foreign key of the relation.
1001
- #
1002
- # @example Determine the inverse foreign key.
1003
- # metadata.determine_inverse_foreign_key
1004
- #
1005
- # @return [ String ] The inverse.
1006
- #
1007
- # @since 2.3.2
1008
- def determine_inverse_foreign_key
1009
- if has_key?(:inverse_of)
1010
- inverse_of ? "#{inverse_of.to_s.singularize}#{relation.foreign_key_suffix}" : nil
1011
- else
1012
- "#{inverse_class_name.demodulize.underscore}#{relation.foreign_key_suffix}"
1013
- end
1014
- end
1015
-
1016
- # Determine the inverse relation. Memoizing #inverse_relation and adding
1017
- # this method dropped 5 seconds off the test suite as a performance
1018
- # improvement.
1019
- #
1020
- # @example Determine the inverse.
1021
- # metadata.determine_inverse_relation
1022
- #
1023
- # @return [ Symbol ] The name of the inverse.
1024
- #
1025
- # @since 2.0.0.rc.1
1026
- def determine_inverse_relation
1027
- default = foreign_key_match || klass.relations[inverse_klass.name.underscore]
1028
- return default.name if default
1029
- names = inverse_relation_candidate_names
1030
- if names.size > 1
1031
- raise Errors::AmbiguousRelationship.new(klass, inverse_klass, name, names)
1032
- end
1033
- names.first
1034
- end
1035
-
1036
- # Return metadata where the foreign key matches the foreign key on this
1037
- # relation.
1038
- #
1039
- # @api private
1040
- #
1041
- # @example Return a foreign key match.
1042
- # meta.foreign_key_match
1043
- #
1044
- # @return [ Metadata ] A match, if any.
1045
- #
1046
- # @since 2.4.11
1047
- def foreign_key_match
1048
- if fk = self[:foreign_key]
1049
- relations_metadata.detect do |meta|
1050
- fk == meta.foreign_key if meta.stores_foreign_key?
1051
- end
1052
- end
1053
- end
1054
-
1055
- # Get the inverse relation candidates.
1056
- #
1057
- # @api private
1058
- #
1059
- # @example Get the inverse relation candidates.
1060
- # metadata.inverse_relation_candidates
1061
- #
1062
- # @return [ Array<Metdata> ] The candidates.
1063
- #
1064
- # @since 3.0.0
1065
- def inverse_relation_candidates
1066
- relations_metadata.select do |meta|
1067
- next if meta.name == name
1068
- (meta.class_name == inverse_class_name) && !meta.forced_nil_inverse?
1069
- end
1070
- end
1071
-
1072
- # Get the candidates for inverse relations.
1073
- #
1074
- # @api private
1075
- #
1076
- # @example Get the candidates.
1077
- # metadata.inverse_relation_candidates
1078
- #
1079
- # @return [ Array<Symbol> ] The candidates.
1080
- #
1081
- # @since 3.0.0
1082
- def inverse_relation_candidate_names
1083
- @candidate_names ||= inverse_relation_candidates.map(&:name)
1084
- end
1085
-
1086
- # Determine the key for the relation in the attributes.
1087
- #
1088
- # @example Get the key.
1089
- # metadata.determine_key
1090
- #
1091
- # @return [ String ] The key in the attributes.
1092
- #
1093
- # @since 2.0.0.rc.1
1094
- def determine_key
1095
- return store_as.to_s if relation.embedded?
1096
- relation.stores_foreign_key? ? foreign_key : primary_key
1097
- end
1098
-
1099
- # Determine the name of the inverse relation.
1100
- #
1101
- # @example Get the inverse name.
1102
- # metadata.inverse_relation
1103
- #
1104
- # @return [ Symbol ] The name of the inverse relation.
1105
- #
1106
- # @since 2.0.0.rc.1
1107
- def inverse_relation
1108
- @inverse_relation ||= determine_inverse_relation
1109
- end
1110
-
1111
- # Infer the name of the inverse relation from the class.
1112
- #
1113
- # @example Get the inverse name
1114
- # metadata.inverse_name
1115
- #
1116
- # @return [ String ] The inverse class name underscored.
1117
- #
1118
- # @since 2.0.0.rc.1
1119
- def inverse_name
1120
- @inverse_name ||= inverse_klass.name.underscore
1121
- end
1122
-
1123
- # For polymorphic children, we need to figure out the inverse from the
1124
- # actual instance on the other side, since we cannot know the exact class
1125
- # name to infer it from at load time.
1126
- #
1127
- # @example Find the inverses.
1128
- # metadata.lookup_inverses(other)
1129
- #
1130
- # @param [ Document ] other The inverse document.
1131
- #
1132
- # @return [ Array<String> ] The inverse names.
1133
- def lookup_inverses(other)
1134
- return [ inverse_of ] if inverse_of
1135
- if other
1136
- matches = []
1137
- other.class.relations.values.each do |meta|
1138
- if meta.as == name && meta.class_name == inverse_class_name
1139
- matches.push(meta.name)
1140
- end
1141
- end
1142
- matches
1143
- end
1144
- end
1145
-
1146
- # For polymorphic children, we need to figure out the inverse from the
1147
- # actual instance on the other side, since we cannot know the exact class
1148
- # name to infer it from at load time.
1149
- #
1150
- # @example Find the inverse.
1151
- # metadata.lookup_inverse(other)
1152
- #
1153
- # @param [ Document ] other The inverse document.
1154
- #
1155
- # @return [ String ] The inverse name.
1156
- #
1157
- # @since 2.0.0.rc.1
1158
- def lookup_inverse(other)
1159
- if invs = lookup_inverses(other) && invs.count == 1
1160
- invs.first
1161
- end
1162
- end
1163
-
1164
- # Get the relation metadata only.
1165
- #
1166
- # @api private
1167
- #
1168
- # @example Get the relation metadata.
1169
- # metadata.relations_metadata
1170
- #
1171
- # @return [ Array<Metadata> ] The metadata.
1172
- #
1173
- # @since 3.0.0
1174
- def relations_metadata
1175
- klass.relations.values
1176
- end
1177
- end
1178
- end
1179
- end