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,28 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module Encryption
5
+ # A +NullEncryptor+ that will raise an error when trying to encrypt data
6
+ #
7
+ # This is useful when you want to reveal ciphertexts for debugging purposes
8
+ # and you want to make sure you won't overwrite any encryptable attribute with
9
+ # the wrong content.
10
+ class ReadOnlyNullEncryptor
11
+ def encrypt(clean_text, key_provider: nil, cipher_options: {})
12
+ raise Errors::Encryption, "This encryptor is read-only"
13
+ end
14
+
15
+ def decrypt(encrypted_text, key_provider: nil, cipher_options: {})
16
+ encrypted_text
17
+ end
18
+
19
+ def encrypted?(text)
20
+ false
21
+ end
22
+
23
+ def binary?
24
+ false
25
+ end
26
+ end
27
+ end
28
+ end
@@ -0,0 +1,107 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module Encryption
5
+ # A container of attribute encryption options.
6
+ #
7
+ # It validates and serves attribute encryption options.
8
+ #
9
+ # See EncryptedAttributeType, Context
10
+ class Scheme
11
+ attr_accessor :previous_schemes
12
+
13
+ def initialize(key_provider: nil, key: nil, deterministic: nil, support_unencrypted_data: nil, downcase: nil, ignore_case: nil,
14
+ previous_schemes: nil, compress: true, compressor: nil, **context_properties)
15
+ # Initializing all attributes to +nil+ as we want to allow a "not set" semantics so that we
16
+ # can merge schemes without overriding values with defaults. See +#merge+
17
+
18
+ @key_provider_param = key_provider
19
+ @key = key
20
+ @deterministic = deterministic
21
+ @support_unencrypted_data = support_unencrypted_data
22
+ @downcase = downcase || ignore_case
23
+ @ignore_case = ignore_case
24
+ @previous_schemes_param = previous_schemes
25
+ @previous_schemes = Array.wrap(previous_schemes)
26
+ @context_properties = context_properties
27
+ @compress = compress
28
+ @compressor = compressor
29
+
30
+ validate_config!
31
+
32
+ @context_properties[:encryptor] = Encryptor.new(compress: @compress) unless @compress
33
+ @context_properties[:encryptor] = Encryptor.new(compressor: compressor) if compressor
34
+ end
35
+
36
+ def ignore_case?
37
+ @ignore_case
38
+ end
39
+
40
+ def downcase?
41
+ @downcase
42
+ end
43
+
44
+ def deterministic?
45
+ !!@deterministic
46
+ end
47
+
48
+ def support_unencrypted_data?
49
+ @support_unencrypted_data.nil? ? ActiveRecord::Encryption.config.support_unencrypted_data : @support_unencrypted_data
50
+ end
51
+
52
+ def fixed?
53
+ # by default deterministic encryption is fixed
54
+ @fixed ||= @deterministic && (!@deterministic.is_a?(Hash) || @deterministic[:fixed])
55
+ end
56
+
57
+ def key_provider
58
+ @key_provider_param || key_provider_from_key || deterministic_key_provider || default_key_provider
59
+ end
60
+
61
+ def merge(other_scheme)
62
+ self.class.new(**to_h.merge(other_scheme.to_h))
63
+ end
64
+
65
+ def to_h
66
+ { key_provider: @key_provider_param, deterministic: @deterministic, downcase: @downcase, ignore_case: @ignore_case,
67
+ previous_schemes: @previous_schemes_param, **@context_properties }.compact
68
+ end
69
+
70
+ def with_context(&block)
71
+ if @context_properties.present?
72
+ ActiveRecord::Encryption.with_encryption_context(**@context_properties, &block)
73
+ else
74
+ block.call
75
+ end
76
+ end
77
+
78
+ def compatible_with?(other_scheme)
79
+ deterministic? == other_scheme.deterministic?
80
+ end
81
+
82
+ private
83
+ def validate_config!
84
+ raise Errors::Configuration, "ignore_case: can only be used with deterministic encryption" if @ignore_case && !@deterministic
85
+ raise Errors::Configuration, "key_provider: and key: can't be used simultaneously" if @key_provider_param && @key
86
+ raise Errors::Configuration, "compressor: can't be used with compress: false" if !@compress && @compressor
87
+ raise Errors::Configuration, "compressor: can't be used with encryptor" if @compressor && @context_properties[:encryptor]
88
+ end
89
+
90
+ def key_provider_from_key
91
+ @key_provider_from_key ||= if @key.present?
92
+ DerivedSecretKeyProvider.new(@key)
93
+ end
94
+ end
95
+
96
+ def deterministic_key_provider
97
+ @deterministic_key_provider ||= if @deterministic
98
+ DeterministicKeyProvider.new(ActiveRecord::Encryption.config.deterministic_key)
99
+ end
100
+ end
101
+
102
+ def default_key_provider
103
+ ActiveRecord::Encryption.key_provider
104
+ end
105
+ end
106
+ end
107
+ end
@@ -0,0 +1,58 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/module"
4
+ require "active_support/core_ext/array"
5
+
6
+ module ActiveRecord
7
+ module Encryption
8
+ extend ActiveSupport::Autoload
9
+
10
+ eager_autoload do
11
+ autoload :AutoFilteredParameters
12
+ autoload :Cipher
13
+ autoload :Config
14
+ autoload :Configurable
15
+ autoload :Context
16
+ autoload :Contexts
17
+ autoload :DerivedSecretKeyProvider
18
+ autoload :EncryptableRecord
19
+ autoload :EncryptedAttributeType
20
+ autoload :EncryptedFixtures
21
+ autoload :EncryptingOnlyEncryptor
22
+ autoload :DeterministicKeyProvider
23
+ autoload :Encryptor
24
+ autoload :EnvelopeEncryptionKeyProvider
25
+ autoload :Errors
26
+ autoload :ExtendedDeterministicQueries
27
+ autoload :ExtendedDeterministicUniquenessValidator
28
+ autoload :Key
29
+ autoload :KeyGenerator
30
+ autoload :KeyProvider
31
+ autoload :Message
32
+ autoload :MessageSerializer
33
+ autoload :NullEncryptor
34
+ autoload :Properties
35
+ autoload :ReadOnlyNullEncryptor
36
+ autoload :Scheme
37
+ end
38
+
39
+ class Cipher
40
+ extend ActiveSupport::Autoload
41
+
42
+ eager_autoload do
43
+ autoload :Aes256Gcm
44
+ end
45
+ end
46
+
47
+ include Configurable
48
+ include Contexts
49
+
50
+ def self.eager_load!
51
+ super
52
+
53
+ Cipher.eager_load!
54
+ end
55
+ end
56
+
57
+ ActiveSupport.run_load_hooks :active_record_encryption, Encryption
58
+ end
@@ -0,0 +1,424 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/hash/slice"
4
+ require "active_support/core_ext/object/deep_dup"
5
+
6
+ module ActiveRecord
7
+ # Declare an enum attribute where the values map to integers in the database,
8
+ # but can be queried by name. Example:
9
+ #
10
+ # class Conversation < ActiveRecord::Base
11
+ # enum :status, [ :active, :archived ]
12
+ # end
13
+ #
14
+ # # conversation.update! status: 0
15
+ # conversation.active!
16
+ # conversation.active? # => true
17
+ # conversation.status # => "active"
18
+ #
19
+ # # conversation.update! status: 1
20
+ # conversation.archived!
21
+ # conversation.archived? # => true
22
+ # conversation.status # => "archived"
23
+ #
24
+ # # conversation.status = 1
25
+ # conversation.status = "archived"
26
+ #
27
+ # conversation.status = nil
28
+ # conversation.status.nil? # => true
29
+ # conversation.status # => nil
30
+ #
31
+ # Scopes based on the allowed values of the enum field will be provided
32
+ # as well. With the above example:
33
+ #
34
+ # Conversation.active
35
+ # Conversation.not_active
36
+ # Conversation.archived
37
+ # Conversation.not_archived
38
+ #
39
+ # Of course, you can also query them directly if the scopes don't fit your
40
+ # needs:
41
+ #
42
+ # Conversation.where(status: [:active, :archived])
43
+ # Conversation.where.not(status: :active)
44
+ #
45
+ # Defining scopes can be disabled by setting +:scopes+ to +false+.
46
+ #
47
+ # class Conversation < ActiveRecord::Base
48
+ # enum :status, [ :active, :archived ], scopes: false
49
+ # end
50
+ #
51
+ # You can set the default enum value by setting +:default+, like:
52
+ #
53
+ # class Conversation < ActiveRecord::Base
54
+ # enum :status, [ :active, :archived ], default: :active
55
+ # end
56
+ #
57
+ # conversation = Conversation.new
58
+ # conversation.status # => "active"
59
+ #
60
+ # It's possible to explicitly map the relation between attribute and
61
+ # database integer with a hash:
62
+ #
63
+ # class Conversation < ActiveRecord::Base
64
+ # enum :status, active: 0, archived: 1
65
+ # end
66
+ #
67
+ # Finally it's also possible to use a string column to persist the enumerated value.
68
+ # Note that this will likely lead to slower database queries:
69
+ #
70
+ # class Conversation < ActiveRecord::Base
71
+ # enum :status, active: "active", archived: "archived"
72
+ # end
73
+ #
74
+ # Note that when an array is used, the implicit mapping from the values to database
75
+ # integers is derived from the order the values appear in the array. In the example,
76
+ # <tt>:active</tt> is mapped to +0+ as it's the first element, and <tt>:archived</tt>
77
+ # is mapped to +1+. In general, the +i+-th element is mapped to <tt>i-1</tt> in the
78
+ # database.
79
+ #
80
+ # Therefore, once a value is added to the enum array, its position in the array must
81
+ # be maintained, and new values should only be added to the end of the array. To
82
+ # remove unused values, the explicit hash syntax should be used.
83
+ #
84
+ # In rare circumstances you might need to access the mapping directly.
85
+ # The mappings are exposed through a class method with the pluralized attribute
86
+ # name, which return the mapping in a ActiveSupport::HashWithIndifferentAccess :
87
+ #
88
+ # Conversation.statuses[:active] # => 0
89
+ # Conversation.statuses["archived"] # => 1
90
+ #
91
+ # Use that class method when you need to know the ordinal value of an enum.
92
+ # For example, you can use that when manually building SQL strings:
93
+ #
94
+ # Conversation.where("status <> ?", Conversation.statuses[:archived])
95
+ #
96
+ # You can use the +:prefix+ or +:suffix+ options when you need to define
97
+ # multiple enums with same values. If the passed value is +true+, the methods
98
+ # are prefixed/suffixed with the name of the enum. It is also possible to
99
+ # supply a custom value:
100
+ #
101
+ # class Conversation < ActiveRecord::Base
102
+ # enum :status, [ :active, :archived ], suffix: true
103
+ # enum :comments_status, [ :active, :inactive ], prefix: :comments
104
+ # end
105
+ #
106
+ # With the above example, the bang and predicate methods along with the
107
+ # associated scopes are now prefixed and/or suffixed accordingly:
108
+ #
109
+ # conversation.active_status!
110
+ # conversation.archived_status? # => false
111
+ #
112
+ # conversation.comments_inactive!
113
+ # conversation.comments_active? # => false
114
+ #
115
+ # If you want to disable the auto-generated methods on the model, you can do
116
+ # so by setting the +:instance_methods+ option to false:
117
+ #
118
+ # class Conversation < ActiveRecord::Base
119
+ # enum :status, [ :active, :archived ], instance_methods: false
120
+ # end
121
+ #
122
+ # If you want the enum value to be validated before saving, use the option +:validate+:
123
+ #
124
+ # class Conversation < ActiveRecord::Base
125
+ # enum :status, [ :active, :archived ], validate: true
126
+ # end
127
+ #
128
+ # conversation = Conversation.new
129
+ #
130
+ # conversation.status = :unknown
131
+ # conversation.valid? # => false
132
+ #
133
+ # conversation.status = nil
134
+ # conversation.valid? # => false
135
+ #
136
+ # conversation.status = :active
137
+ # conversation.valid? # => true
138
+ #
139
+ # It is also possible to pass additional validation options:
140
+ #
141
+ # class Conversation < ActiveRecord::Base
142
+ # enum :status, [ :active, :archived ], validate: { allow_nil: true }
143
+ # end
144
+ #
145
+ # conversation = Conversation.new
146
+ #
147
+ # conversation.status = :unknown
148
+ # conversation.valid? # => false
149
+ #
150
+ # conversation.status = nil
151
+ # conversation.valid? # => true
152
+ #
153
+ # conversation.status = :active
154
+ # conversation.valid? # => true
155
+ #
156
+ # Otherwise +ArgumentError+ will raise:
157
+ #
158
+ # class Conversation < ActiveRecord::Base
159
+ # enum :status, [ :active, :archived ]
160
+ # end
161
+ #
162
+ # conversation = Conversation.new
163
+ #
164
+ # conversation.status = :unknown # 'unknown' is not a valid status (ArgumentError)
165
+ module Enum
166
+ def self.extended(base) # :nodoc:
167
+ base.class_attribute(:defined_enums, instance_writer: false, default: {})
168
+ end
169
+
170
+ class EnumType < Type::Value # :nodoc:
171
+ delegate :type, to: :subtype
172
+
173
+ def initialize(name, mapping, subtype, raise_on_invalid_values: true)
174
+ @name = name
175
+ @mapping = mapping
176
+ @subtype = subtype
177
+ @_raise_on_invalid_values = raise_on_invalid_values
178
+ end
179
+
180
+ def cast(value)
181
+ if mapping.has_key?(value)
182
+ value.to_s
183
+ elsif mapping.has_value?(value)
184
+ mapping.key(value)
185
+ else
186
+ value.presence
187
+ end
188
+ end
189
+
190
+ def deserialize(value)
191
+ mapping.key(subtype.deserialize(value))
192
+ end
193
+
194
+ def serialize(value)
195
+ subtype.serialize(mapping.fetch(value, value))
196
+ end
197
+
198
+ def serializable?(value, &block)
199
+ subtype.serializable?(mapping.fetch(value, value), &block)
200
+ end
201
+
202
+ def assert_valid_value(value)
203
+ return unless @_raise_on_invalid_values
204
+
205
+ unless value.blank? || mapping.has_key?(value) || mapping.has_value?(value)
206
+ raise ArgumentError, "'#{value}' is not a valid #{name}"
207
+ end
208
+ end
209
+
210
+ attr_reader :subtype
211
+
212
+ private
213
+ attr_reader :name, :mapping
214
+ end
215
+
216
+ def enum(name = nil, values = nil, **options)
217
+ if name
218
+ values, options = options, {} unless values
219
+ return _enum(name, values, **options)
220
+ end
221
+
222
+ definitions = options.slice!(:_prefix, :_suffix, :_scopes, :_default, :_instance_methods)
223
+ options.transform_keys! { |key| :"#{key[1..-1]}" }
224
+
225
+ definitions.each { |name, values| _enum(name, values, **options) }
226
+
227
+ ActiveRecord.deprecator.warn(<<~MSG)
228
+ Defining enums with keyword arguments is deprecated and will be removed
229
+ in Rails 8.0. Positional arguments should be used instead:
230
+
231
+ #{definitions.map { |name, values| "enum :#{name}, #{values}" }.join("\n")}
232
+ MSG
233
+ end
234
+
235
+ private
236
+ def inherited(base)
237
+ base.defined_enums = defined_enums.deep_dup
238
+ super
239
+ end
240
+
241
+ def _enum(name, values, prefix: nil, suffix: nil, scopes: true, instance_methods: true, validate: false, **options)
242
+ assert_valid_enum_definition_values(values)
243
+ assert_valid_enum_options(options)
244
+ # statuses = { }
245
+ enum_values = ActiveSupport::HashWithIndifferentAccess.new
246
+ name = name.to_s
247
+
248
+ # def self.statuses() statuses end
249
+ detect_enum_conflict!(name, name.pluralize, true)
250
+ singleton_class.define_method(name.pluralize) { enum_values }
251
+ defined_enums[name] = enum_values
252
+
253
+ detect_enum_conflict!(name, name)
254
+ detect_enum_conflict!(name, "#{name}=")
255
+
256
+ attribute(name, **options)
257
+
258
+ decorate_attributes([name]) do |_name, subtype|
259
+ if subtype == ActiveModel::Type.default_value
260
+ raise "Undeclared attribute type for enum '#{name}' in #{self.name}. Enums must be" \
261
+ " backed by a database column or declared with an explicit type" \
262
+ " via `attribute`."
263
+ end
264
+
265
+ subtype = subtype.subtype if EnumType === subtype
266
+ EnumType.new(name, enum_values, subtype, raise_on_invalid_values: !validate)
267
+ end
268
+
269
+ value_method_names = []
270
+ _enum_methods_module.module_eval do
271
+ prefix = if prefix
272
+ prefix == true ? "#{name}_" : "#{prefix}_"
273
+ end
274
+
275
+ suffix = if suffix
276
+ suffix == true ? "_#{name}" : "_#{suffix}"
277
+ end
278
+
279
+ pairs = values.respond_to?(:each_pair) ? values.each_pair : values.each_with_index
280
+ pairs.each do |label, value|
281
+ enum_values[label] = value
282
+ label = label.to_s
283
+
284
+ value_method_name = "#{prefix}#{label}#{suffix}"
285
+ value_method_names << value_method_name
286
+ define_enum_methods(name, value_method_name, value, scopes, instance_methods)
287
+
288
+ method_friendly_label = label.gsub(/[\W&&[:ascii:]]+/, "_")
289
+ value_method_alias = "#{prefix}#{method_friendly_label}#{suffix}"
290
+
291
+ if value_method_alias != value_method_name && !value_method_names.include?(value_method_alias)
292
+ value_method_names << value_method_alias
293
+ define_enum_methods(name, value_method_alias, value, scopes, instance_methods)
294
+ end
295
+ end
296
+ end
297
+ detect_negative_enum_conditions!(value_method_names) if scopes
298
+
299
+ if validate
300
+ validate = {} unless Hash === validate
301
+ validates_inclusion_of name, in: enum_values.keys, **validate
302
+ end
303
+
304
+ enum_values.freeze
305
+ end
306
+
307
+ class EnumMethods < Module # :nodoc:
308
+ def initialize(klass)
309
+ @klass = klass
310
+ end
311
+
312
+ private
313
+ attr_reader :klass
314
+
315
+ def define_enum_methods(name, value_method_name, value, scopes, instance_methods)
316
+ if instance_methods
317
+ # def active?() status_for_database == 0 end
318
+ klass.send(:detect_enum_conflict!, name, "#{value_method_name}?")
319
+ define_method("#{value_method_name}?") { public_send(:"#{name}_for_database") == value }
320
+
321
+ # def active!() update!(status: 0) end
322
+ klass.send(:detect_enum_conflict!, name, "#{value_method_name}!")
323
+ define_method("#{value_method_name}!") { update!(name => value) }
324
+ end
325
+
326
+ if scopes
327
+ # scope :active, -> { where(status: 0) }
328
+ klass.send(:detect_enum_conflict!, name, value_method_name, true)
329
+ klass.scope value_method_name, -> { where(name => value) }
330
+
331
+ # scope :not_active, -> { where.not(status: 0) }
332
+ klass.send(:detect_enum_conflict!, name, "not_#{value_method_name}", true)
333
+ klass.scope "not_#{value_method_name}", -> { where.not(name => value) }
334
+ end
335
+ end
336
+ end
337
+ private_constant :EnumMethods
338
+
339
+ def _enum_methods_module
340
+ @_enum_methods_module ||= begin
341
+ mod = EnumMethods.new(self)
342
+ include mod
343
+ mod
344
+ end
345
+ end
346
+
347
+ def assert_valid_enum_definition_values(values)
348
+ case values
349
+ when Hash
350
+ if values.empty?
351
+ raise ArgumentError, "Enum values #{values} must not be empty."
352
+ end
353
+
354
+ if values.keys.any?(&:blank?)
355
+ raise ArgumentError, "Enum values #{values} must not contain a blank name."
356
+ end
357
+ when Array
358
+ if values.empty?
359
+ raise ArgumentError, "Enum values #{values} must not be empty."
360
+ end
361
+
362
+ unless values.all?(Symbol) || values.all?(String)
363
+ raise ArgumentError, "Enum values #{values} must only contain symbols or strings."
364
+ end
365
+
366
+ if values.any?(&:blank?)
367
+ raise ArgumentError, "Enum values #{values} must not contain a blank name."
368
+ end
369
+ else
370
+ raise ArgumentError, "Enum values #{values} must be either a non-empty hash or an array."
371
+ end
372
+ end
373
+
374
+ def assert_valid_enum_options(options)
375
+ invalid_keys = options.keys & %i[_prefix _suffix _scopes _default _instance_methods]
376
+ unless invalid_keys.empty?
377
+ raise ArgumentError, "invalid option(s): #{invalid_keys.map(&:inspect).join(", ")}. Valid options are: :prefix, :suffix, :scopes, :default, :instance_methods, and :validate."
378
+ end
379
+ end
380
+
381
+ ENUM_CONFLICT_MESSAGE = \
382
+ "You tried to define an enum named \"%{enum}\" on the model \"%{klass}\", but " \
383
+ "this will generate a %{type} method \"%{method}\", which is already defined " \
384
+ "by %{source}."
385
+ private_constant :ENUM_CONFLICT_MESSAGE
386
+
387
+ def detect_enum_conflict!(enum_name, method_name, klass_method = false)
388
+ if klass_method && dangerous_class_method?(method_name)
389
+ raise_conflict_error(enum_name, method_name, type: "class")
390
+ elsif klass_method && method_defined_within?(method_name, Relation)
391
+ raise_conflict_error(enum_name, method_name, type: "class", source: Relation.name)
392
+ elsif klass_method && method_name.to_sym == :id
393
+ raise_conflict_error(enum_name, method_name)
394
+ elsif !klass_method && dangerous_attribute_method?(method_name)
395
+ raise_conflict_error(enum_name, method_name)
396
+ elsif !klass_method && method_defined_within?(method_name, _enum_methods_module, Module)
397
+ raise_conflict_error(enum_name, method_name, source: "another enum")
398
+ end
399
+ end
400
+
401
+ def raise_conflict_error(enum_name, method_name, type: "instance", source: "Active Record")
402
+ raise ArgumentError, ENUM_CONFLICT_MESSAGE % {
403
+ enum: enum_name,
404
+ klass: name,
405
+ type: type,
406
+ method: method_name,
407
+ source: source
408
+ }
409
+ end
410
+
411
+ def detect_negative_enum_conditions!(method_names)
412
+ return unless logger
413
+
414
+ method_names.select { |m| m.start_with?("not_") }.each do |potential_not|
415
+ inverted_form = potential_not.sub("not_", "")
416
+ if method_names.include?(inverted_form)
417
+ logger.warn "Enum element '#{potential_not}' in #{self.name} uses the prefix 'not_'." \
418
+ " This has caused a conflict with auto generated negative scopes." \
419
+ " Avoid using enum elements starting with 'not' where the positive form is also an element."
420
+ end
421
+ end
422
+ end
423
+ end
424
+ end