omg-activerecord 8.0.0.alpha1

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 (412) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +355 -0
  3. data/MIT-LICENSE +22 -0
  4. data/README.rdoc +219 -0
  5. data/examples/performance.rb +185 -0
  6. data/examples/simple.rb +15 -0
  7. data/lib/active_record/aggregations.rb +287 -0
  8. data/lib/active_record/association_relation.rb +50 -0
  9. data/lib/active_record/associations/alias_tracker.rb +90 -0
  10. data/lib/active_record/associations/association.rb +417 -0
  11. data/lib/active_record/associations/association_scope.rb +175 -0
  12. data/lib/active_record/associations/belongs_to_association.rb +163 -0
  13. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +50 -0
  14. data/lib/active_record/associations/builder/association.rb +170 -0
  15. data/lib/active_record/associations/builder/belongs_to.rb +160 -0
  16. data/lib/active_record/associations/builder/collection_association.rb +80 -0
  17. data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +107 -0
  18. data/lib/active_record/associations/builder/has_many.rb +23 -0
  19. data/lib/active_record/associations/builder/has_one.rb +61 -0
  20. data/lib/active_record/associations/builder/singular_association.rb +48 -0
  21. data/lib/active_record/associations/collection_association.rb +535 -0
  22. data/lib/active_record/associations/collection_proxy.rb +1163 -0
  23. data/lib/active_record/associations/disable_joins_association_scope.rb +59 -0
  24. data/lib/active_record/associations/errors.rb +265 -0
  25. data/lib/active_record/associations/foreign_association.rb +40 -0
  26. data/lib/active_record/associations/has_many_association.rb +167 -0
  27. data/lib/active_record/associations/has_many_through_association.rb +232 -0
  28. data/lib/active_record/associations/has_one_association.rb +142 -0
  29. data/lib/active_record/associations/has_one_through_association.rb +45 -0
  30. data/lib/active_record/associations/join_dependency/join_association.rb +106 -0
  31. data/lib/active_record/associations/join_dependency/join_base.rb +23 -0
  32. data/lib/active_record/associations/join_dependency/join_part.rb +71 -0
  33. data/lib/active_record/associations/join_dependency.rb +301 -0
  34. data/lib/active_record/associations/nested_error.rb +47 -0
  35. data/lib/active_record/associations/preloader/association.rb +316 -0
  36. data/lib/active_record/associations/preloader/batch.rb +48 -0
  37. data/lib/active_record/associations/preloader/branch.rb +153 -0
  38. data/lib/active_record/associations/preloader/through_association.rb +150 -0
  39. data/lib/active_record/associations/preloader.rb +135 -0
  40. data/lib/active_record/associations/singular_association.rb +76 -0
  41. data/lib/active_record/associations/through_association.rb +132 -0
  42. data/lib/active_record/associations.rb +1897 -0
  43. data/lib/active_record/asynchronous_queries_tracker.rb +64 -0
  44. data/lib/active_record/attribute_assignment.rb +82 -0
  45. data/lib/active_record/attribute_methods/before_type_cast.rb +106 -0
  46. data/lib/active_record/attribute_methods/composite_primary_key.rb +84 -0
  47. data/lib/active_record/attribute_methods/dirty.rb +262 -0
  48. data/lib/active_record/attribute_methods/primary_key.rb +158 -0
  49. data/lib/active_record/attribute_methods/query.rb +50 -0
  50. data/lib/active_record/attribute_methods/read.rb +46 -0
  51. data/lib/active_record/attribute_methods/serialization.rb +232 -0
  52. data/lib/active_record/attribute_methods/time_zone_conversion.rb +94 -0
  53. data/lib/active_record/attribute_methods/write.rb +49 -0
  54. data/lib/active_record/attribute_methods.rb +542 -0
  55. data/lib/active_record/attributes.rb +307 -0
  56. data/lib/active_record/autosave_association.rb +586 -0
  57. data/lib/active_record/base.rb +338 -0
  58. data/lib/active_record/callbacks.rb +452 -0
  59. data/lib/active_record/coders/column_serializer.rb +61 -0
  60. data/lib/active_record/coders/json.rb +15 -0
  61. data/lib/active_record/coders/yaml_column.rb +95 -0
  62. data/lib/active_record/connection_adapters/abstract/connection_handler.rb +290 -0
  63. data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +210 -0
  64. data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +78 -0
  65. data/lib/active_record/connection_adapters/abstract/connection_pool.rb +923 -0
  66. data/lib/active_record/connection_adapters/abstract/database_limits.rb +31 -0
  67. data/lib/active_record/connection_adapters/abstract/database_statements.rb +747 -0
  68. data/lib/active_record/connection_adapters/abstract/query_cache.rb +319 -0
  69. data/lib/active_record/connection_adapters/abstract/quoting.rb +239 -0
  70. data/lib/active_record/connection_adapters/abstract/savepoints.rb +24 -0
  71. data/lib/active_record/connection_adapters/abstract/schema_creation.rb +190 -0
  72. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +961 -0
  73. data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +106 -0
  74. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +1883 -0
  75. data/lib/active_record/connection_adapters/abstract/transaction.rb +676 -0
  76. data/lib/active_record/connection_adapters/abstract_adapter.rb +1218 -0
  77. data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +1016 -0
  78. data/lib/active_record/connection_adapters/column.rb +122 -0
  79. data/lib/active_record/connection_adapters/deduplicable.rb +29 -0
  80. data/lib/active_record/connection_adapters/mysql/column.rb +28 -0
  81. data/lib/active_record/connection_adapters/mysql/database_statements.rb +95 -0
  82. data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +71 -0
  83. data/lib/active_record/connection_adapters/mysql/quoting.rb +114 -0
  84. data/lib/active_record/connection_adapters/mysql/schema_creation.rb +106 -0
  85. data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +106 -0
  86. data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +97 -0
  87. data/lib/active_record/connection_adapters/mysql/schema_statements.rb +300 -0
  88. data/lib/active_record/connection_adapters/mysql/type_metadata.rb +40 -0
  89. data/lib/active_record/connection_adapters/mysql2/database_statements.rb +96 -0
  90. data/lib/active_record/connection_adapters/mysql2_adapter.rb +196 -0
  91. data/lib/active_record/connection_adapters/pool_config.rb +83 -0
  92. data/lib/active_record/connection_adapters/pool_manager.rb +57 -0
  93. data/lib/active_record/connection_adapters/postgresql/column.rb +82 -0
  94. data/lib/active_record/connection_adapters/postgresql/database_statements.rb +231 -0
  95. data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
  96. data/lib/active_record/connection_adapters/postgresql/oid/array.rb +91 -0
  97. data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +53 -0
  98. data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +15 -0
  99. data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +17 -0
  100. data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +54 -0
  101. data/lib/active_record/connection_adapters/postgresql/oid/date.rb +31 -0
  102. data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +36 -0
  103. data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +15 -0
  104. data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +20 -0
  105. data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +109 -0
  106. data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +15 -0
  107. data/lib/active_record/connection_adapters/postgresql/oid/interval.rb +49 -0
  108. data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +15 -0
  109. data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +44 -0
  110. data/lib/active_record/connection_adapters/postgresql/oid/macaddr.rb +25 -0
  111. data/lib/active_record/connection_adapters/postgresql/oid/money.rb +42 -0
  112. data/lib/active_record/connection_adapters/postgresql/oid/oid.rb +15 -0
  113. data/lib/active_record/connection_adapters/postgresql/oid/point.rb +74 -0
  114. data/lib/active_record/connection_adapters/postgresql/oid/range.rb +124 -0
  115. data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +18 -0
  116. data/lib/active_record/connection_adapters/postgresql/oid/timestamp.rb +15 -0
  117. data/lib/active_record/connection_adapters/postgresql/oid/timestamp_with_time_zone.rb +30 -0
  118. data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +125 -0
  119. data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +45 -0
  120. data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +28 -0
  121. data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +30 -0
  122. data/lib/active_record/connection_adapters/postgresql/oid.rb +38 -0
  123. data/lib/active_record/connection_adapters/postgresql/quoting.rb +238 -0
  124. data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +71 -0
  125. data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +169 -0
  126. data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +392 -0
  127. data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +127 -0
  128. data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +1162 -0
  129. data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +44 -0
  130. data/lib/active_record/connection_adapters/postgresql/utils.rb +79 -0
  131. data/lib/active_record/connection_adapters/postgresql_adapter.rb +1182 -0
  132. data/lib/active_record/connection_adapters/schema_cache.rb +478 -0
  133. data/lib/active_record/connection_adapters/sql_type_metadata.rb +45 -0
  134. data/lib/active_record/connection_adapters/sqlite3/column.rb +62 -0
  135. data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +145 -0
  136. data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
  137. data/lib/active_record/connection_adapters/sqlite3/quoting.rb +116 -0
  138. data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +37 -0
  139. data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +39 -0
  140. data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +47 -0
  141. data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +221 -0
  142. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +843 -0
  143. data/lib/active_record/connection_adapters/statement_pool.rb +67 -0
  144. data/lib/active_record/connection_adapters/trilogy/database_statements.rb +69 -0
  145. data/lib/active_record/connection_adapters/trilogy_adapter.rb +212 -0
  146. data/lib/active_record/connection_adapters.rb +176 -0
  147. data/lib/active_record/connection_handling.rb +413 -0
  148. data/lib/active_record/core.rb +836 -0
  149. data/lib/active_record/counter_cache.rb +230 -0
  150. data/lib/active_record/database_configurations/connection_url_resolver.rb +105 -0
  151. data/lib/active_record/database_configurations/database_config.rb +104 -0
  152. data/lib/active_record/database_configurations/hash_config.rb +172 -0
  153. data/lib/active_record/database_configurations/url_config.rb +78 -0
  154. data/lib/active_record/database_configurations.rb +309 -0
  155. data/lib/active_record/delegated_type.rb +289 -0
  156. data/lib/active_record/deprecator.rb +7 -0
  157. data/lib/active_record/destroy_association_async_job.rb +38 -0
  158. data/lib/active_record/disable_joins_association_relation.rb +39 -0
  159. data/lib/active_record/dynamic_matchers.rb +121 -0
  160. data/lib/active_record/encryption/auto_filtered_parameters.rb +66 -0
  161. data/lib/active_record/encryption/cipher/aes256_gcm.rb +101 -0
  162. data/lib/active_record/encryption/cipher.rb +53 -0
  163. data/lib/active_record/encryption/config.rb +70 -0
  164. data/lib/active_record/encryption/configurable.rb +60 -0
  165. data/lib/active_record/encryption/context.rb +42 -0
  166. data/lib/active_record/encryption/contexts.rb +76 -0
  167. data/lib/active_record/encryption/derived_secret_key_provider.rb +18 -0
  168. data/lib/active_record/encryption/deterministic_key_provider.rb +14 -0
  169. data/lib/active_record/encryption/encryptable_record.rb +230 -0
  170. data/lib/active_record/encryption/encrypted_attribute_type.rb +184 -0
  171. data/lib/active_record/encryption/encrypted_fixtures.rb +38 -0
  172. data/lib/active_record/encryption/encrypting_only_encryptor.rb +12 -0
  173. data/lib/active_record/encryption/encryptor.rb +177 -0
  174. data/lib/active_record/encryption/envelope_encryption_key_provider.rb +55 -0
  175. data/lib/active_record/encryption/errors.rb +15 -0
  176. data/lib/active_record/encryption/extended_deterministic_queries.rb +159 -0
  177. data/lib/active_record/encryption/extended_deterministic_uniqueness_validator.rb +28 -0
  178. data/lib/active_record/encryption/key.rb +28 -0
  179. data/lib/active_record/encryption/key_generator.rb +53 -0
  180. data/lib/active_record/encryption/key_provider.rb +46 -0
  181. data/lib/active_record/encryption/message.rb +33 -0
  182. data/lib/active_record/encryption/message_pack_message_serializer.rb +76 -0
  183. data/lib/active_record/encryption/message_serializer.rb +96 -0
  184. data/lib/active_record/encryption/null_encryptor.rb +25 -0
  185. data/lib/active_record/encryption/properties.rb +76 -0
  186. data/lib/active_record/encryption/read_only_null_encryptor.rb +28 -0
  187. data/lib/active_record/encryption/scheme.rb +107 -0
  188. data/lib/active_record/encryption.rb +58 -0
  189. data/lib/active_record/enum.rb +424 -0
  190. data/lib/active_record/errors.rb +614 -0
  191. data/lib/active_record/explain.rb +63 -0
  192. data/lib/active_record/explain_registry.rb +37 -0
  193. data/lib/active_record/explain_subscriber.rb +34 -0
  194. data/lib/active_record/fixture_set/file.rb +89 -0
  195. data/lib/active_record/fixture_set/model_metadata.rb +42 -0
  196. data/lib/active_record/fixture_set/render_context.rb +19 -0
  197. data/lib/active_record/fixture_set/table_row.rb +208 -0
  198. data/lib/active_record/fixture_set/table_rows.rb +46 -0
  199. data/lib/active_record/fixtures.rb +850 -0
  200. data/lib/active_record/future_result.rb +182 -0
  201. data/lib/active_record/gem_version.rb +17 -0
  202. data/lib/active_record/inheritance.rb +366 -0
  203. data/lib/active_record/insert_all.rb +328 -0
  204. data/lib/active_record/integration.rb +209 -0
  205. data/lib/active_record/internal_metadata.rb +164 -0
  206. data/lib/active_record/legacy_yaml_adapter.rb +15 -0
  207. data/lib/active_record/locale/en.yml +48 -0
  208. data/lib/active_record/locking/optimistic.rb +228 -0
  209. data/lib/active_record/locking/pessimistic.rb +102 -0
  210. data/lib/active_record/log_subscriber.rb +149 -0
  211. data/lib/active_record/marshalling.rb +56 -0
  212. data/lib/active_record/message_pack.rb +124 -0
  213. data/lib/active_record/middleware/database_selector/resolver/session.rb +48 -0
  214. data/lib/active_record/middleware/database_selector/resolver.rb +92 -0
  215. data/lib/active_record/middleware/database_selector.rb +87 -0
  216. data/lib/active_record/middleware/shard_selector.rb +62 -0
  217. data/lib/active_record/migration/command_recorder.rb +406 -0
  218. data/lib/active_record/migration/compatibility.rb +490 -0
  219. data/lib/active_record/migration/default_strategy.rb +22 -0
  220. data/lib/active_record/migration/execution_strategy.rb +19 -0
  221. data/lib/active_record/migration/join_table.rb +16 -0
  222. data/lib/active_record/migration/pending_migration_connection.rb +21 -0
  223. data/lib/active_record/migration.rb +1626 -0
  224. data/lib/active_record/model_schema.rb +635 -0
  225. data/lib/active_record/nested_attributes.rb +633 -0
  226. data/lib/active_record/no_touching.rb +65 -0
  227. data/lib/active_record/normalization.rb +163 -0
  228. data/lib/active_record/persistence.rb +968 -0
  229. data/lib/active_record/promise.rb +84 -0
  230. data/lib/active_record/query_cache.rb +56 -0
  231. data/lib/active_record/query_logs.rb +247 -0
  232. data/lib/active_record/query_logs_formatter.rb +30 -0
  233. data/lib/active_record/querying.rb +122 -0
  234. data/lib/active_record/railtie.rb +440 -0
  235. data/lib/active_record/railties/console_sandbox.rb +5 -0
  236. data/lib/active_record/railties/controller_runtime.rb +65 -0
  237. data/lib/active_record/railties/databases.rake +641 -0
  238. data/lib/active_record/railties/job_runtime.rb +23 -0
  239. data/lib/active_record/readonly_attributes.rb +66 -0
  240. data/lib/active_record/reflection.rb +1287 -0
  241. data/lib/active_record/relation/batches/batch_enumerator.rb +115 -0
  242. data/lib/active_record/relation/batches.rb +491 -0
  243. data/lib/active_record/relation/calculations.rb +679 -0
  244. data/lib/active_record/relation/delegation.rb +154 -0
  245. data/lib/active_record/relation/finder_methods.rb +661 -0
  246. data/lib/active_record/relation/from_clause.rb +30 -0
  247. data/lib/active_record/relation/merger.rb +192 -0
  248. data/lib/active_record/relation/predicate_builder/array_handler.rb +48 -0
  249. data/lib/active_record/relation/predicate_builder/association_query_value.rb +76 -0
  250. data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +19 -0
  251. data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +60 -0
  252. data/lib/active_record/relation/predicate_builder/range_handler.rb +22 -0
  253. data/lib/active_record/relation/predicate_builder/relation_handler.rb +24 -0
  254. data/lib/active_record/relation/predicate_builder.rb +181 -0
  255. data/lib/active_record/relation/query_attribute.rb +68 -0
  256. data/lib/active_record/relation/query_methods.rb +2235 -0
  257. data/lib/active_record/relation/record_fetch_warning.rb +52 -0
  258. data/lib/active_record/relation/spawn_methods.rb +78 -0
  259. data/lib/active_record/relation/where_clause.rb +218 -0
  260. data/lib/active_record/relation.rb +1495 -0
  261. data/lib/active_record/result.rb +249 -0
  262. data/lib/active_record/runtime_registry.rb +82 -0
  263. data/lib/active_record/sanitization.rb +254 -0
  264. data/lib/active_record/schema.rb +77 -0
  265. data/lib/active_record/schema_dumper.rb +364 -0
  266. data/lib/active_record/schema_migration.rb +106 -0
  267. data/lib/active_record/scoping/default.rb +205 -0
  268. data/lib/active_record/scoping/named.rb +202 -0
  269. data/lib/active_record/scoping.rb +136 -0
  270. data/lib/active_record/secure_password.rb +60 -0
  271. data/lib/active_record/secure_token.rb +66 -0
  272. data/lib/active_record/serialization.rb +29 -0
  273. data/lib/active_record/signed_id.rb +137 -0
  274. data/lib/active_record/statement_cache.rb +164 -0
  275. data/lib/active_record/store.rb +299 -0
  276. data/lib/active_record/suppressor.rb +59 -0
  277. data/lib/active_record/table_metadata.rb +85 -0
  278. data/lib/active_record/tasks/database_tasks.rb +681 -0
  279. data/lib/active_record/tasks/mysql_database_tasks.rb +120 -0
  280. data/lib/active_record/tasks/postgresql_database_tasks.rb +147 -0
  281. data/lib/active_record/tasks/sqlite_database_tasks.rb +89 -0
  282. data/lib/active_record/test_databases.rb +24 -0
  283. data/lib/active_record/test_fixtures.rb +321 -0
  284. data/lib/active_record/testing/query_assertions.rb +121 -0
  285. data/lib/active_record/timestamp.rb +177 -0
  286. data/lib/active_record/token_for.rb +123 -0
  287. data/lib/active_record/touch_later.rb +70 -0
  288. data/lib/active_record/transaction.rb +132 -0
  289. data/lib/active_record/transactions.rb +523 -0
  290. data/lib/active_record/translation.rb +22 -0
  291. data/lib/active_record/type/adapter_specific_registry.rb +144 -0
  292. data/lib/active_record/type/date.rb +9 -0
  293. data/lib/active_record/type/date_time.rb +9 -0
  294. data/lib/active_record/type/decimal_without_scale.rb +15 -0
  295. data/lib/active_record/type/hash_lookup_type_map.rb +57 -0
  296. data/lib/active_record/type/internal/timezone.rb +22 -0
  297. data/lib/active_record/type/json.rb +30 -0
  298. data/lib/active_record/type/serialized.rb +76 -0
  299. data/lib/active_record/type/text.rb +11 -0
  300. data/lib/active_record/type/time.rb +35 -0
  301. data/lib/active_record/type/type_map.rb +58 -0
  302. data/lib/active_record/type/unsigned_integer.rb +16 -0
  303. data/lib/active_record/type.rb +83 -0
  304. data/lib/active_record/type_caster/connection.rb +33 -0
  305. data/lib/active_record/type_caster/map.rb +23 -0
  306. data/lib/active_record/type_caster.rb +9 -0
  307. data/lib/active_record/validations/absence.rb +25 -0
  308. data/lib/active_record/validations/associated.rb +65 -0
  309. data/lib/active_record/validations/length.rb +26 -0
  310. data/lib/active_record/validations/numericality.rb +36 -0
  311. data/lib/active_record/validations/presence.rb +45 -0
  312. data/lib/active_record/validations/uniqueness.rb +295 -0
  313. data/lib/active_record/validations.rb +101 -0
  314. data/lib/active_record/version.rb +10 -0
  315. data/lib/active_record.rb +616 -0
  316. data/lib/arel/alias_predication.rb +9 -0
  317. data/lib/arel/attributes/attribute.rb +33 -0
  318. data/lib/arel/collectors/bind.rb +31 -0
  319. data/lib/arel/collectors/composite.rb +46 -0
  320. data/lib/arel/collectors/plain_string.rb +20 -0
  321. data/lib/arel/collectors/sql_string.rb +27 -0
  322. data/lib/arel/collectors/substitute_binds.rb +35 -0
  323. data/lib/arel/crud.rb +48 -0
  324. data/lib/arel/delete_manager.rb +32 -0
  325. data/lib/arel/errors.rb +19 -0
  326. data/lib/arel/expressions.rb +29 -0
  327. data/lib/arel/factory_methods.rb +53 -0
  328. data/lib/arel/filter_predications.rb +9 -0
  329. data/lib/arel/insert_manager.rb +48 -0
  330. data/lib/arel/math.rb +45 -0
  331. data/lib/arel/nodes/ascending.rb +23 -0
  332. data/lib/arel/nodes/binary.rb +125 -0
  333. data/lib/arel/nodes/bind_param.rb +44 -0
  334. data/lib/arel/nodes/bound_sql_literal.rb +65 -0
  335. data/lib/arel/nodes/case.rb +55 -0
  336. data/lib/arel/nodes/casted.rb +62 -0
  337. data/lib/arel/nodes/comment.rb +29 -0
  338. data/lib/arel/nodes/count.rb +12 -0
  339. data/lib/arel/nodes/cte.rb +36 -0
  340. data/lib/arel/nodes/delete_statement.rb +44 -0
  341. data/lib/arel/nodes/descending.rb +23 -0
  342. data/lib/arel/nodes/equality.rb +15 -0
  343. data/lib/arel/nodes/extract.rb +24 -0
  344. data/lib/arel/nodes/false.rb +16 -0
  345. data/lib/arel/nodes/filter.rb +10 -0
  346. data/lib/arel/nodes/fragments.rb +35 -0
  347. data/lib/arel/nodes/full_outer_join.rb +8 -0
  348. data/lib/arel/nodes/function.rb +45 -0
  349. data/lib/arel/nodes/grouping.rb +11 -0
  350. data/lib/arel/nodes/homogeneous_in.rb +68 -0
  351. data/lib/arel/nodes/in.rb +15 -0
  352. data/lib/arel/nodes/infix_operation.rb +92 -0
  353. data/lib/arel/nodes/inner_join.rb +8 -0
  354. data/lib/arel/nodes/insert_statement.rb +37 -0
  355. data/lib/arel/nodes/join_source.rb +20 -0
  356. data/lib/arel/nodes/leading_join.rb +8 -0
  357. data/lib/arel/nodes/matches.rb +18 -0
  358. data/lib/arel/nodes/named_function.rb +23 -0
  359. data/lib/arel/nodes/nary.rb +39 -0
  360. data/lib/arel/nodes/node.rb +161 -0
  361. data/lib/arel/nodes/node_expression.rb +13 -0
  362. data/lib/arel/nodes/ordering.rb +27 -0
  363. data/lib/arel/nodes/outer_join.rb +8 -0
  364. data/lib/arel/nodes/over.rb +15 -0
  365. data/lib/arel/nodes/regexp.rb +16 -0
  366. data/lib/arel/nodes/right_outer_join.rb +8 -0
  367. data/lib/arel/nodes/select_core.rb +67 -0
  368. data/lib/arel/nodes/select_statement.rb +41 -0
  369. data/lib/arel/nodes/sql_literal.rb +32 -0
  370. data/lib/arel/nodes/string_join.rb +11 -0
  371. data/lib/arel/nodes/table_alias.rb +35 -0
  372. data/lib/arel/nodes/terminal.rb +16 -0
  373. data/lib/arel/nodes/true.rb +16 -0
  374. data/lib/arel/nodes/unary.rb +44 -0
  375. data/lib/arel/nodes/unary_operation.rb +20 -0
  376. data/lib/arel/nodes/unqualified_column.rb +22 -0
  377. data/lib/arel/nodes/update_statement.rb +46 -0
  378. data/lib/arel/nodes/values_list.rb +9 -0
  379. data/lib/arel/nodes/window.rb +126 -0
  380. data/lib/arel/nodes/with.rb +11 -0
  381. data/lib/arel/nodes.rb +75 -0
  382. data/lib/arel/order_predications.rb +13 -0
  383. data/lib/arel/predications.rb +260 -0
  384. data/lib/arel/select_manager.rb +276 -0
  385. data/lib/arel/table.rb +121 -0
  386. data/lib/arel/tree_manager.rb +65 -0
  387. data/lib/arel/update_manager.rb +49 -0
  388. data/lib/arel/visitors/dot.rb +299 -0
  389. data/lib/arel/visitors/mysql.rb +111 -0
  390. data/lib/arel/visitors/postgresql.rb +99 -0
  391. data/lib/arel/visitors/sqlite.rb +38 -0
  392. data/lib/arel/visitors/to_sql.rb +1033 -0
  393. data/lib/arel/visitors/visitor.rb +45 -0
  394. data/lib/arel/visitors.rb +13 -0
  395. data/lib/arel/window_predications.rb +9 -0
  396. data/lib/arel.rb +73 -0
  397. data/lib/rails/generators/active_record/application_record/USAGE +8 -0
  398. data/lib/rails/generators/active_record/application_record/application_record_generator.rb +26 -0
  399. data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +5 -0
  400. data/lib/rails/generators/active_record/migration/migration_generator.rb +76 -0
  401. data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +29 -0
  402. data/lib/rails/generators/active_record/migration/templates/migration.rb.tt +48 -0
  403. data/lib/rails/generators/active_record/migration.rb +54 -0
  404. data/lib/rails/generators/active_record/model/USAGE +113 -0
  405. data/lib/rails/generators/active_record/model/model_generator.rb +94 -0
  406. data/lib/rails/generators/active_record/model/templates/abstract_base_class.rb.tt +7 -0
  407. data/lib/rails/generators/active_record/model/templates/model.rb.tt +22 -0
  408. data/lib/rails/generators/active_record/model/templates/module.rb.tt +7 -0
  409. data/lib/rails/generators/active_record/multi_db/multi_db_generator.rb +16 -0
  410. data/lib/rails/generators/active_record/multi_db/templates/multi_db.rb.tt +44 -0
  411. data/lib/rails/generators/active_record.rb +19 -0
  412. metadata +505 -0
@@ -0,0 +1,661 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/string/filters"
4
+
5
+ module ActiveRecord
6
+ module FinderMethods
7
+ ONE_AS_ONE = "1 AS one"
8
+
9
+ # Find by id - This can either be a specific id (ID), a list of ids (ID, ID, ID), or an array of ids ([ID, ID, ID]).
10
+ # `ID` refers to an "identifier". For models with a single-column primary key, `ID` will be a single value,
11
+ # and for models with a composite primary key, it will be an array of values.
12
+ # If one or more records cannot be found for the requested ids, then ActiveRecord::RecordNotFound will be raised.
13
+ # If the primary key is an integer, find by id coerces its arguments by using +to_i+.
14
+ #
15
+ # Person.find(1) # returns the object for ID = 1
16
+ # Person.find("1") # returns the object for ID = 1
17
+ # Person.find("31-sarah") # returns the object for ID = 31
18
+ # Person.find(1, 2, 6) # returns an array for objects with IDs in (1, 2, 6)
19
+ # Person.find([7, 17]) # returns an array for objects with IDs in (7, 17), or with composite primary key [7, 17]
20
+ # Person.find([1]) # returns an array for the object with ID = 1
21
+ # Person.where("administrator = 1").order("created_on DESC").find(1)
22
+ #
23
+ # ==== Find a record for a composite primary key model
24
+ # TravelRoute.primary_key = [:origin, :destination]
25
+ #
26
+ # TravelRoute.find(["Ottawa", "London"])
27
+ # => #<TravelRoute origin: "Ottawa", destination: "London">
28
+ #
29
+ # TravelRoute.find([["Paris", "Montreal"]])
30
+ # => [#<TravelRoute origin: "Paris", destination: "Montreal">]
31
+ #
32
+ # TravelRoute.find(["New York", "Las Vegas"], ["New York", "Portland"])
33
+ # => [
34
+ # #<TravelRoute origin: "New York", destination: "Las Vegas">,
35
+ # #<TravelRoute origin: "New York", destination: "Portland">
36
+ # ]
37
+ #
38
+ # TravelRoute.find([["Berlin", "London"], ["Barcelona", "Lisbon"]])
39
+ # => [
40
+ # #<TravelRoute origin: "Berlin", destination: "London">,
41
+ # #<TravelRoute origin: "Barcelona", destination: "Lisbon">
42
+ # ]
43
+ #
44
+ # NOTE: The returned records are in the same order as the ids you provide.
45
+ # If you want the results to be sorted by database, you can use ActiveRecord::QueryMethods#where
46
+ # method and provide an explicit ActiveRecord::QueryMethods#order option.
47
+ # But ActiveRecord::QueryMethods#where method doesn't raise ActiveRecord::RecordNotFound.
48
+ #
49
+ # ==== Find with lock
50
+ #
51
+ # Example for find with a lock: Imagine two concurrent transactions:
52
+ # each will read <tt>person.visits == 2</tt>, add 1 to it, and save, resulting
53
+ # in two saves of <tt>person.visits = 3</tt>. By locking the row, the second
54
+ # transaction has to wait until the first is finished; we get the
55
+ # expected <tt>person.visits == 4</tt>.
56
+ #
57
+ # Person.transaction do
58
+ # person = Person.lock(true).find(1)
59
+ # person.visits += 1
60
+ # person.save!
61
+ # end
62
+ #
63
+ # ==== Variations of #find
64
+ #
65
+ # Person.where(name: 'Spartacus', rating: 4)
66
+ # # returns a chainable list (which can be empty).
67
+ #
68
+ # Person.find_by(name: 'Spartacus', rating: 4)
69
+ # # returns the first item or nil.
70
+ #
71
+ # Person.find_or_initialize_by(name: 'Spartacus', rating: 4)
72
+ # # returns the first item or returns a new instance (requires you call .save to persist against the database).
73
+ #
74
+ # Person.find_or_create_by(name: 'Spartacus', rating: 4)
75
+ # # returns the first item or creates it and returns it.
76
+ #
77
+ # ==== Alternatives for #find
78
+ #
79
+ # Person.where(name: 'Spartacus', rating: 4).exists?(conditions = :none)
80
+ # # returns a boolean indicating if any record with the given conditions exist.
81
+ #
82
+ # Person.where(name: 'Spartacus', rating: 4).select("field1, field2, field3")
83
+ # # returns a chainable list of instances with only the mentioned fields.
84
+ #
85
+ # Person.where(name: 'Spartacus', rating: 4).ids
86
+ # # returns an Array of ids.
87
+ #
88
+ # Person.where(name: 'Spartacus', rating: 4).pluck(:field1, :field2)
89
+ # # returns an Array of the required fields.
90
+ #
91
+ # ==== Edge Cases
92
+ #
93
+ # Person.find(37) # raises ActiveRecord::RecordNotFound exception if the record with the given ID does not exist.
94
+ # Person.find([37]) # raises ActiveRecord::RecordNotFound exception if the record with the given ID in the input array does not exist.
95
+ # Person.find(nil) # raises ActiveRecord::RecordNotFound exception if the argument is nil.
96
+ # Person.find([]) # returns an empty array if the argument is an empty array.
97
+ # Person.find # raises ActiveRecord::RecordNotFound exception if the argument is not provided.
98
+ def find(*args)
99
+ return super if block_given?
100
+ find_with_ids(*args)
101
+ end
102
+
103
+ # Finds the first record matching the specified conditions. There
104
+ # is no implied ordering so if order matters, you should specify it
105
+ # yourself.
106
+ #
107
+ # If no record is found, returns <tt>nil</tt>.
108
+ #
109
+ # Post.find_by name: 'Spartacus', rating: 4
110
+ # Post.find_by "published_at < ?", 2.weeks.ago
111
+ def find_by(arg, *args)
112
+ where(arg, *args).take
113
+ end
114
+
115
+ # Like #find_by, except that if no record is found, raises
116
+ # an ActiveRecord::RecordNotFound error.
117
+ def find_by!(arg, *args)
118
+ where(arg, *args).take!
119
+ end
120
+
121
+ # Gives a record (or N records if a parameter is supplied) without any implied
122
+ # order. The order will depend on the database implementation.
123
+ # If an order is supplied it will be respected.
124
+ #
125
+ # Person.take # returns an object fetched by SELECT * FROM people LIMIT 1
126
+ # Person.take(5) # returns 5 objects fetched by SELECT * FROM people LIMIT 5
127
+ # Person.where(["name LIKE '%?'", name]).take
128
+ def take(limit = nil)
129
+ limit ? find_take_with_limit(limit) : find_take
130
+ end
131
+
132
+ # Same as #take but raises ActiveRecord::RecordNotFound if no record
133
+ # is found. Note that #take! accepts no arguments.
134
+ def take!
135
+ take || raise_record_not_found_exception!
136
+ end
137
+
138
+ # Finds the sole matching record. Raises ActiveRecord::RecordNotFound if no
139
+ # record is found. Raises ActiveRecord::SoleRecordExceeded if more than one
140
+ # record is found.
141
+ #
142
+ # Product.where(["price = %?", price]).sole
143
+ def sole
144
+ found, undesired = first(2)
145
+
146
+ if found.nil?
147
+ raise_record_not_found_exception!
148
+ elsif undesired.nil?
149
+ found
150
+ else
151
+ raise ActiveRecord::SoleRecordExceeded.new(model)
152
+ end
153
+ end
154
+
155
+ # Finds the sole matching record. Raises ActiveRecord::RecordNotFound if no
156
+ # record is found. Raises ActiveRecord::SoleRecordExceeded if more than one
157
+ # record is found.
158
+ #
159
+ # Product.find_sole_by(["price = %?", price])
160
+ def find_sole_by(arg, *args)
161
+ where(arg, *args).sole
162
+ end
163
+
164
+ # Find the first record (or first N records if a parameter is supplied).
165
+ # If no order is defined it will order by primary key.
166
+ #
167
+ # Person.first # returns the first object fetched by SELECT * FROM people ORDER BY people.id LIMIT 1
168
+ # Person.where(["user_name = ?", user_name]).first
169
+ # Person.where(["user_name = :u", { u: user_name }]).first
170
+ # Person.order("created_on DESC").offset(5).first
171
+ # Person.first(3) # returns the first three objects fetched by SELECT * FROM people ORDER BY people.id LIMIT 3
172
+ #
173
+ def first(limit = nil)
174
+ if limit
175
+ find_nth_with_limit(0, limit)
176
+ else
177
+ find_nth 0
178
+ end
179
+ end
180
+
181
+ # Same as #first but raises ActiveRecord::RecordNotFound if no record
182
+ # is found. Note that #first! accepts no arguments.
183
+ def first!
184
+ first || raise_record_not_found_exception!
185
+ end
186
+
187
+ # Find the last record (or last N records if a parameter is supplied).
188
+ # If no order is defined it will order by primary key.
189
+ #
190
+ # Person.last # returns the last object fetched by SELECT * FROM people
191
+ # Person.where(["user_name = ?", user_name]).last
192
+ # Person.order("created_on DESC").offset(5).last
193
+ # Person.last(3) # returns the last three objects fetched by SELECT * FROM people.
194
+ #
195
+ # Take note that in that last case, the results are sorted in ascending order:
196
+ #
197
+ # [#<Person id:2>, #<Person id:3>, #<Person id:4>]
198
+ #
199
+ # and not:
200
+ #
201
+ # [#<Person id:4>, #<Person id:3>, #<Person id:2>]
202
+ def last(limit = nil)
203
+ return find_last(limit) if loaded? || has_limit_or_offset?
204
+
205
+ result = ordered_relation.limit(limit)
206
+ result = result.reverse_order!
207
+
208
+ limit ? result.reverse : result.first
209
+ end
210
+
211
+ # Same as #last but raises ActiveRecord::RecordNotFound if no record
212
+ # is found. Note that #last! accepts no arguments.
213
+ def last!
214
+ last || raise_record_not_found_exception!
215
+ end
216
+
217
+ # Find the second record.
218
+ # If no order is defined it will order by primary key.
219
+ #
220
+ # Person.second # returns the second object fetched by SELECT * FROM people
221
+ # Person.offset(3).second # returns the second object from OFFSET 3 (which is OFFSET 4)
222
+ # Person.where(["user_name = :u", { u: user_name }]).second
223
+ def second
224
+ find_nth 1
225
+ end
226
+
227
+ # Same as #second but raises ActiveRecord::RecordNotFound if no record
228
+ # is found.
229
+ def second!
230
+ second || raise_record_not_found_exception!
231
+ end
232
+
233
+ # Find the third record.
234
+ # If no order is defined it will order by primary key.
235
+ #
236
+ # Person.third # returns the third object fetched by SELECT * FROM people
237
+ # Person.offset(3).third # returns the third object from OFFSET 3 (which is OFFSET 5)
238
+ # Person.where(["user_name = :u", { u: user_name }]).third
239
+ def third
240
+ find_nth 2
241
+ end
242
+
243
+ # Same as #third but raises ActiveRecord::RecordNotFound if no record
244
+ # is found.
245
+ def third!
246
+ third || raise_record_not_found_exception!
247
+ end
248
+
249
+ # Find the fourth record.
250
+ # If no order is defined it will order by primary key.
251
+ #
252
+ # Person.fourth # returns the fourth object fetched by SELECT * FROM people
253
+ # Person.offset(3).fourth # returns the fourth object from OFFSET 3 (which is OFFSET 6)
254
+ # Person.where(["user_name = :u", { u: user_name }]).fourth
255
+ def fourth
256
+ find_nth 3
257
+ end
258
+
259
+ # Same as #fourth but raises ActiveRecord::RecordNotFound if no record
260
+ # is found.
261
+ def fourth!
262
+ fourth || raise_record_not_found_exception!
263
+ end
264
+
265
+ # Find the fifth record.
266
+ # If no order is defined it will order by primary key.
267
+ #
268
+ # Person.fifth # returns the fifth object fetched by SELECT * FROM people
269
+ # Person.offset(3).fifth # returns the fifth object from OFFSET 3 (which is OFFSET 7)
270
+ # Person.where(["user_name = :u", { u: user_name }]).fifth
271
+ def fifth
272
+ find_nth 4
273
+ end
274
+
275
+ # Same as #fifth but raises ActiveRecord::RecordNotFound if no record
276
+ # is found.
277
+ def fifth!
278
+ fifth || raise_record_not_found_exception!
279
+ end
280
+
281
+ # Find the forty-second record. Also known as accessing "the reddit".
282
+ # If no order is defined it will order by primary key.
283
+ #
284
+ # Person.forty_two # returns the forty-second object fetched by SELECT * FROM people
285
+ # Person.offset(3).forty_two # returns the forty-second object from OFFSET 3 (which is OFFSET 44)
286
+ # Person.where(["user_name = :u", { u: user_name }]).forty_two
287
+ def forty_two
288
+ find_nth 41
289
+ end
290
+
291
+ # Same as #forty_two but raises ActiveRecord::RecordNotFound if no record
292
+ # is found.
293
+ def forty_two!
294
+ forty_two || raise_record_not_found_exception!
295
+ end
296
+
297
+ # Find the third-to-last record.
298
+ # If no order is defined it will order by primary key.
299
+ #
300
+ # Person.third_to_last # returns the third-to-last object fetched by SELECT * FROM people
301
+ # Person.offset(3).third_to_last # returns the third-to-last object from OFFSET 3
302
+ # Person.where(["user_name = :u", { u: user_name }]).third_to_last
303
+ def third_to_last
304
+ find_nth_from_last 3
305
+ end
306
+
307
+ # Same as #third_to_last but raises ActiveRecord::RecordNotFound if no record
308
+ # is found.
309
+ def third_to_last!
310
+ third_to_last || raise_record_not_found_exception!
311
+ end
312
+
313
+ # Find the second-to-last record.
314
+ # If no order is defined it will order by primary key.
315
+ #
316
+ # Person.second_to_last # returns the second-to-last object fetched by SELECT * FROM people
317
+ # Person.offset(3).second_to_last # returns the second-to-last object from OFFSET 3
318
+ # Person.where(["user_name = :u", { u: user_name }]).second_to_last
319
+ def second_to_last
320
+ find_nth_from_last 2
321
+ end
322
+
323
+ # Same as #second_to_last but raises ActiveRecord::RecordNotFound if no record
324
+ # is found.
325
+ def second_to_last!
326
+ second_to_last || raise_record_not_found_exception!
327
+ end
328
+
329
+ # Returns true if a record exists in the table that matches the +id+ or
330
+ # conditions given, or false otherwise. The argument can take six forms:
331
+ #
332
+ # * Integer - Finds the record with this primary key.
333
+ # * String - Finds the record with a primary key corresponding to this
334
+ # string (such as <tt>'5'</tt>).
335
+ # * Array - Finds the record that matches these +where+-style conditions
336
+ # (such as <tt>['name LIKE ?', "%#{query}%"]</tt>).
337
+ # * Hash - Finds the record that matches these +where+-style conditions
338
+ # (such as <tt>{name: 'David'}</tt>).
339
+ # * +false+ - Returns always +false+.
340
+ # * No args - Returns +false+ if the relation is empty, +true+ otherwise.
341
+ #
342
+ # For more information about specifying conditions as a hash or array,
343
+ # see the Conditions section in the introduction to ActiveRecord::Base.
344
+ #
345
+ # Note: You can't pass in a condition as a string (like <tt>name =
346
+ # 'Jamie'</tt>), since it would be sanitized and then queried against
347
+ # the primary key column, like <tt>id = 'name = \'Jamie\''</tt>.
348
+ #
349
+ # Person.exists?(5)
350
+ # Person.exists?('5')
351
+ # Person.exists?(['name LIKE ?', "%#{query}%"])
352
+ # Person.exists?(id: [1, 4, 8])
353
+ # Person.exists?(name: 'David')
354
+ # Person.exists?(false)
355
+ # Person.exists?
356
+ # Person.where(name: 'Spartacus', rating: 4).exists?
357
+ def exists?(conditions = :none)
358
+ return false if @none
359
+
360
+ if Base === conditions
361
+ raise ArgumentError, <<-MSG.squish
362
+ You are passing an instance of ActiveRecord::Base to `exists?`.
363
+ Please pass the id of the object by calling `.id`.
364
+ MSG
365
+ end
366
+
367
+ return false if !conditions || limit_value == 0
368
+
369
+ if eager_loading?
370
+ relation = apply_join_dependency(eager_loading: false)
371
+ return relation.exists?(conditions)
372
+ end
373
+
374
+ relation = construct_relation_for_exists(conditions)
375
+ return false if relation.where_clause.contradiction?
376
+
377
+ skip_query_cache_if_necessary do
378
+ with_connection do |c|
379
+ c.select_rows(relation.arel, "#{model.name} Exists?").size == 1
380
+ end
381
+ end
382
+ end
383
+
384
+ # Returns true if the relation contains the given record or false otherwise.
385
+ #
386
+ # No query is performed if the relation is loaded; the given record is
387
+ # compared to the records in memory. If the relation is unloaded, an
388
+ # efficient existence query is performed, as in #exists?.
389
+ def include?(record)
390
+ # The existing implementation relies on receiving an Active Record instance as the input parameter named record.
391
+ # Any non-Active Record object passed to this implementation is guaranteed to return `false`.
392
+ return false unless record.is_a?(model)
393
+
394
+ if loaded? || offset_value || limit_value || having_clause.any?
395
+ records.include?(record)
396
+ else
397
+ id = if record.class.composite_primary_key?
398
+ record.class.primary_key.zip(record.id).to_h
399
+ else
400
+ record.id
401
+ end
402
+
403
+ exists?(id)
404
+ end
405
+ end
406
+
407
+ alias :member? :include?
408
+
409
+ # This method is called whenever no records are found with either a single
410
+ # id or multiple ids and raises an ActiveRecord::RecordNotFound exception.
411
+ #
412
+ # The error message is different depending on whether a single id or
413
+ # multiple ids are provided. If multiple ids are provided, then the number
414
+ # of results obtained should be provided in the +result_size+ argument and
415
+ # the expected number of results should be provided in the +expected_size+
416
+ # argument.
417
+ def raise_record_not_found_exception!(ids = nil, result_size = nil, expected_size = nil, key = primary_key, not_found_ids = nil) # :nodoc:
418
+ conditions = " [#{arel.where_sql(model)}]" unless where_clause.empty?
419
+
420
+ name = model.name
421
+
422
+ if ids.nil?
423
+ error = +"Couldn't find #{name}"
424
+ error << " with#{conditions}" if conditions
425
+ raise RecordNotFound.new(error, name, key)
426
+ elsif Array.wrap(ids).size == 1
427
+ error = "Couldn't find #{name} with '#{key}'=#{ids}#{conditions}"
428
+ raise RecordNotFound.new(error, name, key, ids)
429
+ else
430
+ error = +"Couldn't find all #{name.pluralize} with '#{key}': "
431
+ error << "(#{ids.join(", ")})#{conditions} (found #{result_size} results, but was looking for #{expected_size})."
432
+ error << " Couldn't find #{name.pluralize(not_found_ids.size)} with #{key.to_s.pluralize(not_found_ids.size)} #{not_found_ids.join(', ')}." if not_found_ids
433
+ raise RecordNotFound.new(error, name, key, ids)
434
+ end
435
+ end
436
+
437
+ private
438
+ def construct_relation_for_exists(conditions)
439
+ conditions = sanitize_forbidden_attributes(conditions)
440
+
441
+ if distinct_value && offset_value
442
+ relation = except(:order).limit!(1)
443
+ else
444
+ relation = except(:select, :distinct, :order)._select!(ONE_AS_ONE).limit!(1)
445
+ end
446
+
447
+ case conditions
448
+ when Array, Hash
449
+ relation.where!(conditions) unless conditions.empty?
450
+ else
451
+ relation.where!(primary_key => conditions) unless conditions == :none
452
+ end
453
+
454
+ relation
455
+ end
456
+
457
+ def apply_join_dependency(eager_loading: group_values.empty?)
458
+ join_dependency = construct_join_dependency(
459
+ eager_load_values | includes_values, Arel::Nodes::OuterJoin
460
+ )
461
+ relation = except(:includes, :eager_load, :preload).joins!(join_dependency)
462
+
463
+ if eager_loading && has_limit_or_offset? && !(
464
+ using_limitable_reflections?(join_dependency.reflections) &&
465
+ using_limitable_reflections?(
466
+ construct_join_dependency(
467
+ select_association_list(joins_values).concat(
468
+ select_association_list(left_outer_joins_values)
469
+ ), nil
470
+ ).reflections
471
+ )
472
+ )
473
+ relation = skip_query_cache_if_necessary do
474
+ model.with_connection do |c|
475
+ c.distinct_relation_for_primary_key(relation)
476
+ end
477
+ end
478
+ end
479
+
480
+ if block_given?
481
+ yield relation, join_dependency
482
+ else
483
+ relation
484
+ end
485
+ end
486
+
487
+ def using_limitable_reflections?(reflections)
488
+ reflections.none?(&:collection?)
489
+ end
490
+
491
+ def find_with_ids(*ids)
492
+ raise UnknownPrimaryKey.new(model) if primary_key.nil?
493
+
494
+ expects_array = if model.composite_primary_key?
495
+ ids.first.first.is_a?(Array)
496
+ else
497
+ ids.first.is_a?(Array)
498
+ end
499
+
500
+ return [] if expects_array && ids.first.empty?
501
+
502
+ ids = ids.first if expects_array
503
+
504
+ ids = ids.compact.uniq
505
+
506
+ model_name = model.name
507
+
508
+ case ids.size
509
+ when 0
510
+ error_message = "Couldn't find #{model_name} without an ID"
511
+ raise RecordNotFound.new(error_message, model_name, primary_key)
512
+ when 1
513
+ result = find_one(ids.first)
514
+ expects_array ? [ result ] : result
515
+ else
516
+ find_some(ids)
517
+ end
518
+ end
519
+
520
+ def find_one(id)
521
+ if ActiveRecord::Base === id
522
+ raise ArgumentError, <<-MSG.squish
523
+ You are passing an instance of ActiveRecord::Base to `find`.
524
+ Please pass the id of the object by calling `.id`.
525
+ MSG
526
+ end
527
+
528
+ relation = if model.composite_primary_key?
529
+ where(primary_key.zip(id).to_h)
530
+ else
531
+ where(primary_key => id)
532
+ end
533
+
534
+ record = relation.take
535
+
536
+ raise_record_not_found_exception!(id, 0, 1) unless record
537
+
538
+ record
539
+ end
540
+
541
+ def find_some(ids)
542
+ return find_some_ordered(ids) unless order_values.present?
543
+
544
+ relation = where(primary_key => ids)
545
+ relation = relation.select(table[primary_key]) unless select_values.empty?
546
+ result = relation.to_a
547
+
548
+ expected_size =
549
+ if limit_value && ids.size > limit_value
550
+ limit_value
551
+ else
552
+ ids.size
553
+ end
554
+
555
+ # 11 ids with limit 3, offset 9 should give 2 results.
556
+ if offset_value && (ids.size - offset_value < expected_size)
557
+ expected_size = ids.size - offset_value
558
+ end
559
+
560
+ if result.size == expected_size
561
+ result
562
+ else
563
+ raise_record_not_found_exception!(ids, result.size, expected_size)
564
+ end
565
+ end
566
+
567
+ def find_some_ordered(ids)
568
+ ids = ids.slice(offset_value || 0, limit_value || ids.size) || []
569
+
570
+ relation = except(:limit, :offset)
571
+ relation = relation.where(primary_key => ids)
572
+ relation = relation.select(table[primary_key]) unless select_values.empty?
573
+ result = relation.records
574
+
575
+ if result.size == ids.size
576
+ result.in_order_of(:id, ids.map { |id| model.type_for_attribute(primary_key).cast(id) })
577
+ else
578
+ raise_record_not_found_exception!(ids, result.size, ids.size)
579
+ end
580
+ end
581
+
582
+ def find_take
583
+ if loaded?
584
+ records.first
585
+ else
586
+ @take ||= limit(1).records.first
587
+ end
588
+ end
589
+
590
+ def find_take_with_limit(limit)
591
+ if loaded?
592
+ records.take(limit)
593
+ else
594
+ limit(limit).to_a
595
+ end
596
+ end
597
+
598
+ def find_nth(index)
599
+ @offsets ||= {}
600
+ @offsets[index] ||= find_nth_with_limit(index, 1).first
601
+ end
602
+
603
+ def find_nth_with_limit(index, limit)
604
+ if loaded?
605
+ records[index, limit] || []
606
+ else
607
+ relation = ordered_relation
608
+
609
+ if limit_value
610
+ limit = [limit_value - index, limit].min
611
+ end
612
+
613
+ if limit > 0
614
+ relation = relation.offset((offset_value || 0) + index) unless index.zero?
615
+ relation.limit(limit).to_a
616
+ else
617
+ []
618
+ end
619
+ end
620
+ end
621
+
622
+ def find_nth_from_last(index)
623
+ if loaded?
624
+ records[-index]
625
+ else
626
+ relation = ordered_relation
627
+
628
+ if relation.order_values.empty? || relation.has_limit_or_offset?
629
+ relation.records[-index]
630
+ else
631
+ relation.reverse_order.offset(index - 1).first
632
+ end
633
+ end
634
+ end
635
+
636
+ def find_last(limit)
637
+ limit ? records.last(limit) : records.last
638
+ end
639
+
640
+ def ordered_relation
641
+ if order_values.empty? && (model.implicit_order_column || !model.query_constraints_list.nil? || primary_key)
642
+ order(_order_columns.map { |column| table[column].asc })
643
+ else
644
+ self
645
+ end
646
+ end
647
+
648
+ def _order_columns
649
+ oc = []
650
+
651
+ oc << model.implicit_order_column if model.implicit_order_column
652
+ oc << model.query_constraints_list if model.query_constraints_list
653
+
654
+ if model.primary_key && model.query_constraints_list.nil?
655
+ oc << model.primary_key
656
+ end
657
+
658
+ oc.flatten.uniq.compact
659
+ end
660
+ end
661
+ end
@@ -0,0 +1,30 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ class Relation
5
+ class FromClause # :nodoc:
6
+ attr_reader :value, :name
7
+
8
+ def initialize(value, name)
9
+ @value = value
10
+ @name = name
11
+ end
12
+
13
+ def merge(other)
14
+ self
15
+ end
16
+
17
+ def empty?
18
+ value.nil?
19
+ end
20
+
21
+ def ==(other)
22
+ self.class == other.class && value == other.value && name == other.name
23
+ end
24
+
25
+ def self.empty
26
+ @empty ||= new(nil, nil).freeze
27
+ end
28
+ end
29
+ end
30
+ end