mongoid 6.4.2 → 7.0.3

Sign up to get free protection for your applications and to get access to all the features.
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