activerecord 5.0.7.2 → 6.1.1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of activerecord might be problematic. Click here for more details.

Files changed (363) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +829 -2015
  3. data/MIT-LICENSE +3 -1
  4. data/README.rdoc +11 -9
  5. data/examples/performance.rb +31 -29
  6. data/examples/simple.rb +5 -3
  7. data/lib/active_record.rb +37 -29
  8. data/lib/active_record/aggregations.rb +249 -247
  9. data/lib/active_record/association_relation.rb +30 -18
  10. data/lib/active_record/associations.rb +1714 -1596
  11. data/lib/active_record/associations/alias_tracker.rb +36 -42
  12. data/lib/active_record/associations/association.rb +143 -68
  13. data/lib/active_record/associations/association_scope.rb +98 -94
  14. data/lib/active_record/associations/belongs_to_association.rb +76 -46
  15. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +13 -12
  16. data/lib/active_record/associations/builder/association.rb +27 -28
  17. data/lib/active_record/associations/builder/belongs_to.rb +52 -60
  18. data/lib/active_record/associations/builder/collection_association.rb +12 -22
  19. data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +40 -62
  20. data/lib/active_record/associations/builder/has_many.rb +10 -2
  21. data/lib/active_record/associations/builder/has_one.rb +35 -2
  22. data/lib/active_record/associations/builder/singular_association.rb +5 -1
  23. data/lib/active_record/associations/collection_association.rb +104 -259
  24. data/lib/active_record/associations/collection_proxy.rb +169 -125
  25. data/lib/active_record/associations/foreign_association.rb +22 -0
  26. data/lib/active_record/associations/has_many_association.rb +46 -31
  27. data/lib/active_record/associations/has_many_through_association.rb +66 -46
  28. data/lib/active_record/associations/has_one_association.rb +71 -52
  29. data/lib/active_record/associations/has_one_through_association.rb +20 -11
  30. data/lib/active_record/associations/join_dependency.rb +169 -180
  31. data/lib/active_record/associations/join_dependency/join_association.rb +53 -79
  32. data/lib/active_record/associations/join_dependency/join_base.rb +10 -9
  33. data/lib/active_record/associations/join_dependency/join_part.rb +14 -14
  34. data/lib/active_record/associations/preloader.rb +97 -104
  35. data/lib/active_record/associations/preloader/association.rb +109 -97
  36. data/lib/active_record/associations/preloader/through_association.rb +77 -76
  37. data/lib/active_record/associations/singular_association.rb +12 -45
  38. data/lib/active_record/associations/through_association.rb +27 -15
  39. data/lib/active_record/attribute_assignment.rb +55 -60
  40. data/lib/active_record/attribute_methods.rb +111 -141
  41. data/lib/active_record/attribute_methods/before_type_cast.rb +17 -9
  42. data/lib/active_record/attribute_methods/dirty.rb +172 -112
  43. data/lib/active_record/attribute_methods/primary_key.rb +88 -91
  44. data/lib/active_record/attribute_methods/query.rb +6 -8
  45. data/lib/active_record/attribute_methods/read.rb +18 -50
  46. data/lib/active_record/attribute_methods/serialization.rb +38 -10
  47. data/lib/active_record/attribute_methods/time_zone_conversion.rb +38 -66
  48. data/lib/active_record/attribute_methods/write.rb +25 -32
  49. data/lib/active_record/attributes.rb +69 -31
  50. data/lib/active_record/autosave_association.rb +102 -66
  51. data/lib/active_record/base.rb +16 -25
  52. data/lib/active_record/callbacks.rb +202 -43
  53. data/lib/active_record/coders/json.rb +2 -0
  54. data/lib/active_record/coders/yaml_column.rb +11 -12
  55. data/lib/active_record/connection_adapters.rb +50 -0
  56. data/lib/active_record/connection_adapters/abstract/connection_pool.rb +661 -375
  57. data/lib/active_record/connection_adapters/abstract/database_limits.rb +14 -38
  58. data/lib/active_record/connection_adapters/abstract/database_statements.rb +269 -105
  59. data/lib/active_record/connection_adapters/abstract/query_cache.rb +54 -35
  60. data/lib/active_record/connection_adapters/abstract/quoting.rb +137 -93
  61. data/lib/active_record/connection_adapters/abstract/savepoints.rb +5 -3
  62. data/lib/active_record/connection_adapters/abstract/schema_creation.rb +155 -113
  63. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +328 -162
  64. data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +68 -80
  65. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +591 -259
  66. data/lib/active_record/connection_adapters/abstract/transaction.rb +229 -91
  67. data/lib/active_record/connection_adapters/abstract_adapter.rb +392 -244
  68. data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +457 -582
  69. data/lib/active_record/connection_adapters/column.rb +55 -13
  70. data/lib/active_record/connection_adapters/deduplicable.rb +29 -0
  71. data/lib/active_record/connection_adapters/legacy_pool_manager.rb +31 -0
  72. data/lib/active_record/connection_adapters/mysql/column.rb +8 -31
  73. data/lib/active_record/connection_adapters/mysql/database_statements.rb +135 -49
  74. data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +24 -23
  75. data/lib/active_record/connection_adapters/mysql/quoting.rb +50 -20
  76. data/lib/active_record/connection_adapters/mysql/schema_creation.rb +79 -49
  77. data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +66 -56
  78. data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +70 -36
  79. data/lib/active_record/connection_adapters/mysql/schema_statements.rb +268 -0
  80. data/lib/active_record/connection_adapters/mysql/type_metadata.rb +20 -12
  81. data/lib/active_record/connection_adapters/mysql2_adapter.rb +74 -37
  82. data/lib/active_record/connection_adapters/pool_config.rb +63 -0
  83. data/lib/active_record/connection_adapters/pool_manager.rb +43 -0
  84. data/lib/active_record/connection_adapters/postgresql/column.rb +39 -28
  85. data/lib/active_record/connection_adapters/postgresql/database_statements.rb +70 -101
  86. data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +5 -3
  87. data/lib/active_record/connection_adapters/postgresql/oid.rb +26 -21
  88. data/lib/active_record/connection_adapters/postgresql/oid/array.rb +22 -11
  89. data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +6 -5
  90. data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +2 -0
  91. data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +2 -0
  92. data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +6 -6
  93. data/lib/active_record/connection_adapters/postgresql/oid/date.rb +23 -0
  94. data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +14 -4
  95. data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +3 -1
  96. data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +5 -4
  97. data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +19 -18
  98. data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +2 -0
  99. data/lib/active_record/connection_adapters/postgresql/oid/interval.rb +49 -0
  100. data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +3 -11
  101. data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +44 -0
  102. data/lib/active_record/connection_adapters/postgresql/oid/macaddr.rb +25 -0
  103. data/lib/active_record/connection_adapters/postgresql/oid/money.rb +7 -5
  104. data/lib/active_record/connection_adapters/postgresql/oid/{json.rb → oid.rb} +6 -1
  105. data/lib/active_record/connection_adapters/postgresql/oid/point.rb +30 -9
  106. data/lib/active_record/connection_adapters/postgresql/oid/range.rb +52 -30
  107. data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +4 -1
  108. data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +58 -54
  109. data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +18 -4
  110. data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +2 -0
  111. data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +2 -0
  112. data/lib/active_record/connection_adapters/postgresql/quoting.rb +98 -38
  113. data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +21 -27
  114. data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +80 -0
  115. data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +147 -105
  116. data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +34 -32
  117. data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +426 -324
  118. data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +32 -23
  119. data/lib/active_record/connection_adapters/postgresql/utils.rb +9 -6
  120. data/lib/active_record/connection_adapters/postgresql_adapter.rb +418 -293
  121. data/lib/active_record/connection_adapters/schema_cache.rb +135 -18
  122. data/lib/active_record/connection_adapters/sql_type_metadata.rb +22 -7
  123. data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +144 -0
  124. data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +3 -1
  125. data/lib/active_record/connection_adapters/sqlite3/quoting.rb +72 -18
  126. data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +5 -6
  127. data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +19 -0
  128. data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +18 -0
  129. data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +170 -0
  130. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +282 -290
  131. data/lib/active_record/connection_adapters/statement_pool.rb +9 -8
  132. data/lib/active_record/connection_handling.rb +287 -45
  133. data/lib/active_record/core.rb +385 -181
  134. data/lib/active_record/counter_cache.rb +60 -28
  135. data/lib/active_record/database_configurations.rb +272 -0
  136. data/lib/active_record/database_configurations/connection_url_resolver.rb +98 -0
  137. data/lib/active_record/database_configurations/database_config.rb +80 -0
  138. data/lib/active_record/database_configurations/hash_config.rb +96 -0
  139. data/lib/active_record/database_configurations/url_config.rb +53 -0
  140. data/lib/active_record/delegated_type.rb +209 -0
  141. data/lib/active_record/destroy_association_async_job.rb +36 -0
  142. data/lib/active_record/dynamic_matchers.rb +87 -87
  143. data/lib/active_record/enum.rb +122 -47
  144. data/lib/active_record/errors.rb +153 -22
  145. data/lib/active_record/explain.rb +13 -8
  146. data/lib/active_record/explain_registry.rb +3 -1
  147. data/lib/active_record/explain_subscriber.rb +9 -4
  148. data/lib/active_record/fixture_set/file.rb +20 -22
  149. data/lib/active_record/fixture_set/model_metadata.rb +32 -0
  150. data/lib/active_record/fixture_set/render_context.rb +17 -0
  151. data/lib/active_record/fixture_set/table_row.rb +152 -0
  152. data/lib/active_record/fixture_set/table_rows.rb +46 -0
  153. data/lib/active_record/fixtures.rb +246 -507
  154. data/lib/active_record/gem_version.rb +6 -4
  155. data/lib/active_record/inheritance.rb +168 -95
  156. data/lib/active_record/insert_all.rb +208 -0
  157. data/lib/active_record/integration.rb +114 -25
  158. data/lib/active_record/internal_metadata.rb +30 -24
  159. data/lib/active_record/legacy_yaml_adapter.rb +11 -5
  160. data/lib/active_record/locking/optimistic.rb +81 -85
  161. data/lib/active_record/locking/pessimistic.rb +22 -6
  162. data/lib/active_record/log_subscriber.rb +68 -31
  163. data/lib/active_record/middleware/database_selector.rb +77 -0
  164. data/lib/active_record/middleware/database_selector/resolver.rb +92 -0
  165. data/lib/active_record/middleware/database_selector/resolver/session.rb +48 -0
  166. data/lib/active_record/migration.rb +439 -342
  167. data/lib/active_record/migration/command_recorder.rb +152 -98
  168. data/lib/active_record/migration/compatibility.rb +229 -60
  169. data/lib/active_record/migration/join_table.rb +8 -7
  170. data/lib/active_record/model_schema.rb +230 -122
  171. data/lib/active_record/nested_attributes.rb +213 -203
  172. data/lib/active_record/no_touching.rb +11 -2
  173. data/lib/active_record/null_relation.rb +12 -34
  174. data/lib/active_record/persistence.rb +471 -97
  175. data/lib/active_record/query_cache.rb +23 -12
  176. data/lib/active_record/querying.rb +43 -25
  177. data/lib/active_record/railtie.rb +155 -43
  178. data/lib/active_record/railties/console_sandbox.rb +2 -0
  179. data/lib/active_record/railties/controller_runtime.rb +34 -33
  180. data/lib/active_record/railties/databases.rake +507 -195
  181. data/lib/active_record/readonly_attributes.rb +9 -4
  182. data/lib/active_record/reflection.rb +245 -269
  183. data/lib/active_record/relation.rb +475 -324
  184. data/lib/active_record/relation/batches.rb +125 -72
  185. data/lib/active_record/relation/batches/batch_enumerator.rb +28 -10
  186. data/lib/active_record/relation/calculations.rb +267 -171
  187. data/lib/active_record/relation/delegation.rb +73 -69
  188. data/lib/active_record/relation/finder_methods.rb +238 -248
  189. data/lib/active_record/relation/from_clause.rb +7 -9
  190. data/lib/active_record/relation/merger.rb +95 -77
  191. data/lib/active_record/relation/predicate_builder.rb +109 -110
  192. data/lib/active_record/relation/predicate_builder/array_handler.rb +22 -17
  193. data/lib/active_record/relation/predicate_builder/association_query_value.rb +42 -0
  194. data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +6 -4
  195. data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +55 -0
  196. data/lib/active_record/relation/predicate_builder/range_handler.rb +7 -18
  197. data/lib/active_record/relation/predicate_builder/relation_handler.rb +7 -1
  198. data/lib/active_record/relation/query_attribute.rb +33 -2
  199. data/lib/active_record/relation/query_methods.rb +654 -374
  200. data/lib/active_record/relation/record_fetch_warning.rb +8 -6
  201. data/lib/active_record/relation/spawn_methods.rb +15 -14
  202. data/lib/active_record/relation/where_clause.rb +171 -109
  203. data/lib/active_record/result.rb +88 -51
  204. data/lib/active_record/runtime_registry.rb +5 -3
  205. data/lib/active_record/sanitization.rb +73 -100
  206. data/lib/active_record/schema.rb +7 -14
  207. data/lib/active_record/schema_dumper.rb +101 -69
  208. data/lib/active_record/schema_migration.rb +16 -12
  209. data/lib/active_record/scoping.rb +20 -20
  210. data/lib/active_record/scoping/default.rb +92 -95
  211. data/lib/active_record/scoping/named.rb +39 -30
  212. data/lib/active_record/secure_token.rb +19 -9
  213. data/lib/active_record/serialization.rb +7 -3
  214. data/lib/active_record/signed_id.rb +116 -0
  215. data/lib/active_record/statement_cache.rb +80 -29
  216. data/lib/active_record/store.rb +122 -42
  217. data/lib/active_record/suppressor.rb +6 -3
  218. data/lib/active_record/table_metadata.rb +51 -39
  219. data/lib/active_record/tasks/database_tasks.rb +332 -115
  220. data/lib/active_record/tasks/mysql_database_tasks.rb +66 -104
  221. data/lib/active_record/tasks/postgresql_database_tasks.rb +84 -56
  222. data/lib/active_record/tasks/sqlite_database_tasks.rb +40 -19
  223. data/lib/active_record/test_databases.rb +24 -0
  224. data/lib/active_record/test_fixtures.rb +246 -0
  225. data/lib/active_record/timestamp.rb +70 -38
  226. data/lib/active_record/touch_later.rb +26 -24
  227. data/lib/active_record/transactions.rb +121 -184
  228. data/lib/active_record/translation.rb +3 -1
  229. data/lib/active_record/type.rb +29 -17
  230. data/lib/active_record/type/adapter_specific_registry.rb +44 -48
  231. data/lib/active_record/type/date.rb +2 -0
  232. data/lib/active_record/type/date_time.rb +2 -0
  233. data/lib/active_record/type/decimal_without_scale.rb +15 -0
  234. data/lib/active_record/type/hash_lookup_type_map.rb +5 -4
  235. data/lib/active_record/type/internal/timezone.rb +2 -0
  236. data/lib/active_record/type/json.rb +30 -0
  237. data/lib/active_record/type/serialized.rb +20 -9
  238. data/lib/active_record/type/text.rb +11 -0
  239. data/lib/active_record/type/time.rb +12 -1
  240. data/lib/active_record/type/type_map.rb +14 -17
  241. data/lib/active_record/type/unsigned_integer.rb +16 -0
  242. data/lib/active_record/type_caster.rb +4 -2
  243. data/lib/active_record/type_caster/connection.rb +17 -13
  244. data/lib/active_record/type_caster/map.rb +10 -6
  245. data/lib/active_record/validations.rb +8 -5
  246. data/lib/active_record/validations/absence.rb +2 -0
  247. data/lib/active_record/validations/associated.rb +4 -3
  248. data/lib/active_record/validations/length.rb +2 -0
  249. data/lib/active_record/validations/numericality.rb +35 -0
  250. data/lib/active_record/validations/presence.rb +4 -2
  251. data/lib/active_record/validations/uniqueness.rb +52 -45
  252. data/lib/active_record/version.rb +3 -1
  253. data/lib/arel.rb +54 -0
  254. data/lib/arel/alias_predication.rb +9 -0
  255. data/lib/arel/attributes/attribute.rb +41 -0
  256. data/lib/arel/collectors/bind.rb +29 -0
  257. data/lib/arel/collectors/composite.rb +39 -0
  258. data/lib/arel/collectors/plain_string.rb +20 -0
  259. data/lib/arel/collectors/sql_string.rb +27 -0
  260. data/lib/arel/collectors/substitute_binds.rb +35 -0
  261. data/lib/arel/crud.rb +42 -0
  262. data/lib/arel/delete_manager.rb +18 -0
  263. data/lib/arel/errors.rb +9 -0
  264. data/lib/arel/expressions.rb +29 -0
  265. data/lib/arel/factory_methods.rb +49 -0
  266. data/lib/arel/insert_manager.rb +49 -0
  267. data/lib/arel/math.rb +45 -0
  268. data/lib/arel/nodes.rb +70 -0
  269. data/lib/arel/nodes/and.rb +32 -0
  270. data/lib/arel/nodes/ascending.rb +23 -0
  271. data/lib/arel/nodes/binary.rb +126 -0
  272. data/lib/arel/nodes/bind_param.rb +44 -0
  273. data/lib/arel/nodes/case.rb +55 -0
  274. data/lib/arel/nodes/casted.rb +62 -0
  275. data/lib/arel/nodes/comment.rb +29 -0
  276. data/lib/arel/nodes/count.rb +12 -0
  277. data/lib/arel/nodes/delete_statement.rb +45 -0
  278. data/lib/arel/nodes/descending.rb +23 -0
  279. data/lib/arel/nodes/equality.rb +15 -0
  280. data/lib/arel/nodes/extract.rb +24 -0
  281. data/lib/arel/nodes/false.rb +16 -0
  282. data/lib/arel/nodes/full_outer_join.rb +8 -0
  283. data/lib/arel/nodes/function.rb +44 -0
  284. data/lib/arel/nodes/grouping.rb +11 -0
  285. data/lib/arel/nodes/homogeneous_in.rb +72 -0
  286. data/lib/arel/nodes/in.rb +15 -0
  287. data/lib/arel/nodes/infix_operation.rb +92 -0
  288. data/lib/arel/nodes/inner_join.rb +8 -0
  289. data/lib/arel/nodes/insert_statement.rb +37 -0
  290. data/lib/arel/nodes/join_source.rb +20 -0
  291. data/lib/arel/nodes/matches.rb +18 -0
  292. data/lib/arel/nodes/named_function.rb +23 -0
  293. data/lib/arel/nodes/node.rb +51 -0
  294. data/lib/arel/nodes/node_expression.rb +13 -0
  295. data/lib/arel/nodes/ordering.rb +27 -0
  296. data/lib/arel/nodes/outer_join.rb +8 -0
  297. data/lib/arel/nodes/over.rb +15 -0
  298. data/lib/arel/nodes/regexp.rb +16 -0
  299. data/lib/arel/nodes/right_outer_join.rb +8 -0
  300. data/lib/arel/nodes/select_core.rb +67 -0
  301. data/lib/arel/nodes/select_statement.rb +41 -0
  302. data/lib/arel/nodes/sql_literal.rb +19 -0
  303. data/lib/arel/nodes/string_join.rb +11 -0
  304. data/lib/arel/nodes/table_alias.rb +31 -0
  305. data/lib/arel/nodes/terminal.rb +16 -0
  306. data/lib/arel/nodes/true.rb +16 -0
  307. data/lib/arel/nodes/unary.rb +44 -0
  308. data/lib/arel/nodes/unary_operation.rb +20 -0
  309. data/lib/arel/nodes/unqualified_column.rb +22 -0
  310. data/lib/arel/nodes/update_statement.rb +41 -0
  311. data/lib/arel/nodes/values_list.rb +9 -0
  312. data/lib/arel/nodes/window.rb +126 -0
  313. data/lib/arel/nodes/with.rb +11 -0
  314. data/lib/arel/order_predications.rb +13 -0
  315. data/lib/arel/predications.rb +250 -0
  316. data/lib/arel/select_manager.rb +270 -0
  317. data/lib/arel/table.rb +118 -0
  318. data/lib/arel/tree_manager.rb +72 -0
  319. data/lib/arel/update_manager.rb +34 -0
  320. data/lib/arel/visitors.rb +13 -0
  321. data/lib/arel/visitors/dot.rb +308 -0
  322. data/lib/arel/visitors/mysql.rb +93 -0
  323. data/lib/arel/visitors/postgresql.rb +120 -0
  324. data/lib/arel/visitors/sqlite.rb +38 -0
  325. data/lib/arel/visitors/to_sql.rb +899 -0
  326. data/lib/arel/visitors/visitor.rb +45 -0
  327. data/lib/arel/window_predications.rb +9 -0
  328. data/lib/rails/generators/active_record.rb +7 -5
  329. data/lib/rails/generators/active_record/application_record/application_record_generator.rb +26 -0
  330. data/lib/rails/generators/active_record/{model/templates/application_record.rb → application_record/templates/application_record.rb.tt} +0 -0
  331. data/lib/rails/generators/active_record/migration.rb +22 -3
  332. data/lib/rails/generators/active_record/migration/migration_generator.rb +38 -35
  333. data/lib/rails/generators/active_record/migration/templates/{create_table_migration.rb → create_table_migration.rb.tt} +3 -1
  334. data/lib/rails/generators/active_record/migration/templates/{migration.rb → migration.rb.tt} +7 -5
  335. data/lib/rails/generators/active_record/model/model_generator.rb +41 -25
  336. data/lib/rails/generators/active_record/model/templates/abstract_base_class.rb.tt +7 -0
  337. data/lib/rails/generators/active_record/model/templates/{model.rb → model.rb.tt} +10 -1
  338. data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +0 -0
  339. metadata +141 -57
  340. data/lib/active_record/associations/preloader/belongs_to.rb +0 -17
  341. data/lib/active_record/associations/preloader/collection_association.rb +0 -17
  342. data/lib/active_record/associations/preloader/has_many.rb +0 -17
  343. data/lib/active_record/associations/preloader/has_many_through.rb +0 -19
  344. data/lib/active_record/associations/preloader/has_one.rb +0 -15
  345. data/lib/active_record/associations/preloader/has_one_through.rb +0 -9
  346. data/lib/active_record/associations/preloader/singular_association.rb +0 -20
  347. data/lib/active_record/attribute.rb +0 -213
  348. data/lib/active_record/attribute/user_provided_default.rb +0 -28
  349. data/lib/active_record/attribute_decorators.rb +0 -67
  350. data/lib/active_record/attribute_mutation_tracker.rb +0 -70
  351. data/lib/active_record/attribute_set.rb +0 -110
  352. data/lib/active_record/attribute_set/builder.rb +0 -132
  353. data/lib/active_record/collection_cache_key.rb +0 -50
  354. data/lib/active_record/connection_adapters/connection_specification.rb +0 -263
  355. data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +0 -22
  356. data/lib/active_record/connection_adapters/postgresql/oid/rails_5_1_point.rb +0 -50
  357. data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
  358. data/lib/active_record/relation/predicate_builder/association_query_handler.rb +0 -88
  359. data/lib/active_record/relation/predicate_builder/base_handler.rb +0 -17
  360. data/lib/active_record/relation/predicate_builder/class_handler.rb +0 -27
  361. data/lib/active_record/relation/predicate_builder/polymorphic_array_handler.rb +0 -57
  362. data/lib/active_record/relation/where_clause_factory.rb +0 -38
  363. data/lib/active_record/type/internal/abstract_json.rb +0 -33
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActiveRecord
2
4
  # = Active Record Counter Cache
3
5
  module CounterCache
@@ -12,13 +14,21 @@ module ActiveRecord
12
14
  #
13
15
  # * +id+ - The id of the object you wish to reset a counter on.
14
16
  # * +counters+ - One or more association counters to reset. Association name or counter name can be given.
17
+ # * <tt>:touch</tt> - Touch timestamp columns when updating.
18
+ # Pass +true+ to touch +updated_at+ and/or +updated_on+. Pass a symbol to
19
+ # touch that column or an array of symbols to touch just those ones.
15
20
  #
16
21
  # ==== Examples
17
22
  #
18
- # # For Post with id #1 records reset the comments_count
23
+ # # For the Post with id #1, reset the comments_count
19
24
  # Post.reset_counters(1, :comments)
20
- def reset_counters(id, *counters)
25
+ #
26
+ # # Like above, but also touch the +updated_at+ and/or +updated_on+
27
+ # # attributes.
28
+ # Post.reset_counters(1, :comments, touch: true)
29
+ def reset_counters(id, *counters, touch: nil)
21
30
  object = find(id)
31
+
22
32
  counters.each do |counter_association|
23
33
  has_many_association = _reflect_on_association(counter_association)
24
34
  unless has_many_association
@@ -26,7 +36,7 @@ module ActiveRecord
26
36
  has_many_association = has_many.find { |association| association.counter_cache_column && association.counter_cache_column.to_sym == counter_association.to_sym }
27
37
  counter_association = has_many_association.plural_name if has_many_association
28
38
  end
29
- raise ArgumentError, "'#{self.name}' has no association called '#{counter_association}'" unless has_many_association
39
+ raise ArgumentError, "'#{name}' has no association called '#{counter_association}'" unless has_many_association
30
40
 
31
41
  if has_many_association.is_a? ActiveRecord::Reflection::ThroughReflection
32
42
  has_many_association = has_many_association.through_reflection
@@ -37,11 +47,20 @@ module ActiveRecord
37
47
  reflection = child_class._reflections.values.find { |e| e.belongs_to? && e.foreign_key.to_s == foreign_key && e.options[:counter_cache].present? }
38
48
  counter_name = reflection.counter_cache_column
39
49
 
40
- unscoped.where(primary_key => object.id).update_all(
41
- counter_name => object.send(counter_association).count(:all)
42
- )
50
+ updates = { counter_name => object.send(counter_association).count(:all) }
51
+
52
+ if touch
53
+ names = touch if touch != true
54
+ names = Array.wrap(names)
55
+ options = names.extract_options!
56
+ touch_updates = touch_attributes_with_time(*names, **options)
57
+ updates.merge!(touch_updates)
58
+ end
59
+
60
+ unscoped.where(primary_key => object.id).update_all(updates)
43
61
  end
44
- return true
62
+
63
+ true
45
64
  end
46
65
 
47
66
  # A generic "counter updater" implementation, intended primarily to be
@@ -55,6 +74,9 @@ module ActiveRecord
55
74
  # * +id+ - The id of the object you wish to update a counter on or an array of ids.
56
75
  # * +counters+ - A Hash containing the names of the fields
57
76
  # to update as keys and the amount to update the field by as values.
77
+ # * <tt>:touch</tt> option - Touch timestamp columns when updating.
78
+ # If attribute names are passed, they are updated along with updated_at/on
79
+ # attributes.
58
80
  #
59
81
  # ==== Examples
60
82
  #
@@ -73,14 +95,17 @@ module ActiveRecord
73
95
  # # UPDATE posts
74
96
  # # SET comment_count = COALESCE(comment_count, 0) + 1
75
97
  # # WHERE id IN (10, 15)
98
+ #
99
+ # # For the Posts with id of 10 and 15, increment the comment_count by 1
100
+ # # and update the updated_at value for each counter.
101
+ # Post.update_counters [10, 15], comment_count: 1, touch: true
102
+ # # Executes the following SQL:
103
+ # # UPDATE posts
104
+ # # SET comment_count = COALESCE(comment_count, 0) + 1,
105
+ # # `updated_at` = '2016-10-13T09:59:23-05:00'
106
+ # # WHERE id IN (10, 15)
76
107
  def update_counters(id, counters)
77
- updates = counters.map do |counter_name, value|
78
- operator = value < 0 ? '-' : '+'
79
- quoted_column = connection.quote_column_name(counter_name)
80
- "#{quoted_column} = COALESCE(#{quoted_column}, 0) #{operator} #{value.abs}"
81
- end
82
-
83
- unscoped.where(primary_key => id).update_all updates.join(', ')
108
+ unscoped.where!(primary_key => id).update_counters(counters)
84
109
  end
85
110
 
86
111
  # Increment a numeric field by one, via a direct SQL update.
@@ -94,13 +119,20 @@ module ActiveRecord
94
119
  #
95
120
  # * +counter_name+ - The name of the field that should be incremented.
96
121
  # * +id+ - The id of the object that should be incremented or an array of ids.
122
+ # * <tt>:touch</tt> - Touch timestamp columns when updating.
123
+ # Pass +true+ to touch +updated_at+ and/or +updated_on+. Pass a symbol to
124
+ # touch that column or an array of symbols to touch just those ones.
97
125
  #
98
126
  # ==== Examples
99
127
  #
100
128
  # # Increment the posts_count column for the record with an id of 5
101
129
  # DiscussionBoard.increment_counter(:posts_count, 5)
102
- def increment_counter(counter_name, id)
103
- update_counters(id, counter_name => 1)
130
+ #
131
+ # # Increment the posts_count column for the record with an id of 5
132
+ # # and update the updated_at value.
133
+ # DiscussionBoard.increment_counter(:posts_count, 5, touch: true)
134
+ def increment_counter(counter_name, id, touch: nil)
135
+ update_counters(id, counter_name => 1, touch: touch)
104
136
  end
105
137
 
106
138
  # Decrement a numeric field by one, via a direct SQL update.
@@ -112,26 +144,29 @@ module ActiveRecord
112
144
  #
113
145
  # * +counter_name+ - The name of the field that should be decremented.
114
146
  # * +id+ - The id of the object that should be decremented or an array of ids.
147
+ # * <tt>:touch</tt> - Touch timestamp columns when updating.
148
+ # Pass +true+ to touch +updated_at+ and/or +updated_on+. Pass a symbol to
149
+ # touch that column or an array of symbols to touch just those ones.
115
150
  #
116
151
  # ==== Examples
117
152
  #
118
153
  # # Decrement the posts_count column for the record with an id of 5
119
154
  # DiscussionBoard.decrement_counter(:posts_count, 5)
120
- def decrement_counter(counter_name, id)
121
- update_counters(id, counter_name => -1)
155
+ #
156
+ # # Decrement the posts_count column for the record with an id of 5
157
+ # # and update the updated_at value.
158
+ # DiscussionBoard.decrement_counter(:posts_count, 5, touch: true)
159
+ def decrement_counter(counter_name, id, touch: nil)
160
+ update_counters(id, counter_name => -1, touch: touch)
122
161
  end
123
162
  end
124
163
 
125
164
  private
126
-
127
- def _create_record(*)
165
+ def _create_record(attribute_names = self.attribute_names)
128
166
  id = super
129
167
 
130
168
  each_counter_cached_associations do |association|
131
- if send(association.reflection.name)
132
- association.increment_counters
133
- @_after_create_counter_called = true
134
- end
169
+ association.increment_counters
135
170
  end
136
171
 
137
172
  id
@@ -144,9 +179,7 @@ module ActiveRecord
144
179
  each_counter_cached_associations do |association|
145
180
  foreign_key = association.reflection.foreign_key.to_sym
146
181
  unless destroyed_by_association && destroyed_by_association.foreign_key.to_sym == foreign_key
147
- if send(association.reflection.name)
148
- association.decrement_counters
149
- end
182
+ association.decrement_counters
150
183
  end
151
184
  end
152
185
  end
@@ -159,6 +192,5 @@ module ActiveRecord
159
192
  yield association(name.to_sym) if reflection.belongs_to? && reflection.counter_cache_column
160
193
  end
161
194
  end
162
-
163
195
  end
164
196
  end
@@ -0,0 +1,272 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_record/database_configurations/database_config"
4
+ require "active_record/database_configurations/hash_config"
5
+ require "active_record/database_configurations/url_config"
6
+ require "active_record/database_configurations/connection_url_resolver"
7
+
8
+ module ActiveRecord
9
+ # ActiveRecord::DatabaseConfigurations returns an array of DatabaseConfig
10
+ # objects (either a HashConfig or UrlConfig) that are constructed from the
11
+ # application's database configuration hash or URL string.
12
+ class DatabaseConfigurations
13
+ class InvalidConfigurationError < StandardError; end
14
+
15
+ attr_reader :configurations
16
+ delegate :any?, to: :configurations
17
+
18
+ def initialize(configurations = {})
19
+ @configurations = build_configs(configurations)
20
+ end
21
+
22
+ # Collects the configs for the environment and optionally the specification
23
+ # name passed in. To include replica configurations pass <tt>include_replicas: true</tt>.
24
+ #
25
+ # If a name is provided a single DatabaseConfig object will be
26
+ # returned, otherwise an array of DatabaseConfig objects will be
27
+ # returned that corresponds with the environment and type requested.
28
+ #
29
+ # ==== Options
30
+ #
31
+ # * <tt>env_name:</tt> The environment name. Defaults to +nil+ which will collect
32
+ # configs for all environments.
33
+ # * <tt>name:</tt> The db config name (i.e. primary, animals, etc.). Defaults
34
+ # to +nil+. If no +env_name+ is specified the config for the default env and the
35
+ # passed +name+ will be returned.
36
+ # * <tt>include_replicas:</tt> Determines whether to include replicas in
37
+ # the returned list. Most of the time we're only iterating over the write
38
+ # connection (i.e. migrations don't need to run for the write and read connection).
39
+ # Defaults to +false+.
40
+ def configs_for(env_name: nil, spec_name: nil, name: nil, include_replicas: false)
41
+ if spec_name
42
+ name = spec_name
43
+ ActiveSupport::Deprecation.warn("The kwarg `spec_name` is deprecated in favor of `name`. `spec_name` will be removed in Rails 6.2")
44
+ end
45
+
46
+ env_name ||= default_env if name
47
+ configs = env_with_configs(env_name)
48
+
49
+ unless include_replicas
50
+ configs = configs.select do |db_config|
51
+ !db_config.replica?
52
+ end
53
+ end
54
+
55
+ if name
56
+ configs.find do |db_config|
57
+ db_config.name == name
58
+ end
59
+ else
60
+ configs
61
+ end
62
+ end
63
+
64
+ # Returns the config hash that corresponds with the environment
65
+ #
66
+ # If the application has multiple databases +default_hash+ will
67
+ # return the first config hash for the environment.
68
+ #
69
+ # { database: "my_db", adapter: "mysql2" }
70
+ def default_hash(env = default_env)
71
+ default = find_db_config(env)
72
+ default.configuration_hash if default
73
+ end
74
+ alias :[] :default_hash
75
+ deprecate "[]": "Use configs_for", default_hash: "Use configs_for"
76
+
77
+ # Returns a single DatabaseConfig object based on the requested environment.
78
+ #
79
+ # If the application has multiple databases +find_db_config+ will return
80
+ # the first DatabaseConfig for the environment.
81
+ def find_db_config(env)
82
+ configurations
83
+ .sort_by.with_index { |db_config, i| db_config.for_current_env? ? [0, i] : [1, i] }
84
+ .find do |db_config|
85
+ db_config.env_name == env.to_s ||
86
+ (db_config.for_current_env? && db_config.name == env.to_s)
87
+ end
88
+ end
89
+
90
+ # A primary configuration is one that is named primary or if there is
91
+ # no primary, the first configuration for an environment will be treated
92
+ # as primary. This is used as the "default" configuration and is used
93
+ # when the application needs to treat one configuration differently. For
94
+ # example, when Rails dumps the schema, the primary configuration's schema
95
+ # file will be named `schema.rb` instead of `primary_schema.rb`.
96
+ def primary?(name) # :nodoc:
97
+ return true if name == "primary"
98
+
99
+ first_config = find_db_config(default_env)
100
+ first_config && name == first_config.name
101
+ end
102
+
103
+ # Returns the DatabaseConfigurations object as a Hash.
104
+ def to_h
105
+ configurations.inject({}) do |memo, db_config|
106
+ memo.merge(db_config.env_name => db_config.configuration_hash.stringify_keys)
107
+ end
108
+ end
109
+ deprecate to_h: "You can use `ActiveRecord::Base.configurations.configs_for(env_name: 'env', name: 'primary').configuration_hash` to get the configuration hashes."
110
+
111
+ # Checks if the application's configurations are empty.
112
+ #
113
+ # Aliased to blank?
114
+ def empty?
115
+ configurations.empty?
116
+ end
117
+ alias :blank? :empty?
118
+
119
+ # Returns fully resolved connection, accepts hash, string or symbol.
120
+ # Always returns a DatabaseConfiguration::DatabaseConfig
121
+ #
122
+ # == Examples
123
+ #
124
+ # Symbol representing current environment.
125
+ #
126
+ # DatabaseConfigurations.new("production" => {}).resolve(:production)
127
+ # # => DatabaseConfigurations::HashConfig.new(env_name: "production", config: {})
128
+ #
129
+ # One layer deep hash of connection values.
130
+ #
131
+ # DatabaseConfigurations.new({}).resolve("adapter" => "sqlite3")
132
+ # # => DatabaseConfigurations::HashConfig.new(config: {"adapter" => "sqlite3"})
133
+ #
134
+ # Connection URL.
135
+ #
136
+ # DatabaseConfigurations.new({}).resolve("postgresql://localhost/foo")
137
+ # # => DatabaseConfigurations::UrlConfig.new(config: {"adapter" => "postgresql", "host" => "localhost", "database" => "foo"})
138
+ def resolve(config) # :nodoc:
139
+ return config if DatabaseConfigurations::DatabaseConfig === config
140
+
141
+ case config
142
+ when Symbol
143
+ resolve_symbol_connection(config)
144
+ when Hash, String
145
+ build_db_config_from_raw_config(default_env, "primary", config)
146
+ else
147
+ raise TypeError, "Invalid type for configuration. Expected Symbol, String, or Hash. Got #{config.inspect}"
148
+ end
149
+ end
150
+
151
+ private
152
+ def default_env
153
+ ActiveRecord::ConnectionHandling::DEFAULT_ENV.call.to_s
154
+ end
155
+
156
+ def env_with_configs(env = nil)
157
+ if env
158
+ configurations.select { |db_config| db_config.env_name == env }
159
+ else
160
+ configurations
161
+ end
162
+ end
163
+
164
+ def build_configs(configs)
165
+ return configs.configurations if configs.is_a?(DatabaseConfigurations)
166
+ return configs if configs.is_a?(Array)
167
+
168
+ db_configs = configs.flat_map do |env_name, config|
169
+ if config.is_a?(Hash) && config.all? { |_, v| v.is_a?(Hash) }
170
+ walk_configs(env_name.to_s, config)
171
+ else
172
+ build_db_config_from_raw_config(env_name.to_s, "primary", config)
173
+ end
174
+ end
175
+
176
+ unless db_configs.find(&:for_current_env?)
177
+ db_configs << environment_url_config(default_env, "primary", {})
178
+ end
179
+
180
+ merge_db_environment_variables(default_env, db_configs.compact)
181
+ end
182
+
183
+ def walk_configs(env_name, config)
184
+ config.map do |name, sub_config|
185
+ build_db_config_from_raw_config(env_name, name.to_s, sub_config)
186
+ end
187
+ end
188
+
189
+ def resolve_symbol_connection(name)
190
+ if db_config = find_db_config(name)
191
+ db_config
192
+ else
193
+ raise AdapterNotSpecified, <<~MSG
194
+ The `#{name}` database is not configured for the `#{default_env}` environment.
195
+
196
+ Available databases configurations are:
197
+
198
+ #{build_configuration_sentence}
199
+ MSG
200
+ end
201
+ end
202
+
203
+ def build_configuration_sentence
204
+ configs = configs_for(include_replicas: true)
205
+
206
+ configs.group_by(&:env_name).map do |env, config|
207
+ names = config.map(&:name)
208
+ if names.size > 1
209
+ "#{env}: #{names.join(", ")}"
210
+ else
211
+ env
212
+ end
213
+ end.join("\n")
214
+ end
215
+
216
+ def build_db_config_from_raw_config(env_name, name, config)
217
+ case config
218
+ when String
219
+ build_db_config_from_string(env_name, name, config)
220
+ when Hash
221
+ build_db_config_from_hash(env_name, name, config.symbolize_keys)
222
+ else
223
+ raise InvalidConfigurationError, "'{ #{env_name} => #{config} }' is not a valid configuration. Expected '#{config}' to be a URL string or a Hash."
224
+ end
225
+ end
226
+
227
+ def build_db_config_from_string(env_name, name, config)
228
+ url = config
229
+ uri = URI.parse(url)
230
+ if uri.scheme
231
+ UrlConfig.new(env_name, name, url)
232
+ else
233
+ raise InvalidConfigurationError, "'{ #{env_name} => #{config} }' is not a valid configuration. Expected '#{config}' to be a URL string or a Hash."
234
+ end
235
+ end
236
+
237
+ def build_db_config_from_hash(env_name, name, config)
238
+ if config.has_key?(:url)
239
+ url = config[:url]
240
+ config_without_url = config.dup
241
+ config_without_url.delete :url
242
+
243
+ UrlConfig.new(env_name, name, url, config_without_url)
244
+ else
245
+ HashConfig.new(env_name, name, config)
246
+ end
247
+ end
248
+
249
+ def merge_db_environment_variables(current_env, configs)
250
+ configs.map do |config|
251
+ next config if config.is_a?(UrlConfig) || config.env_name != current_env
252
+
253
+ url_config = environment_url_config(current_env, config.name, config.configuration_hash)
254
+ url_config || config
255
+ end
256
+ end
257
+
258
+ def environment_url_config(env, name, config)
259
+ url = environment_value_for(name)
260
+ return unless url
261
+
262
+ UrlConfig.new(env, name, url, config)
263
+ end
264
+
265
+ def environment_value_for(name)
266
+ name_env_key = "#{name.upcase}_DATABASE_URL"
267
+ url = ENV[name_env_key]
268
+ url ||= ENV["DATABASE_URL"] if name == "primary"
269
+ url
270
+ end
271
+ end
272
+ end
@@ -0,0 +1,98 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/enumerable"
4
+
5
+ module ActiveRecord
6
+ class DatabaseConfigurations
7
+ # Expands a connection string into a hash.
8
+ class ConnectionUrlResolver # :nodoc:
9
+ # == Example
10
+ #
11
+ # url = "postgresql://foo:bar@localhost:9000/foo_test?pool=5&timeout=3000"
12
+ # ConnectionUrlResolver.new(url).to_hash
13
+ # # => {
14
+ # adapter: "postgresql",
15
+ # host: "localhost",
16
+ # port: 9000,
17
+ # database: "foo_test",
18
+ # username: "foo",
19
+ # password: "bar",
20
+ # pool: "5",
21
+ # timeout: "3000"
22
+ # }
23
+ def initialize(url)
24
+ raise "Database URL cannot be empty" if url.blank?
25
+ @uri = uri_parser.parse(url)
26
+ @adapter = @uri.scheme && @uri.scheme.tr("-", "_")
27
+ @adapter = "postgresql" if @adapter == "postgres"
28
+
29
+ if @uri.opaque
30
+ @uri.opaque, @query = @uri.opaque.split("?", 2)
31
+ else
32
+ @query = @uri.query
33
+ end
34
+ end
35
+
36
+ # Converts the given URL to a full connection hash.
37
+ def to_hash
38
+ config = raw_config.compact_blank
39
+ config.map { |key, value| config[key] = uri_parser.unescape(value) if value.is_a? String }
40
+ config
41
+ end
42
+
43
+ private
44
+ attr_reader :uri
45
+
46
+ def uri_parser
47
+ @uri_parser ||= URI::Parser.new
48
+ end
49
+
50
+ # Converts the query parameters of the URI into a hash.
51
+ #
52
+ # "localhost?pool=5&reaping_frequency=2"
53
+ # # => { pool: "5", reaping_frequency: "2" }
54
+ #
55
+ # returns empty hash if no query present.
56
+ #
57
+ # "localhost"
58
+ # # => {}
59
+ def query_hash
60
+ Hash[(@query || "").split("&").map { |pair| pair.split("=", 2) }].symbolize_keys
61
+ end
62
+
63
+ def raw_config
64
+ if uri.opaque
65
+ query_hash.merge(
66
+ adapter: @adapter,
67
+ database: uri.opaque
68
+ )
69
+ else
70
+ query_hash.merge(
71
+ adapter: @adapter,
72
+ username: uri.user,
73
+ password: uri.password,
74
+ port: uri.port,
75
+ database: database_from_path,
76
+ host: uri.hostname
77
+ )
78
+ end
79
+ end
80
+
81
+ # Returns name of the database.
82
+ def database_from_path
83
+ if @adapter == "sqlite3"
84
+ # 'sqlite3:/foo' is absolute, because that makes sense. The
85
+ # corresponding relative version, 'sqlite3:foo', is handled
86
+ # elsewhere, as an "opaque".
87
+
88
+ uri.path
89
+ else
90
+ # Only SQLite uses a filename as the "database" name; for
91
+ # anything else, a leading slash would be silly.
92
+
93
+ uri.path.delete_prefix("/")
94
+ end
95
+ end
96
+ end
97
+ end
98
+ end