activerecord 4.2.11.2 → 6.0.0

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 (372) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +613 -1638
  3. data/MIT-LICENSE +4 -2
  4. data/README.rdoc +13 -12
  5. data/examples/performance.rb +33 -32
  6. data/examples/simple.rb +5 -4
  7. data/lib/active_record.rb +41 -22
  8. data/lib/active_record/aggregations.rb +267 -251
  9. data/lib/active_record/association_relation.rb +11 -6
  10. data/lib/active_record/associations.rb +1737 -1597
  11. data/lib/active_record/associations/alias_tracker.rb +29 -35
  12. data/lib/active_record/associations/association.rb +125 -58
  13. data/lib/active_record/associations/association_scope.rb +103 -132
  14. data/lib/active_record/associations/belongs_to_association.rb +65 -60
  15. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +8 -12
  16. data/lib/active_record/associations/builder/association.rb +27 -40
  17. data/lib/active_record/associations/builder/belongs_to.rb +69 -55
  18. data/lib/active_record/associations/builder/collection_association.rb +10 -33
  19. data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +52 -66
  20. data/lib/active_record/associations/builder/has_many.rb +8 -4
  21. data/lib/active_record/associations/builder/has_one.rb +46 -5
  22. data/lib/active_record/associations/builder/singular_association.rb +16 -10
  23. data/lib/active_record/associations/collection_association.rb +131 -287
  24. data/lib/active_record/associations/collection_proxy.rb +241 -146
  25. data/lib/active_record/associations/foreign_association.rb +10 -1
  26. data/lib/active_record/associations/has_many_association.rb +34 -97
  27. data/lib/active_record/associations/has_many_through_association.rb +60 -87
  28. data/lib/active_record/associations/has_one_association.rb +61 -49
  29. data/lib/active_record/associations/has_one_through_association.rb +20 -11
  30. data/lib/active_record/associations/join_dependency.rb +137 -167
  31. data/lib/active_record/associations/join_dependency/join_association.rb +38 -86
  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 +90 -92
  35. data/lib/active_record/associations/preloader/association.rb +90 -123
  36. data/lib/active_record/associations/preloader/through_association.rb +85 -65
  37. data/lib/active_record/associations/singular_association.rb +18 -39
  38. data/lib/active_record/associations/through_association.rb +38 -18
  39. data/lib/active_record/attribute_assignment.rb +56 -183
  40. data/lib/active_record/attribute_decorators.rb +39 -15
  41. data/lib/active_record/attribute_methods.rb +120 -135
  42. data/lib/active_record/attribute_methods/before_type_cast.rb +13 -8
  43. data/lib/active_record/attribute_methods/dirty.rb +174 -144
  44. data/lib/active_record/attribute_methods/primary_key.rb +91 -83
  45. data/lib/active_record/attribute_methods/query.rb +6 -5
  46. data/lib/active_record/attribute_methods/read.rb +20 -76
  47. data/lib/active_record/attribute_methods/serialization.rb +40 -20
  48. data/lib/active_record/attribute_methods/time_zone_conversion.rb +58 -36
  49. data/lib/active_record/attribute_methods/write.rb +32 -54
  50. data/lib/active_record/attributes.rb +214 -82
  51. data/lib/active_record/autosave_association.rb +91 -37
  52. data/lib/active_record/base.rb +57 -45
  53. data/lib/active_record/callbacks.rb +100 -74
  54. data/lib/active_record/coders/json.rb +3 -1
  55. data/lib/active_record/coders/yaml_column.rb +24 -12
  56. data/lib/active_record/connection_adapters/abstract/connection_pool.rb +796 -296
  57. data/lib/active_record/connection_adapters/abstract/database_limits.rb +26 -8
  58. data/lib/active_record/connection_adapters/abstract/database_statements.rb +234 -115
  59. data/lib/active_record/connection_adapters/abstract/query_cache.rb +82 -23
  60. data/lib/active_record/connection_adapters/abstract/quoting.rb +170 -53
  61. data/lib/active_record/connection_adapters/abstract/savepoints.rb +5 -3
  62. data/lib/active_record/connection_adapters/abstract/schema_creation.rb +74 -46
  63. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +356 -227
  64. data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +79 -36
  65. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +664 -243
  66. data/lib/active_record/connection_adapters/abstract/transaction.rb +191 -83
  67. data/lib/active_record/connection_adapters/abstract_adapter.rb +460 -204
  68. data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +510 -635
  69. data/lib/active_record/connection_adapters/column.rb +56 -43
  70. data/lib/active_record/connection_adapters/connection_specification.rb +174 -152
  71. data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +29 -0
  72. data/lib/active_record/connection_adapters/mysql/column.rb +27 -0
  73. data/lib/active_record/connection_adapters/mysql/database_statements.rb +200 -0
  74. data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +72 -0
  75. data/lib/active_record/connection_adapters/mysql/quoting.rb +81 -0
  76. data/lib/active_record/connection_adapters/mysql/schema_creation.rb +72 -0
  77. data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +95 -0
  78. data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +88 -0
  79. data/lib/active_record/connection_adapters/mysql/schema_statements.rb +264 -0
  80. data/lib/active_record/connection_adapters/mysql/type_metadata.rb +31 -0
  81. data/lib/active_record/connection_adapters/mysql2_adapter.rb +58 -180
  82. data/lib/active_record/connection_adapters/postgresql/column.rb +21 -11
  83. data/lib/active_record/connection_adapters/postgresql/database_statements.rb +64 -114
  84. data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
  85. data/lib/active_record/connection_adapters/postgresql/oid.rb +23 -25
  86. data/lib/active_record/connection_adapters/postgresql/oid/array.rb +50 -58
  87. data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +9 -8
  88. data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +2 -0
  89. data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +4 -2
  90. data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +5 -1
  91. data/lib/active_record/connection_adapters/postgresql/oid/date.rb +13 -1
  92. data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +9 -22
  93. data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +3 -1
  94. data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +5 -3
  95. data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +31 -19
  96. data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +2 -0
  97. data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +3 -11
  98. data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +45 -0
  99. data/lib/active_record/connection_adapters/postgresql/oid/money.rb +7 -9
  100. data/lib/active_record/connection_adapters/postgresql/oid/{integer.rb → oid.rb} +6 -2
  101. data/lib/active_record/connection_adapters/postgresql/oid/point.rb +33 -11
  102. data/lib/active_record/connection_adapters/postgresql/oid/range.rb +52 -34
  103. data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +4 -5
  104. data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +58 -54
  105. data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +10 -5
  106. data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +3 -1
  107. data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +3 -1
  108. data/lib/active_record/connection_adapters/postgresql/quoting.rb +144 -47
  109. data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +27 -14
  110. data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +76 -0
  111. data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +178 -108
  112. data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
  113. data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +470 -290
  114. data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +36 -0
  115. data/lib/active_record/connection_adapters/postgresql/utils.rb +12 -8
  116. data/lib/active_record/connection_adapters/postgresql_adapter.rb +551 -356
  117. data/lib/active_record/connection_adapters/schema_cache.rb +72 -25
  118. data/lib/active_record/connection_adapters/sql_type_metadata.rb +37 -0
  119. data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +118 -0
  120. data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
  121. data/lib/active_record/connection_adapters/sqlite3/quoting.rb +103 -0
  122. data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +17 -0
  123. data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +19 -0
  124. data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +18 -0
  125. data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +137 -0
  126. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +290 -345
  127. data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
  128. data/lib/active_record/connection_handling.rb +176 -41
  129. data/lib/active_record/core.rb +251 -231
  130. data/lib/active_record/counter_cache.rb +67 -49
  131. data/lib/active_record/database_configurations.rb +233 -0
  132. data/lib/active_record/database_configurations/database_config.rb +37 -0
  133. data/lib/active_record/database_configurations/hash_config.rb +50 -0
  134. data/lib/active_record/database_configurations/url_config.rb +79 -0
  135. data/lib/active_record/define_callbacks.rb +22 -0
  136. data/lib/active_record/dynamic_matchers.rb +87 -105
  137. data/lib/active_record/enum.rb +163 -86
  138. data/lib/active_record/errors.rb +188 -53
  139. data/lib/active_record/explain.rb +23 -11
  140. data/lib/active_record/explain_registry.rb +4 -2
  141. data/lib/active_record/explain_subscriber.rb +10 -5
  142. data/lib/active_record/fixture_set/file.rb +35 -9
  143. data/lib/active_record/fixture_set/model_metadata.rb +33 -0
  144. data/lib/active_record/fixture_set/render_context.rb +17 -0
  145. data/lib/active_record/fixture_set/table_row.rb +153 -0
  146. data/lib/active_record/fixture_set/table_rows.rb +47 -0
  147. data/lib/active_record/fixtures.rb +228 -499
  148. data/lib/active_record/gem_version.rb +6 -4
  149. data/lib/active_record/inheritance.rb +158 -112
  150. data/lib/active_record/insert_all.rb +179 -0
  151. data/lib/active_record/integration.rb +123 -29
  152. data/lib/active_record/internal_metadata.rb +53 -0
  153. data/lib/active_record/legacy_yaml_adapter.rb +21 -3
  154. data/lib/active_record/locale/en.yml +3 -2
  155. data/lib/active_record/locking/optimistic.rb +87 -96
  156. data/lib/active_record/locking/pessimistic.rb +18 -6
  157. data/lib/active_record/log_subscriber.rb +76 -33
  158. data/lib/active_record/middleware/database_selector.rb +75 -0
  159. data/lib/active_record/middleware/database_selector/resolver.rb +92 -0
  160. data/lib/active_record/middleware/database_selector/resolver/session.rb +45 -0
  161. data/lib/active_record/migration.rb +621 -303
  162. data/lib/active_record/migration/command_recorder.rb +177 -90
  163. data/lib/active_record/migration/compatibility.rb +244 -0
  164. data/lib/active_record/migration/join_table.rb +8 -6
  165. data/lib/active_record/model_schema.rb +312 -112
  166. data/lib/active_record/nested_attributes.rb +264 -222
  167. data/lib/active_record/no_touching.rb +14 -1
  168. data/lib/active_record/null_relation.rb +24 -37
  169. data/lib/active_record/persistence.rb +557 -125
  170. data/lib/active_record/query_cache.rb +19 -23
  171. data/lib/active_record/querying.rb +43 -29
  172. data/lib/active_record/railtie.rb +143 -44
  173. data/lib/active_record/railties/collection_cache_association_loading.rb +34 -0
  174. data/lib/active_record/railties/console_sandbox.rb +2 -0
  175. data/lib/active_record/railties/controller_runtime.rb +34 -33
  176. data/lib/active_record/railties/databases.rake +328 -185
  177. data/lib/active_record/readonly_attributes.rb +5 -4
  178. data/lib/active_record/reflection.rb +428 -279
  179. data/lib/active_record/relation.rb +518 -341
  180. data/lib/active_record/relation/batches.rb +207 -55
  181. data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
  182. data/lib/active_record/relation/calculations.rb +267 -253
  183. data/lib/active_record/relation/delegation.rb +70 -80
  184. data/lib/active_record/relation/finder_methods.rb +277 -241
  185. data/lib/active_record/relation/from_clause.rb +26 -0
  186. data/lib/active_record/relation/merger.rb +78 -87
  187. data/lib/active_record/relation/predicate_builder.rb +114 -119
  188. data/lib/active_record/relation/predicate_builder/array_handler.rb +27 -26
  189. data/lib/active_record/relation/predicate_builder/association_query_value.rb +43 -0
  190. data/lib/active_record/relation/predicate_builder/base_handler.rb +18 -0
  191. data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +19 -0
  192. data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +53 -0
  193. data/lib/active_record/relation/predicate_builder/range_handler.rb +22 -0
  194. data/lib/active_record/relation/predicate_builder/relation_handler.rb +7 -1
  195. data/lib/active_record/relation/query_attribute.rb +50 -0
  196. data/lib/active_record/relation/query_methods.rb +575 -394
  197. data/lib/active_record/relation/record_fetch_warning.rb +51 -0
  198. data/lib/active_record/relation/spawn_methods.rb +11 -13
  199. data/lib/active_record/relation/where_clause.rb +190 -0
  200. data/lib/active_record/relation/where_clause_factory.rb +33 -0
  201. data/lib/active_record/result.rb +79 -42
  202. data/lib/active_record/runtime_registry.rb +6 -4
  203. data/lib/active_record/sanitization.rb +144 -121
  204. data/lib/active_record/schema.rb +21 -24
  205. data/lib/active_record/schema_dumper.rb +112 -93
  206. data/lib/active_record/schema_migration.rb +24 -17
  207. data/lib/active_record/scoping.rb +45 -26
  208. data/lib/active_record/scoping/default.rb +101 -85
  209. data/lib/active_record/scoping/named.rb +86 -33
  210. data/lib/active_record/secure_token.rb +40 -0
  211. data/lib/active_record/serialization.rb +5 -5
  212. data/lib/active_record/statement_cache.rb +73 -36
  213. data/lib/active_record/store.rb +127 -42
  214. data/lib/active_record/suppressor.rb +61 -0
  215. data/lib/active_record/table_metadata.rb +75 -0
  216. data/lib/active_record/tasks/database_tasks.rb +307 -100
  217. data/lib/active_record/tasks/mysql_database_tasks.rb +55 -99
  218. data/lib/active_record/tasks/postgresql_database_tasks.rb +81 -41
  219. data/lib/active_record/tasks/sqlite_database_tasks.rb +38 -16
  220. data/lib/active_record/test_databases.rb +23 -0
  221. data/lib/active_record/test_fixtures.rb +224 -0
  222. data/lib/active_record/timestamp.rb +86 -40
  223. data/lib/active_record/touch_later.rb +66 -0
  224. data/lib/active_record/transactions.rb +216 -150
  225. data/lib/active_record/translation.rb +3 -1
  226. data/lib/active_record/type.rb +78 -23
  227. data/lib/active_record/type/adapter_specific_registry.rb +129 -0
  228. data/lib/active_record/type/date.rb +4 -45
  229. data/lib/active_record/type/date_time.rb +4 -49
  230. data/lib/active_record/type/decimal_without_scale.rb +6 -2
  231. data/lib/active_record/type/hash_lookup_type_map.rb +5 -3
  232. data/lib/active_record/type/internal/timezone.rb +17 -0
  233. data/lib/active_record/type/json.rb +30 -0
  234. data/lib/active_record/type/serialized.rb +24 -15
  235. data/lib/active_record/type/text.rb +2 -2
  236. data/lib/active_record/type/time.rb +11 -16
  237. data/lib/active_record/type/type_map.rb +15 -17
  238. data/lib/active_record/type/unsigned_integer.rb +9 -7
  239. data/lib/active_record/type_caster.rb +9 -0
  240. data/lib/active_record/type_caster/connection.rb +34 -0
  241. data/lib/active_record/type_caster/map.rb +20 -0
  242. data/lib/active_record/validations.rb +39 -35
  243. data/lib/active_record/validations/absence.rb +25 -0
  244. data/lib/active_record/validations/associated.rb +13 -4
  245. data/lib/active_record/validations/length.rb +26 -0
  246. data/lib/active_record/validations/presence.rb +14 -13
  247. data/lib/active_record/validations/uniqueness.rb +42 -55
  248. data/lib/active_record/version.rb +3 -1
  249. data/lib/arel.rb +51 -0
  250. data/lib/arel/alias_predication.rb +9 -0
  251. data/lib/arel/attributes.rb +22 -0
  252. data/lib/arel/attributes/attribute.rb +37 -0
  253. data/lib/arel/collectors/bind.rb +24 -0
  254. data/lib/arel/collectors/composite.rb +31 -0
  255. data/lib/arel/collectors/plain_string.rb +20 -0
  256. data/lib/arel/collectors/sql_string.rb +20 -0
  257. data/lib/arel/collectors/substitute_binds.rb +28 -0
  258. data/lib/arel/crud.rb +42 -0
  259. data/lib/arel/delete_manager.rb +18 -0
  260. data/lib/arel/errors.rb +9 -0
  261. data/lib/arel/expressions.rb +29 -0
  262. data/lib/arel/factory_methods.rb +49 -0
  263. data/lib/arel/insert_manager.rb +49 -0
  264. data/lib/arel/math.rb +45 -0
  265. data/lib/arel/nodes.rb +68 -0
  266. data/lib/arel/nodes/and.rb +32 -0
  267. data/lib/arel/nodes/ascending.rb +23 -0
  268. data/lib/arel/nodes/binary.rb +52 -0
  269. data/lib/arel/nodes/bind_param.rb +36 -0
  270. data/lib/arel/nodes/case.rb +55 -0
  271. data/lib/arel/nodes/casted.rb +50 -0
  272. data/lib/arel/nodes/comment.rb +29 -0
  273. data/lib/arel/nodes/count.rb +12 -0
  274. data/lib/arel/nodes/delete_statement.rb +45 -0
  275. data/lib/arel/nodes/descending.rb +23 -0
  276. data/lib/arel/nodes/equality.rb +18 -0
  277. data/lib/arel/nodes/extract.rb +24 -0
  278. data/lib/arel/nodes/false.rb +16 -0
  279. data/lib/arel/nodes/full_outer_join.rb +8 -0
  280. data/lib/arel/nodes/function.rb +44 -0
  281. data/lib/arel/nodes/grouping.rb +8 -0
  282. data/lib/arel/nodes/in.rb +8 -0
  283. data/lib/arel/nodes/infix_operation.rb +80 -0
  284. data/lib/arel/nodes/inner_join.rb +8 -0
  285. data/lib/arel/nodes/insert_statement.rb +37 -0
  286. data/lib/arel/nodes/join_source.rb +20 -0
  287. data/lib/arel/nodes/matches.rb +18 -0
  288. data/lib/arel/nodes/named_function.rb +23 -0
  289. data/lib/arel/nodes/node.rb +50 -0
  290. data/lib/arel/nodes/node_expression.rb +13 -0
  291. data/lib/arel/nodes/outer_join.rb +8 -0
  292. data/lib/arel/nodes/over.rb +15 -0
  293. data/lib/arel/nodes/regexp.rb +16 -0
  294. data/lib/arel/nodes/right_outer_join.rb +8 -0
  295. data/lib/arel/nodes/select_core.rb +67 -0
  296. data/lib/arel/nodes/select_statement.rb +41 -0
  297. data/lib/arel/nodes/sql_literal.rb +16 -0
  298. data/lib/arel/nodes/string_join.rb +11 -0
  299. data/lib/arel/nodes/table_alias.rb +27 -0
  300. data/lib/arel/nodes/terminal.rb +16 -0
  301. data/lib/arel/nodes/true.rb +16 -0
  302. data/lib/arel/nodes/unary.rb +45 -0
  303. data/lib/arel/nodes/unary_operation.rb +20 -0
  304. data/lib/arel/nodes/unqualified_column.rb +22 -0
  305. data/lib/arel/nodes/update_statement.rb +41 -0
  306. data/lib/arel/nodes/values_list.rb +9 -0
  307. data/lib/arel/nodes/window.rb +126 -0
  308. data/lib/arel/nodes/with.rb +11 -0
  309. data/lib/arel/order_predications.rb +13 -0
  310. data/lib/arel/predications.rb +257 -0
  311. data/lib/arel/select_manager.rb +271 -0
  312. data/lib/arel/table.rb +110 -0
  313. data/lib/arel/tree_manager.rb +72 -0
  314. data/lib/arel/update_manager.rb +34 -0
  315. data/lib/arel/visitors.rb +20 -0
  316. data/lib/arel/visitors/depth_first.rb +204 -0
  317. data/lib/arel/visitors/dot.rb +297 -0
  318. data/lib/arel/visitors/ibm_db.rb +34 -0
  319. data/lib/arel/visitors/informix.rb +62 -0
  320. data/lib/arel/visitors/mssql.rb +157 -0
  321. data/lib/arel/visitors/mysql.rb +83 -0
  322. data/lib/arel/visitors/oracle.rb +159 -0
  323. data/lib/arel/visitors/oracle12.rb +66 -0
  324. data/lib/arel/visitors/postgresql.rb +110 -0
  325. data/lib/arel/visitors/sqlite.rb +39 -0
  326. data/lib/arel/visitors/to_sql.rb +889 -0
  327. data/lib/arel/visitors/visitor.rb +46 -0
  328. data/lib/arel/visitors/where_sql.rb +23 -0
  329. data/lib/arel/window_predications.rb +9 -0
  330. data/lib/rails/generators/active_record.rb +7 -5
  331. data/lib/rails/generators/active_record/application_record/application_record_generator.rb +27 -0
  332. data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +5 -0
  333. data/lib/rails/generators/active_record/migration.rb +31 -1
  334. data/lib/rails/generators/active_record/migration/migration_generator.rb +42 -37
  335. data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +24 -0
  336. data/lib/rails/generators/active_record/migration/templates/{migration.rb → migration.rb.tt} +11 -2
  337. data/lib/rails/generators/active_record/model/model_generator.rb +19 -22
  338. data/lib/rails/generators/active_record/model/templates/model.rb.tt +22 -0
  339. data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +0 -0
  340. metadata +164 -59
  341. data/lib/active_record/associations/preloader/belongs_to.rb +0 -17
  342. data/lib/active_record/associations/preloader/collection_association.rb +0 -24
  343. data/lib/active_record/associations/preloader/has_many.rb +0 -17
  344. data/lib/active_record/associations/preloader/has_many_through.rb +0 -19
  345. data/lib/active_record/associations/preloader/has_one.rb +0 -23
  346. data/lib/active_record/associations/preloader/has_one_through.rb +0 -9
  347. data/lib/active_record/associations/preloader/singular_association.rb +0 -21
  348. data/lib/active_record/attribute.rb +0 -163
  349. data/lib/active_record/attribute_set.rb +0 -81
  350. data/lib/active_record/attribute_set/builder.rb +0 -106
  351. data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -498
  352. data/lib/active_record/connection_adapters/postgresql/array_parser.rb +0 -93
  353. data/lib/active_record/connection_adapters/postgresql/oid/float.rb +0 -21
  354. data/lib/active_record/connection_adapters/postgresql/oid/infinity.rb +0 -13
  355. data/lib/active_record/connection_adapters/postgresql/oid/json.rb +0 -35
  356. data/lib/active_record/connection_adapters/postgresql/oid/time.rb +0 -11
  357. data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
  358. data/lib/active_record/serializers/xml_serializer.rb +0 -193
  359. data/lib/active_record/type/big_integer.rb +0 -13
  360. data/lib/active_record/type/binary.rb +0 -50
  361. data/lib/active_record/type/boolean.rb +0 -31
  362. data/lib/active_record/type/decimal.rb +0 -64
  363. data/lib/active_record/type/decorator.rb +0 -14
  364. data/lib/active_record/type/float.rb +0 -19
  365. data/lib/active_record/type/integer.rb +0 -59
  366. data/lib/active_record/type/mutable.rb +0 -16
  367. data/lib/active_record/type/numeric.rb +0 -36
  368. data/lib/active_record/type/string.rb +0 -40
  369. data/lib/active_record/type/time_value.rb +0 -38
  370. data/lib/active_record/type/value.rb +0 -110
  371. data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb +0 -19
  372. data/lib/rails/generators/active_record/model/templates/model.rb +0 -10
@@ -1,26 +1,35 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/deprecation"
4
+
1
5
  module ActiveRecord
2
6
  module ConnectionAdapters # :nodoc:
3
7
  module DatabaseLimits
8
+ def max_identifier_length # :nodoc:
9
+ 64
10
+ end
4
11
 
5
12
  # Returns the maximum length of a table alias.
6
13
  def table_alias_length
7
- 255
14
+ max_identifier_length
8
15
  end
9
16
 
10
17
  # Returns the maximum length of a column name.
11
18
  def column_name_length
12
- 64
19
+ max_identifier_length
13
20
  end
21
+ deprecate :column_name_length
14
22
 
15
23
  # Returns the maximum length of a table name.
16
24
  def table_name_length
17
- 64
25
+ max_identifier_length
18
26
  end
27
+ deprecate :table_name_length
19
28
 
20
29
  # Returns the maximum allowed length for an index name. This
21
- # limit is enforced by rails and Is less than or equal to
22
- # <tt>index_name_length</tt>. The gap between
23
- # <tt>index_name_length</tt> is to allow internal rails
30
+ # limit is enforced by \Rails and is less than or equal to
31
+ # #index_name_length. The gap between
32
+ # #index_name_length is to allow internal \Rails
24
33
  # operations to use prefixes in temporary operations.
25
34
  def allowed_index_name_length
26
35
  index_name_length
@@ -28,26 +37,29 @@ module ActiveRecord
28
37
 
29
38
  # Returns the maximum length of an index name.
30
39
  def index_name_length
31
- 64
40
+ max_identifier_length
32
41
  end
33
42
 
34
43
  # Returns the maximum number of columns per table.
35
44
  def columns_per_table
36
45
  1024
37
46
  end
47
+ deprecate :columns_per_table
38
48
 
39
49
  # Returns the maximum number of indexes per table.
40
50
  def indexes_per_table
41
51
  16
42
52
  end
53
+ deprecate :indexes_per_table
43
54
 
44
55
  # Returns the maximum number of columns in a multicolumn index.
45
56
  def columns_per_multicolumn_index
46
57
  16
47
58
  end
59
+ deprecate :columns_per_multicolumn_index
48
60
 
49
61
  # Returns the maximum number of elements in an IN (x,y,z) clause.
50
- # nil means no limit.
62
+ # +nil+ means no limit.
51
63
  def in_clause_length
52
64
  nil
53
65
  end
@@ -56,12 +68,18 @@ module ActiveRecord
56
68
  def sql_query_length
57
69
  1048575
58
70
  end
71
+ deprecate :sql_query_length
59
72
 
60
73
  # Returns maximum number of joins in a single query.
61
74
  def joins_per_query
62
75
  256
63
76
  end
77
+ deprecate :joins_per_query
64
78
 
79
+ private
80
+ def bind_params_length
81
+ 65535
82
+ end
65
83
  end
66
84
  end
67
85
  end
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActiveRecord
2
4
  module ConnectionAdapters # :nodoc:
3
5
  module DatabaseStatements
@@ -7,29 +9,66 @@ module ActiveRecord
7
9
  end
8
10
 
9
11
  # Converts an arel AST to SQL
10
- def to_sql(arel, binds = [])
11
- if arel.respond_to?(:ast)
12
- collected = visitor.accept(arel.ast, collector)
13
- collected.compile(binds.dup, self)
12
+ def to_sql(arel_or_sql_string, binds = [])
13
+ sql, _ = to_sql_and_binds(arel_or_sql_string, binds)
14
+ sql
15
+ end
16
+
17
+ def to_sql_and_binds(arel_or_sql_string, binds = []) # :nodoc:
18
+ if arel_or_sql_string.respond_to?(:ast)
19
+ unless binds.empty?
20
+ raise "Passing bind parameters with an arel AST is forbidden. " \
21
+ "The values must be stored on the AST directly"
22
+ end
23
+
24
+ if prepared_statements
25
+ sql, binds = visitor.compile(arel_or_sql_string.ast, collector)
26
+
27
+ if binds.length > bind_params_length
28
+ unprepared_statement do
29
+ sql, binds = to_sql_and_binds(arel_or_sql_string)
30
+ visitor.preparable = false
31
+ end
32
+ end
33
+ else
34
+ sql = visitor.compile(arel_or_sql_string.ast, collector)
35
+ end
36
+ [sql.freeze, binds]
14
37
  else
15
- arel
38
+ visitor.preparable = false if prepared_statements
39
+ [arel_or_sql_string.dup.freeze, binds]
16
40
  end
17
41
  end
42
+ private :to_sql_and_binds
18
43
 
19
44
  # This is used in the StatementCache object. It returns an object that
20
45
  # can be used to query the database repeatedly.
21
- def cacheable_query(arel) # :nodoc:
46
+ def cacheable_query(klass, arel) # :nodoc:
22
47
  if prepared_statements
23
- ActiveRecord::StatementCache.query visitor, arel.ast
48
+ sql, binds = visitor.compile(arel.ast, collector)
49
+ query = klass.query(sql)
24
50
  else
25
- ActiveRecord::StatementCache.partial_query visitor, arel.ast, collector
51
+ collector = klass.partial_query_collector
52
+ parts, binds = visitor.compile(arel.ast, collector)
53
+ query = klass.partial_query(parts)
26
54
  end
55
+ [query, binds]
27
56
  end
28
57
 
29
58
  # Returns an ActiveRecord::Result instance.
30
- def select_all(arel, name = nil, binds = [])
31
- arel, binds = binds_from_relation arel, binds
32
- select(to_sql(arel, binds), name, binds)
59
+ def select_all(arel, name = nil, binds = [], preparable: nil)
60
+ arel = arel_from_relation(arel)
61
+ sql, binds = to_sql_and_binds(arel, binds)
62
+
63
+ if preparable.nil?
64
+ preparable = prepared_statements ? visitor.preparable : false
65
+ end
66
+
67
+ if prepared_statements && preparable
68
+ select_prepared(sql, name, binds)
69
+ else
70
+ select(sql, name, binds)
71
+ end
33
72
  end
34
73
 
35
74
  # Returns a record hash with the column names as keys and column values
@@ -40,89 +79,118 @@ module ActiveRecord
40
79
 
41
80
  # Returns a single value from a record
42
81
  def select_value(arel, name = nil, binds = [])
43
- if result = select_one(arel, name, binds)
44
- result.values.first
45
- end
82
+ single_value_from_rows(select_rows(arel, name, binds))
46
83
  end
47
84
 
48
85
  # Returns an array of the values of the first column in a select:
49
86
  # select_values("SELECT id FROM companies LIMIT 3") => [1,2,3]
50
- def select_values(arel, name = nil)
51
- arel, binds = binds_from_relation arel, []
52
- select_rows(to_sql(arel, binds), name, binds).map(&:first)
87
+ def select_values(arel, name = nil, binds = [])
88
+ select_rows(arel, name, binds).map(&:first)
53
89
  end
54
90
 
55
91
  # Returns an array of arrays containing the field values.
56
92
  # Order is the same as that returned by +columns+.
57
- def select_rows(sql, name = nil, binds = [])
93
+ def select_rows(arel, name = nil, binds = [])
94
+ select_all(arel, name, binds).rows
95
+ end
96
+
97
+ def query_value(sql, name = nil) # :nodoc:
98
+ single_value_from_rows(query(sql, name))
99
+ end
100
+
101
+ def query_values(sql, name = nil) # :nodoc:
102
+ query(sql, name).map(&:first)
58
103
  end
59
- undef_method :select_rows
60
104
 
61
- # Executes the SQL statement in the context of this connection.
105
+ def query(sql, name = nil) # :nodoc:
106
+ exec_query(sql, name).rows
107
+ end
108
+
109
+ # Determines whether the SQL statement is a write query.
110
+ def write_query?(sql)
111
+ raise NotImplementedError
112
+ end
113
+
114
+ # Executes the SQL statement in the context of this connection and returns
115
+ # the raw result from the connection adapter.
116
+ # Note: depending on your database connector, the result returned by this
117
+ # method may be manually memory managed. Consider using the exec_query
118
+ # wrapper instead.
62
119
  def execute(sql, name = nil)
120
+ raise NotImplementedError
63
121
  end
64
- undef_method :execute
65
122
 
66
123
  # Executes +sql+ statement in the context of this connection using
67
124
  # +binds+ as the bind substitutes. +name+ is logged along with
68
125
  # the executed +sql+ statement.
69
- def exec_query(sql, name = 'SQL', binds = [])
126
+ def exec_query(sql, name = "SQL", binds = [], prepare: false)
127
+ raise NotImplementedError
70
128
  end
71
129
 
72
130
  # Executes insert +sql+ statement in the context of this connection using
73
131
  # +binds+ as the bind substitutes. +name+ is logged along with
74
132
  # the executed +sql+ statement.
75
- def exec_insert(sql, name, binds, pk = nil, sequence_name = nil)
133
+ def exec_insert(sql, name = nil, binds = [], pk = nil, sequence_name = nil)
134
+ sql, binds = sql_for_insert(sql, pk, binds)
76
135
  exec_query(sql, name, binds)
77
136
  end
78
137
 
79
138
  # Executes delete +sql+ statement in the context of this connection using
80
139
  # +binds+ as the bind substitutes. +name+ is logged along with
81
140
  # the executed +sql+ statement.
82
- def exec_delete(sql, name, binds)
141
+ def exec_delete(sql, name = nil, binds = [])
83
142
  exec_query(sql, name, binds)
84
143
  end
85
144
 
86
- # Executes the truncate statement.
87
- def truncate(table_name, name = nil)
88
- raise NotImplementedError
89
- end
90
-
91
145
  # Executes update +sql+ statement in the context of this connection using
92
146
  # +binds+ as the bind substitutes. +name+ is logged along with
93
147
  # the executed +sql+ statement.
94
- def exec_update(sql, name, binds)
148
+ def exec_update(sql, name = nil, binds = [])
95
149
  exec_query(sql, name, binds)
96
150
  end
97
151
 
98
- # Returns the last auto-generated ID from the affected table.
152
+ # Executes an INSERT query and returns the new record's ID
99
153
  #
100
- # +id_value+ will be returned unless the value is nil, in
154
+ # +id_value+ will be returned unless the value is +nil+, in
101
155
  # which case the database will attempt to calculate the last inserted
102
156
  # id and return that value.
103
157
  #
104
158
  # If the next id was calculated in advance (as in Oracle), it should be
105
159
  # passed in as +id_value+.
106
160
  def insert(arel, name = nil, pk = nil, id_value = nil, sequence_name = nil, binds = [])
107
- sql, binds = sql_for_insert(to_sql(arel, binds), pk, id_value, sequence_name, binds)
108
- value = exec_insert(sql, name, binds, pk, sequence_name)
161
+ sql, binds = to_sql_and_binds(arel, binds)
162
+ value = exec_insert(sql, name, binds, pk, sequence_name)
109
163
  id_value || last_inserted_id(value)
110
164
  end
165
+ alias create insert
111
166
 
112
167
  # Executes the update statement and returns the number of rows affected.
113
168
  def update(arel, name = nil, binds = [])
114
- exec_update(to_sql(arel, binds), name, binds)
169
+ sql, binds = to_sql_and_binds(arel, binds)
170
+ exec_update(sql, name, binds)
115
171
  end
116
172
 
117
173
  # Executes the delete statement and returns the number of rows affected.
118
174
  def delete(arel, name = nil, binds = [])
119
- exec_delete(to_sql(arel, binds), name, binds)
175
+ sql, binds = to_sql_and_binds(arel, binds)
176
+ exec_delete(sql, name, binds)
177
+ end
178
+
179
+ # Executes the truncate statement.
180
+ def truncate(table_name, name = nil)
181
+ execute(build_truncate_statements(table_name), name)
120
182
  end
121
183
 
122
- # Returns +true+ when the connection adapter supports prepared statement
123
- # caching, otherwise returns +false+
124
- def supports_statement_cache?
125
- false
184
+ def truncate_tables(*table_names) # :nodoc:
185
+ return if table_names.empty?
186
+
187
+ with_multi_statements do
188
+ disable_referential_integrity do
189
+ Array(build_truncate_statements(*table_names)).each do |sql|
190
+ execute_batch(sql, "Truncate Tables")
191
+ end
192
+ end
193
+ end
126
194
  end
127
195
 
128
196
  # Runs the given block in a database transaction, and returns the result
@@ -136,7 +204,7 @@ module ActiveRecord
136
204
  #
137
205
  # In order to get around this problem, #transaction will emulate the effect
138
206
  # of nested transactions, by using savepoints:
139
- # http://dev.mysql.com/doc/refman/5.0/en/savepoint.html
207
+ # https://dev.mysql.com/doc/refman/5.7/en/savepoint.html
140
208
  # Savepoints are supported by MySQL and PostgreSQL. SQLite3 version >= '3.6.8'
141
209
  # supports savepoints.
142
210
  #
@@ -188,29 +256,25 @@ module ActiveRecord
188
256
  # You should consult the documentation for your database to understand the
189
257
  # semantics of these different levels:
190
258
  #
191
- # * http://www.postgresql.org/docs/9.1/static/transaction-iso.html
192
- # * https://dev.mysql.com/doc/refman/5.0/en/set-transaction.html
259
+ # * https://www.postgresql.org/docs/current/static/transaction-iso.html
260
+ # * https://dev.mysql.com/doc/refman/5.7/en/set-transaction.html
193
261
  #
194
- # An <tt>ActiveRecord::TransactionIsolationError</tt> will be raised if:
262
+ # An ActiveRecord::TransactionIsolationError will be raised if:
195
263
  #
196
264
  # * The adapter does not support setting the isolation level
197
265
  # * You are joining an existing open transaction
198
266
  # * You are creating a nested (savepoint) transaction
199
267
  #
200
- # The mysql, mysql2 and postgresql adapters support setting the transaction
201
- # isolation level. However, support is disabled for MySQL versions below 5,
202
- # because they are affected by a bug[http://bugs.mysql.com/bug.php?id=39170]
203
- # which means the isolation level gets persisted outside the transaction.
204
- def transaction(options = {})
205
- options.assert_valid_keys :requires_new, :joinable, :isolation
206
-
207
- if !options[:requires_new] && current_transaction.joinable?
208
- if options[:isolation]
268
+ # The mysql2 and postgresql adapters support setting the transaction
269
+ # isolation level.
270
+ def transaction(requires_new: nil, isolation: nil, joinable: true)
271
+ if !requires_new && current_transaction.joinable?
272
+ if isolation
209
273
  raise ActiveRecord::TransactionIsolationError, "cannot set isolation when joining a transaction"
210
274
  end
211
275
  yield
212
276
  else
213
- transaction_manager.within_new_transaction(options) { yield }
277
+ transaction_manager.within_new_transaction(isolation: isolation, joinable: joinable) { yield }
214
278
  end
215
279
  rescue ActiveRecord::Rollback
216
280
  # rollbacks are silently swallowed
@@ -218,14 +282,16 @@ module ActiveRecord
218
282
 
219
283
  attr_reader :transaction_manager #:nodoc:
220
284
 
221
- delegate :within_new_transaction, :open_transactions, :current_transaction, :begin_transaction, :commit_transaction, :rollback_transaction, to: :transaction_manager
285
+ delegate :within_new_transaction, :open_transactions, :current_transaction, :begin_transaction,
286
+ :commit_transaction, :rollback_transaction, :materialize_transactions,
287
+ :disable_lazy_transactions!, :enable_lazy_transactions!, to: :transaction_manager
222
288
 
223
289
  def transaction_open?
224
290
  current_transaction.open?
225
291
  end
226
292
 
227
293
  def reset_transaction #:nodoc:
228
- @transaction_manager = TransactionManager.new(self)
294
+ @transaction_manager = ConnectionAdapters::TransactionManager.new(self)
229
295
  end
230
296
 
231
297
  # Register a record with the current transaction so that its after_commit and after_rollback callbacks
@@ -272,9 +338,6 @@ module ActiveRecord
272
338
  exec_rollback_to_savepoint(name)
273
339
  end
274
340
 
275
- def exec_rollback_to_savepoint(name = nil) #:nodoc:
276
- end
277
-
278
341
  def default_sequence_name(table, column)
279
342
  nil
280
343
  end
@@ -286,107 +349,163 @@ module ActiveRecord
286
349
 
287
350
  # Inserts the given fixture into the table. Overridden in adapters that require
288
351
  # something beyond a simple insert (eg. Oracle).
352
+ # Most of adapters should implement `insert_fixtures_set` that leverages bulk SQL insert.
353
+ # We keep this method to provide fallback
354
+ # for databases like sqlite that do not support bulk inserts.
289
355
  def insert_fixture(fixture, table_name)
290
- fixture = fixture.stringify_keys
291
- columns = schema_cache.columns_hash(table_name)
292
-
293
- key_list = []
294
- value_list = fixture.map do |name, value|
295
- if column = columns[name]
296
- key_list << quote_column_name(name)
297
- quote(value, column)
298
- else
299
- raise Fixture::FixtureError, %(table "#{table_name}" has no column named #{name.inspect}.)
356
+ execute(build_fixture_sql(Array.wrap(fixture), table_name), "Fixture Insert")
357
+ end
358
+
359
+ def insert_fixtures_set(fixture_set, tables_to_delete = [])
360
+ fixture_inserts = build_fixture_statements(fixture_set)
361
+ table_deletes = tables_to_delete.map { |table| "DELETE FROM #{quote_table_name(table)}" }
362
+ total_sql = Array(combine_multi_statements(table_deletes + fixture_inserts))
363
+
364
+ with_multi_statements do
365
+ disable_referential_integrity do
366
+ transaction(requires_new: true) do
367
+ total_sql.each do |sql|
368
+ execute_batch(sql, "Fixtures Load")
369
+ end
370
+ end
300
371
  end
301
372
  end
302
-
303
- execute "INSERT INTO #{quote_table_name(table_name)} (#{key_list.join(', ')}) VALUES (#{value_list.join(', ')})", 'Fixture Insert'
304
373
  end
305
374
 
306
- def empty_insert_statement_value
375
+ def empty_insert_statement_value(primary_key = nil)
307
376
  "DEFAULT VALUES"
308
377
  end
309
378
 
310
379
  # Sanitizes the given LIMIT parameter in order to prevent SQL injection.
311
380
  #
312
381
  # The +limit+ may be anything that can evaluate to a string via #to_s. It
313
- # should look like an integer, or a comma-delimited list of integers, or
314
- # an Arel SQL literal.
382
+ # should look like an integer, or an Arel SQL literal.
315
383
  #
316
384
  # Returns Integer and Arel::Nodes::SqlLiteral limits as is.
317
- # Returns the sanitized limit parameter, either as an integer, or as a
318
- # string which contains a comma-delimited list of integers.
319
385
  def sanitize_limit(limit)
320
386
  if limit.is_a?(Integer) || limit.is_a?(Arel::Nodes::SqlLiteral)
321
387
  limit
322
- elsif limit.to_s.include?(',')
323
- Arel.sql limit.to_s.split(',').map{ |i| Integer(i) }.join(',')
324
388
  else
325
389
  Integer(limit)
326
390
  end
327
391
  end
328
392
 
329
- # The default strategy for an UPDATE with joins is to use a subquery. This doesn't work
330
- # on MySQL (even when aliasing the tables), but MySQL allows using JOIN directly in
331
- # an UPDATE statement, so in the MySQL adapters we redefine this to do that.
332
- def join_to_update(update, select) #:nodoc:
333
- key = update.key
334
- subselect = subquery_for(key, select)
335
-
336
- update.where key.in(subselect)
393
+ # Fixture value is quoted by Arel, however scalar values
394
+ # are not quotable. In this case we want to convert
395
+ # the column value to YAML.
396
+ def with_yaml_fallback(value) # :nodoc:
397
+ if value.is_a?(Hash) || value.is_a?(Array)
398
+ YAML.dump(value)
399
+ else
400
+ value
401
+ end
337
402
  end
338
403
 
339
- def join_to_delete(delete, select, key) #:nodoc:
340
- subselect = subquery_for(key, select)
404
+ private
405
+ def execute_batch(sql, name = nil)
406
+ execute(sql, name)
407
+ end
341
408
 
342
- delete.where key.in(subselect)
343
- end
409
+ DEFAULT_INSERT_VALUE = Arel.sql("DEFAULT").freeze
410
+ private_constant :DEFAULT_INSERT_VALUE
411
+
412
+ def default_insert_value(column)
413
+ DEFAULT_INSERT_VALUE
414
+ end
344
415
 
345
- protected
416
+ def build_fixture_sql(fixtures, table_name)
417
+ columns = schema_cache.columns_hash(table_name)
418
+
419
+ values_list = fixtures.map do |fixture|
420
+ fixture = fixture.stringify_keys
421
+
422
+ unknown_columns = fixture.keys - columns.keys
423
+ if unknown_columns.any?
424
+ raise Fixture::FixtureError, %(table "#{table_name}" has no columns named #{unknown_columns.map(&:inspect).join(', ')}.)
425
+ end
426
+
427
+ columns.map do |name, column|
428
+ if fixture.key?(name)
429
+ type = lookup_cast_type_from_column(column)
430
+ with_yaml_fallback(type.serialize(fixture[name]))
431
+ else
432
+ default_insert_value(column)
433
+ end
434
+ end
435
+ end
346
436
 
347
- # Returns a subquery for the given key using the join information.
348
- def subquery_for(key, select)
349
- subselect = select.clone
350
- subselect.projections = [key]
351
- subselect
437
+ table = Arel::Table.new(table_name)
438
+ manager = Arel::InsertManager.new
439
+ manager.into(table)
440
+
441
+ if values_list.size == 1
442
+ values = values_list.shift
443
+ new_values = []
444
+ columns.each_key.with_index { |column, i|
445
+ unless values[i].equal?(DEFAULT_INSERT_VALUE)
446
+ new_values << values[i]
447
+ manager.columns << table[column]
448
+ end
449
+ }
450
+ values_list << new_values
451
+ else
452
+ columns.each_key { |column| manager.columns << table[column] }
453
+ end
454
+
455
+ manager.values = manager.create_values_list(values_list)
456
+ manager.to_sql
352
457
  end
353
458
 
354
- # Returns an ActiveRecord::Result instance.
355
- def select(sql, name = nil, binds = [])
356
- exec_query(sql, name, binds)
459
+ def build_fixture_statements(fixture_set)
460
+ fixture_set.map do |table_name, fixtures|
461
+ next if fixtures.empty?
462
+ build_fixture_sql(fixtures, table_name)
463
+ end.compact
357
464
  end
358
465
 
466
+ def build_truncate_statements(*table_names)
467
+ truncate_tables = table_names.map do |table_name|
468
+ "TRUNCATE TABLE #{quote_table_name(table_name)}"
469
+ end
470
+ combine_multi_statements(truncate_tables)
471
+ end
359
472
 
360
- # Returns the last auto-generated ID from the affected table.
361
- def insert_sql(sql, name = nil, pk = nil, id_value = nil, sequence_name = nil)
362
- execute(sql, name)
363
- id_value
473
+ def with_multi_statements
474
+ yield
364
475
  end
365
476
 
366
- # Executes the update statement and returns the number of rows affected.
367
- def update_sql(sql, name = nil)
368
- execute(sql, name)
477
+ def combine_multi_statements(total_sql)
478
+ total_sql.join(";\n")
479
+ end
480
+
481
+ # Returns an ActiveRecord::Result instance.
482
+ def select(sql, name = nil, binds = [])
483
+ exec_query(sql, name, binds, prepare: false)
369
484
  end
370
485
 
371
- # Executes the delete statement and returns the number of rows affected.
372
- def delete_sql(sql, name = nil)
373
- update_sql(sql, name)
486
+ def select_prepared(sql, name = nil, binds = [])
487
+ exec_query(sql, name, binds, prepare: true)
374
488
  end
375
489
 
376
- def sql_for_insert(sql, pk, id_value, sequence_name, binds)
490
+ def sql_for_insert(sql, pk, binds)
377
491
  [sql, binds]
378
492
  end
379
493
 
380
494
  def last_inserted_id(result)
381
- row = result.rows.first
495
+ single_value_from_rows(result.rows)
496
+ end
497
+
498
+ def single_value_from_rows(rows)
499
+ row = rows.first
382
500
  row && row.first
383
501
  end
384
502
 
385
- def binds_from_relation(relation, binds)
386
- if relation.is_a?(Relation) && binds.empty?
387
- relation, binds = relation.arel, relation.bind_values
503
+ def arel_from_relation(relation)
504
+ if relation.is_a?(Relation)
505
+ relation.arel
506
+ else
507
+ relation
388
508
  end
389
- [relation, binds]
390
509
  end
391
510
  end
392
511
  end