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,196 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_record/connection_adapters/abstract_mysql_adapter"
4
+ require "active_record/connection_adapters/mysql2/database_statements"
5
+
6
+ gem "mysql2", "~> 0.5"
7
+ require "mysql2"
8
+
9
+ module ActiveRecord
10
+ module ConnectionAdapters
11
+ # = Active Record MySQL2 Adapter
12
+ class Mysql2Adapter < AbstractMysqlAdapter
13
+ ER_BAD_DB_ERROR = 1049
14
+ ER_DBACCESS_DENIED_ERROR = 1044
15
+ ER_ACCESS_DENIED_ERROR = 1045
16
+ ER_CONN_HOST_ERROR = 2003
17
+ ER_UNKNOWN_HOST_ERROR = 2005
18
+
19
+ ADAPTER_NAME = "Mysql2"
20
+
21
+ include Mysql2::DatabaseStatements
22
+
23
+ class << self
24
+ def new_client(config)
25
+ ::Mysql2::Client.new(config)
26
+ rescue ::Mysql2::Error => error
27
+ case error.error_number
28
+ when ER_BAD_DB_ERROR
29
+ raise ActiveRecord::NoDatabaseError.db_error(config[:database])
30
+ when ER_DBACCESS_DENIED_ERROR, ER_ACCESS_DENIED_ERROR
31
+ raise ActiveRecord::DatabaseConnectionError.username_error(config[:username])
32
+ when ER_CONN_HOST_ERROR, ER_UNKNOWN_HOST_ERROR
33
+ raise ActiveRecord::DatabaseConnectionError.hostname_error(config[:host])
34
+ else
35
+ raise ActiveRecord::ConnectionNotEstablished, error.message
36
+ end
37
+ end
38
+
39
+ private
40
+ def initialize_type_map(m)
41
+ super
42
+
43
+ m.register_type(%r(char)i) do |sql_type|
44
+ limit = extract_limit(sql_type)
45
+ Type.lookup(:string, adapter: :mysql2, limit: limit)
46
+ end
47
+
48
+ m.register_type %r(^enum)i, Type.lookup(:string, adapter: :mysql2)
49
+ m.register_type %r(^set)i, Type.lookup(:string, adapter: :mysql2)
50
+ end
51
+ end
52
+
53
+ TYPE_MAP = Type::TypeMap.new.tap { |m| initialize_type_map(m) }
54
+
55
+ def initialize(...)
56
+ super
57
+
58
+ @affected_rows_before_warnings = nil
59
+ @config[:flags] ||= 0
60
+
61
+ if @config[:flags].kind_of? Array
62
+ @config[:flags].push "FOUND_ROWS"
63
+ else
64
+ @config[:flags] |= ::Mysql2::Client::FOUND_ROWS
65
+ end
66
+
67
+ @connection_parameters ||= @config
68
+ end
69
+
70
+ def supports_json?
71
+ !mariadb? && database_version >= "5.7.8"
72
+ end
73
+
74
+ def supports_comments?
75
+ true
76
+ end
77
+
78
+ def supports_comments_in_create?
79
+ true
80
+ end
81
+
82
+ def supports_savepoints?
83
+ true
84
+ end
85
+
86
+ def savepoint_errors_invalidate_transactions?
87
+ true
88
+ end
89
+
90
+ def supports_lazy_transactions?
91
+ true
92
+ end
93
+
94
+ # HELPER METHODS ===========================================
95
+
96
+ def error_number(exception)
97
+ exception.error_number if exception.respond_to?(:error_number)
98
+ end
99
+
100
+ #--
101
+ # CONNECTION MANAGEMENT ====================================
102
+ #++
103
+
104
+ def connected?
105
+ !(@raw_connection.nil? || @raw_connection.closed?)
106
+ end
107
+
108
+ def active?
109
+ connected? && @lock.synchronize { @raw_connection&.ping } || false
110
+ end
111
+
112
+ alias :reset! :reconnect!
113
+
114
+ # Disconnects from the database if already connected.
115
+ # Otherwise, this method does nothing.
116
+ def disconnect!
117
+ @lock.synchronize do
118
+ super
119
+ @raw_connection&.close
120
+ @raw_connection = nil
121
+ end
122
+ end
123
+
124
+ def discard! # :nodoc:
125
+ @lock.synchronize do
126
+ super
127
+ @raw_connection&.automatic_close = false
128
+ @raw_connection = nil
129
+ end
130
+ end
131
+
132
+ private
133
+ def text_type?(type)
134
+ TYPE_MAP.lookup(type).is_a?(Type::String) || TYPE_MAP.lookup(type).is_a?(Type::Text)
135
+ end
136
+
137
+ def connect
138
+ @raw_connection = self.class.new_client(@connection_parameters)
139
+ rescue ConnectionNotEstablished => ex
140
+ raise ex.set_pool(@pool)
141
+ end
142
+
143
+ def reconnect
144
+ @lock.synchronize do
145
+ @raw_connection&.close
146
+ @raw_connection = nil
147
+ connect
148
+ end
149
+ end
150
+
151
+ def configure_connection
152
+ @raw_connection.query_options[:as] = :array
153
+ @raw_connection.query_options[:database_timezone] = default_timezone
154
+ super
155
+ end
156
+
157
+ def full_version
158
+ database_version.full_version_string
159
+ end
160
+
161
+ def get_full_version
162
+ any_raw_connection.server_info[:version]
163
+ end
164
+
165
+ def translate_exception(exception, message:, sql:, binds:)
166
+ if exception.is_a?(::Mysql2::Error::TimeoutError) && !exception.error_number
167
+ ActiveRecord::AdapterTimeout.new(message, sql: sql, binds: binds, connection_pool: @pool)
168
+ elsif exception.is_a?(::Mysql2::Error::ConnectionError)
169
+ if exception.message.match?(/MySQL client is not connected/i)
170
+ ActiveRecord::ConnectionNotEstablished.new(exception, connection_pool: @pool)
171
+ else
172
+ ActiveRecord::ConnectionFailed.new(message, sql: sql, binds: binds, connection_pool: @pool)
173
+ end
174
+ else
175
+ super
176
+ end
177
+ end
178
+
179
+ def default_prepared_statements
180
+ false
181
+ end
182
+
183
+ ActiveRecord::Type.register(:immutable_string, adapter: :mysql2) do |_, **args|
184
+ Type::ImmutableString.new(true: "1", false: "0", **args)
185
+ end
186
+
187
+ ActiveRecord::Type.register(:string, adapter: :mysql2) do |_, **args|
188
+ Type::String.new(true: "1", false: "0", **args)
189
+ end
190
+
191
+ ActiveRecord::Type.register(:unsigned_integer, Type::UnsignedInteger, adapter: :mysql2)
192
+ end
193
+
194
+ ActiveSupport.run_load_hooks(:active_record_mysql2adapter, Mysql2Adapter)
195
+ end
196
+ end
@@ -0,0 +1,83 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module ConnectionAdapters
5
+ class PoolConfig # :nodoc:
6
+ include MonitorMixin
7
+
8
+ attr_reader :db_config, :role, :shard
9
+ attr_writer :schema_reflection, :server_version
10
+ attr_accessor :connection_class
11
+
12
+ def schema_reflection
13
+ @schema_reflection ||= SchemaReflection.new(db_config.lazy_schema_cache_path)
14
+ end
15
+
16
+ INSTANCES = ObjectSpace::WeakMap.new
17
+ private_constant :INSTANCES
18
+
19
+ class << self
20
+ def discard_pools!
21
+ INSTANCES.each_key(&:discard_pool!)
22
+ end
23
+
24
+ def disconnect_all!
25
+ INSTANCES.each_key { |c| c.disconnect!(automatic_reconnect: true) }
26
+ end
27
+ end
28
+
29
+ def initialize(connection_class, db_config, role, shard)
30
+ super()
31
+ @server_version = nil
32
+ @connection_class = connection_class
33
+ @db_config = db_config
34
+ @role = role
35
+ @shard = shard
36
+ @pool = nil
37
+ INSTANCES[self] = self
38
+ end
39
+
40
+ def server_version(connection)
41
+ @server_version || synchronize { @server_version ||= connection.get_database_version }
42
+ end
43
+
44
+ def connection_name
45
+ if connection_class.primary_class?
46
+ "ActiveRecord::Base"
47
+ else
48
+ connection_class.name
49
+ end
50
+ end
51
+
52
+ def disconnect!(automatic_reconnect: false)
53
+ return unless @pool
54
+
55
+ synchronize do
56
+ return unless @pool
57
+
58
+ @pool.automatic_reconnect = automatic_reconnect
59
+ @pool.disconnect!
60
+ end
61
+
62
+ nil
63
+ end
64
+
65
+ def pool
66
+ @pool || synchronize { @pool ||= ConnectionAdapters::ConnectionPool.new(self) }
67
+ end
68
+
69
+ def discard_pool!
70
+ return unless @pool
71
+
72
+ synchronize do
73
+ return unless @pool
74
+
75
+ @pool.discard!
76
+ @pool = nil
77
+ end
78
+ end
79
+ end
80
+ end
81
+ end
82
+
83
+ ActiveSupport::ForkTracker.after_fork { ActiveRecord::ConnectionAdapters::PoolConfig.discard_pools! }
@@ -0,0 +1,57 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module ConnectionAdapters
5
+ class PoolManager # :nodoc:
6
+ def initialize
7
+ @role_to_shard_mapping = Hash.new { |h, k| h[k] = {} }
8
+ end
9
+
10
+ def shard_names
11
+ @role_to_shard_mapping.values.flat_map { |shard_map| shard_map.keys }.uniq
12
+ end
13
+
14
+ def role_names
15
+ @role_to_shard_mapping.keys
16
+ end
17
+
18
+ def pool_configs(role = nil)
19
+ if role
20
+ @role_to_shard_mapping[role].values
21
+ else
22
+ @role_to_shard_mapping.flat_map { |_, shard_map| shard_map.values }
23
+ end
24
+ end
25
+
26
+ def each_pool_config(role = nil, &block)
27
+ if role
28
+ @role_to_shard_mapping[role].each_value(&block)
29
+ else
30
+ @role_to_shard_mapping.each_value do |shard_map|
31
+ shard_map.each_value(&block)
32
+ end
33
+ end
34
+ end
35
+
36
+ def remove_role(role)
37
+ @role_to_shard_mapping.delete(role)
38
+ end
39
+
40
+ def remove_pool_config(role, shard)
41
+ @role_to_shard_mapping[role].delete(shard)
42
+ end
43
+
44
+ def get_pool_config(role, shard)
45
+ @role_to_shard_mapping[role][shard]
46
+ end
47
+
48
+ def set_pool_config(role, shard, pool_config)
49
+ if pool_config
50
+ @role_to_shard_mapping[role][shard] = pool_config
51
+ else
52
+ raise ArgumentError, "The `pool_config` for the :#{role} role and :#{shard} shard was `nil`. Please check your configuration. If you want your writing role to be something other than `:writing` set `config.active_record.writing_role` in your application configuration. The same setting should be applied for the `reading_role` if applicable."
53
+ end
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,82 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module ConnectionAdapters
5
+ module PostgreSQL
6
+ class Column < ConnectionAdapters::Column # :nodoc:
7
+ delegate :oid, :fmod, to: :sql_type_metadata
8
+
9
+ def initialize(*, serial: nil, identity: nil, generated: nil, **)
10
+ super
11
+ @serial = serial
12
+ @identity = identity
13
+ @generated = generated
14
+ end
15
+
16
+ def identity?
17
+ @identity
18
+ end
19
+
20
+ def serial?
21
+ @serial
22
+ end
23
+
24
+ def auto_incremented_by_db?
25
+ serial? || identity?
26
+ end
27
+
28
+ def virtual?
29
+ # We assume every generated column is virtual, no matter the concrete type
30
+ @generated.present?
31
+ end
32
+
33
+ def has_default?
34
+ super && !virtual?
35
+ end
36
+
37
+ def array
38
+ sql_type_metadata.sql_type.end_with?("[]")
39
+ end
40
+ alias :array? :array
41
+
42
+ def enum?
43
+ type == :enum
44
+ end
45
+
46
+ def sql_type
47
+ super.delete_suffix("[]")
48
+ end
49
+
50
+ def init_with(coder)
51
+ @serial = coder["serial"]
52
+ @identity = coder["identity"]
53
+ @generated = coder["generated"]
54
+ super
55
+ end
56
+
57
+ def encode_with(coder)
58
+ coder["serial"] = @serial
59
+ coder["identity"] = @identity
60
+ coder["generated"] = @generated
61
+ super
62
+ end
63
+
64
+ def ==(other)
65
+ other.is_a?(Column) &&
66
+ super &&
67
+ identity? == other.identity? &&
68
+ serial? == other.serial?
69
+ end
70
+ alias :eql? :==
71
+
72
+ def hash
73
+ Column.hash ^
74
+ super.hash ^
75
+ identity?.hash ^
76
+ serial?.hash
77
+ end
78
+ end
79
+ end
80
+ PostgreSQLColumn = PostgreSQL::Column # :nodoc:
81
+ end
82
+ end
@@ -0,0 +1,231 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module ConnectionAdapters
5
+ module PostgreSQL
6
+ module DatabaseStatements
7
+ def explain(arel, binds = [], options = [])
8
+ sql = build_explain_clause(options) + " " + to_sql(arel, binds)
9
+ result = internal_exec_query(sql, "EXPLAIN", binds)
10
+ PostgreSQL::ExplainPrettyPrinter.new.pp(result)
11
+ end
12
+
13
+ # Queries the database and returns the results in an Array-like object
14
+ def query(sql, name = nil) # :nodoc:
15
+ result = internal_execute(sql, name)
16
+ result.map_types!(@type_map_for_results).values
17
+ end
18
+
19
+ READ_QUERY = ActiveRecord::ConnectionAdapters::AbstractAdapter.build_read_query_regexp(
20
+ :close, :declare, :fetch, :move, :set, :show
21
+ ) # :nodoc:
22
+ private_constant :READ_QUERY
23
+
24
+ def write_query?(sql) # :nodoc:
25
+ !READ_QUERY.match?(sql)
26
+ rescue ArgumentError # Invalid encoding
27
+ !READ_QUERY.match?(sql.b)
28
+ end
29
+
30
+ # Executes an SQL statement, returning a PG::Result object on success
31
+ # or raising a PG::Error exception otherwise.
32
+ #
33
+ # Setting +allow_retry+ to true causes the db to reconnect and retry
34
+ # executing the SQL statement in case of a connection-related exception.
35
+ # This option should only be enabled for known idempotent queries.
36
+ #
37
+ # Note: the PG::Result object is manually memory managed; if you don't
38
+ # need it specifically, you may want consider the <tt>exec_query</tt> wrapper.
39
+ def execute(...) # :nodoc:
40
+ super
41
+ ensure
42
+ @notice_receiver_sql_warnings = []
43
+ end
44
+
45
+ def exec_insert(sql, name = nil, binds = [], pk = nil, sequence_name = nil, returning: nil) # :nodoc:
46
+ if use_insert_returning? || pk == false
47
+ super
48
+ else
49
+ result = internal_exec_query(sql, name, binds)
50
+ unless sequence_name
51
+ table_ref = extract_table_ref_from_insert_sql(sql)
52
+ if table_ref
53
+ pk = primary_key(table_ref) if pk.nil?
54
+ pk = suppress_composite_primary_key(pk)
55
+ sequence_name = default_sequence_name(table_ref, pk)
56
+ end
57
+ return result unless sequence_name
58
+ end
59
+ last_insert_id_result(sequence_name)
60
+ end
61
+ end
62
+
63
+ # Begins a transaction.
64
+ def begin_db_transaction # :nodoc:
65
+ internal_execute("BEGIN", "TRANSACTION", allow_retry: true, materialize_transactions: false)
66
+ end
67
+
68
+ def begin_isolated_db_transaction(isolation) # :nodoc:
69
+ internal_execute("BEGIN ISOLATION LEVEL #{transaction_isolation_levels.fetch(isolation)}", "TRANSACTION", allow_retry: true, materialize_transactions: false)
70
+ end
71
+
72
+ # Commits a transaction.
73
+ def commit_db_transaction # :nodoc:
74
+ internal_execute("COMMIT", "TRANSACTION", allow_retry: false, materialize_transactions: true)
75
+ end
76
+
77
+ # Aborts a transaction.
78
+ def exec_rollback_db_transaction # :nodoc:
79
+ cancel_any_running_query
80
+ internal_execute("ROLLBACK", "TRANSACTION", allow_retry: false, materialize_transactions: true)
81
+ end
82
+
83
+ def exec_restart_db_transaction # :nodoc:
84
+ cancel_any_running_query
85
+ internal_execute("ROLLBACK AND CHAIN", "TRANSACTION", allow_retry: false, materialize_transactions: true)
86
+ end
87
+
88
+ # From https://www.postgresql.org/docs/current/functions-datetime.html#FUNCTIONS-DATETIME-CURRENT
89
+ HIGH_PRECISION_CURRENT_TIMESTAMP = Arel.sql("CURRENT_TIMESTAMP", retryable: true).freeze # :nodoc:
90
+ private_constant :HIGH_PRECISION_CURRENT_TIMESTAMP
91
+
92
+ def high_precision_current_timestamp
93
+ HIGH_PRECISION_CURRENT_TIMESTAMP
94
+ end
95
+
96
+ def build_explain_clause(options = [])
97
+ return "EXPLAIN" if options.empty?
98
+
99
+ "EXPLAIN (#{options.join(", ").upcase})"
100
+ end
101
+
102
+ # Set when constraints will be checked for the current transaction.
103
+ #
104
+ # Not passing any specific constraint names will set the value for all deferrable constraints.
105
+ #
106
+ # [<tt>deferred</tt>]
107
+ # Valid values are +:deferred+ or +:immediate+.
108
+ #
109
+ # See https://www.postgresql.org/docs/current/sql-set-constraints.html
110
+ def set_constraints(deferred, *constraints)
111
+ unless %i[deferred immediate].include?(deferred)
112
+ raise ArgumentError, "deferred must be :deferred or :immediate"
113
+ end
114
+
115
+ constraints = if constraints.empty?
116
+ "ALL"
117
+ else
118
+ constraints.map { |c| quote_table_name(c) }.join(", ")
119
+ end
120
+ execute("SET CONSTRAINTS #{constraints} #{deferred.to_s.upcase}")
121
+ end
122
+
123
+ private
124
+ IDLE_TRANSACTION_STATUSES = [PG::PQTRANS_IDLE, PG::PQTRANS_INTRANS, PG::PQTRANS_INERROR]
125
+ private_constant :IDLE_TRANSACTION_STATUSES
126
+
127
+ def cancel_any_running_query
128
+ return if @raw_connection.nil? || IDLE_TRANSACTION_STATUSES.include?(@raw_connection.transaction_status)
129
+
130
+ @raw_connection.cancel
131
+ @raw_connection.block
132
+ rescue PG::Error
133
+ end
134
+
135
+ def perform_query(raw_connection, sql, binds, type_casted_binds, prepare:, notification_payload:, batch: false)
136
+ update_typemap_for_default_timezone
137
+ result = if prepare
138
+ begin
139
+ stmt_key = prepare_statement(sql, binds, raw_connection)
140
+ notification_payload[:statement_name] = stmt_key
141
+ raw_connection.exec_prepared(stmt_key, type_casted_binds)
142
+ rescue PG::FeatureNotSupported => error
143
+ if is_cached_plan_failure?(error)
144
+ # Nothing we can do if we are in a transaction because all commands
145
+ # will raise InFailedSQLTransaction
146
+ if in_transaction?
147
+ raise PreparedStatementCacheExpired.new(error.message, connection_pool: @pool)
148
+ else
149
+ @lock.synchronize do
150
+ # outside of transactions we can simply flush this query and retry
151
+ @statements.delete sql_key(sql)
152
+ end
153
+ retry
154
+ end
155
+ end
156
+
157
+ raise
158
+ end
159
+ elsif binds.nil? || binds.empty?
160
+ raw_connection.async_exec(sql)
161
+ else
162
+ raw_connection.exec_params(sql, type_casted_binds)
163
+ end
164
+
165
+ verified!
166
+ handle_warnings(result)
167
+ notification_payload[:row_count] = result.count
168
+ result
169
+ end
170
+
171
+ def cast_result(result)
172
+ if result.fields.empty?
173
+ result.clear
174
+ return ActiveRecord::Result.empty
175
+ end
176
+
177
+ types = {}
178
+ fields = result.fields
179
+ fields.each_with_index do |fname, i|
180
+ ftype = result.ftype i
181
+ fmod = result.fmod i
182
+ types[fname] = types[i] = get_oid_type(ftype, fmod, fname)
183
+ end
184
+ ar_result = ActiveRecord::Result.new(fields, result.values, types.freeze)
185
+ result.clear
186
+ ar_result
187
+ end
188
+
189
+ def affected_rows(result)
190
+ affected_rows = result.cmd_tuples
191
+ result.clear
192
+ affected_rows
193
+ end
194
+
195
+ def execute_batch(statements, name = nil, **kwargs)
196
+ raw_execute(combine_multi_statements(statements), name, batch: true, **kwargs)
197
+ end
198
+
199
+ def build_truncate_statements(table_names)
200
+ ["TRUNCATE TABLE #{table_names.map(&method(:quote_table_name)).join(", ")}"]
201
+ end
202
+
203
+ # Returns the current ID of a table's sequence.
204
+ def last_insert_id_result(sequence_name)
205
+ internal_exec_query("SELECT currval(#{quote(sequence_name)})", "SQL")
206
+ end
207
+
208
+ def returning_column_values(result)
209
+ result.rows.first
210
+ end
211
+
212
+ def suppress_composite_primary_key(pk)
213
+ pk unless pk.is_a?(Array)
214
+ end
215
+
216
+ def handle_warnings(sql)
217
+ @notice_receiver_sql_warnings.each do |warning|
218
+ next if warning_ignored?(warning)
219
+
220
+ warning.sql = sql
221
+ ActiveRecord.db_warnings_action.call(warning)
222
+ end
223
+ end
224
+
225
+ def warning_ignored?(warning)
226
+ ["WARNING", "ERROR", "FATAL", "PANIC"].exclude?(warning.level) || super
227
+ end
228
+ end
229
+ end
230
+ end
231
+ end
@@ -0,0 +1,44 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module ConnectionAdapters
5
+ module PostgreSQL
6
+ class ExplainPrettyPrinter # :nodoc:
7
+ # Pretty prints the result of an EXPLAIN in a way that resembles the output of the
8
+ # PostgreSQL shell:
9
+ #
10
+ # QUERY PLAN
11
+ # ------------------------------------------------------------------------------
12
+ # Nested Loop Left Join (cost=0.00..37.24 rows=8 width=0)
13
+ # Join Filter: (posts.user_id = users.id)
14
+ # -> Index Scan using users_pkey on users (cost=0.00..8.27 rows=1 width=4)
15
+ # Index Cond: (id = 1)
16
+ # -> Seq Scan on posts (cost=0.00..28.88 rows=8 width=4)
17
+ # Filter: (posts.user_id = 1)
18
+ # (6 rows)
19
+ #
20
+ def pp(result)
21
+ header = result.columns.first
22
+ lines = result.rows.map(&:first)
23
+
24
+ # We add 2 because there's one char of padding at both sides, note
25
+ # the extra hyphens in the example above.
26
+ width = [header, *lines].map(&:length).max + 2
27
+
28
+ pp = []
29
+
30
+ pp << header.center(width).rstrip
31
+ pp << "-" * width
32
+
33
+ pp += lines.map { |line| " #{line}" }
34
+
35
+ nrows = result.rows.length
36
+ rows_label = nrows == 1 ? "row" : "rows"
37
+ pp << "(#{nrows} #{rows_label})"
38
+
39
+ pp.join("\n") + "\n"
40
+ end
41
+ end
42
+ end
43
+ end
44
+ end