activerecord 5.2.8 → 7.0.2
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.
Potentially problematic release.
This version of activerecord might be problematic. Click here for more details.
- checksums.yaml +4 -4
 - data/CHANGELOG.md +1393 -587
 - data/MIT-LICENSE +3 -1
 - data/README.rdoc +7 -5
 - data/examples/performance.rb +1 -1
 - data/lib/active_record/aggregations.rb +10 -9
 - data/lib/active_record/association_relation.rb +22 -12
 - data/lib/active_record/associations/alias_tracker.rb +19 -16
 - data/lib/active_record/associations/association.rb +122 -47
 - data/lib/active_record/associations/association_scope.rb +24 -24
 - data/lib/active_record/associations/belongs_to_association.rb +67 -49
 - data/lib/active_record/associations/belongs_to_polymorphic_association.rb +16 -7
 - data/lib/active_record/associations/builder/association.rb +52 -23
 - data/lib/active_record/associations/builder/belongs_to.rb +44 -61
 - data/lib/active_record/associations/builder/collection_association.rb +17 -19
 - data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +17 -41
 - data/lib/active_record/associations/builder/has_many.rb +10 -3
 - data/lib/active_record/associations/builder/has_one.rb +35 -3
 - data/lib/active_record/associations/builder/singular_association.rb +5 -3
 - data/lib/active_record/associations/collection_association.rb +59 -50
 - data/lib/active_record/associations/collection_proxy.rb +32 -23
 - data/lib/active_record/associations/disable_joins_association_scope.rb +59 -0
 - data/lib/active_record/associations/foreign_association.rb +20 -0
 - data/lib/active_record/associations/has_many_association.rb +27 -14
 - data/lib/active_record/associations/has_many_through_association.rb +26 -19
 - data/lib/active_record/associations/has_one_association.rb +52 -37
 - data/lib/active_record/associations/has_one_through_association.rb +6 -6
 - data/lib/active_record/associations/join_dependency/join_association.rb +44 -22
 - data/lib/active_record/associations/join_dependency/join_part.rb +5 -5
 - data/lib/active_record/associations/join_dependency.rb +97 -62
 - data/lib/active_record/associations/preloader/association.rb +220 -60
 - data/lib/active_record/associations/preloader/batch.rb +48 -0
 - data/lib/active_record/associations/preloader/branch.rb +147 -0
 - data/lib/active_record/associations/preloader/through_association.rb +85 -40
 - data/lib/active_record/associations/preloader.rb +44 -105
 - data/lib/active_record/associations/singular_association.rb +9 -17
 - data/lib/active_record/associations/through_association.rb +4 -4
 - data/lib/active_record/associations.rb +207 -66
 - data/lib/active_record/asynchronous_queries_tracker.rb +60 -0
 - data/lib/active_record/attribute_assignment.rb +17 -19
 - data/lib/active_record/attribute_methods/before_type_cast.rb +19 -8
 - data/lib/active_record/attribute_methods/dirty.rb +141 -47
 - data/lib/active_record/attribute_methods/primary_key.rb +22 -27
 - data/lib/active_record/attribute_methods/query.rb +6 -10
 - data/lib/active_record/attribute_methods/read.rb +15 -55
 - data/lib/active_record/attribute_methods/serialization.rb +77 -18
 - data/lib/active_record/attribute_methods/time_zone_conversion.rb +16 -18
 - data/lib/active_record/attribute_methods/write.rb +18 -37
 - data/lib/active_record/attribute_methods.rb +90 -153
 - data/lib/active_record/attributes.rb +38 -12
 - data/lib/active_record/autosave_association.rb +50 -50
 - data/lib/active_record/base.rb +23 -18
 - data/lib/active_record/callbacks.rb +159 -44
 - data/lib/active_record/coders/yaml_column.rb +12 -3
 - data/lib/active_record/connection_adapters/abstract/connection_handler.rb +292 -0
 - data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +209 -0
 - data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +76 -0
 - data/lib/active_record/connection_adapters/abstract/connection_pool.rb +92 -464
 - data/lib/active_record/connection_adapters/abstract/database_limits.rb +5 -51
 - data/lib/active_record/connection_adapters/abstract/database_statements.rb +209 -164
 - data/lib/active_record/connection_adapters/abstract/query_cache.rb +38 -22
 - data/lib/active_record/connection_adapters/abstract/quoting.rb +103 -82
 - data/lib/active_record/connection_adapters/abstract/savepoints.rb +3 -3
 - data/lib/active_record/connection_adapters/abstract/schema_creation.rb +140 -110
 - data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +236 -94
 - data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +16 -5
 - data/lib/active_record/connection_adapters/abstract/schema_statements.rb +456 -159
 - data/lib/active_record/connection_adapters/abstract/transaction.rb +169 -78
 - data/lib/active_record/connection_adapters/abstract_adapter.rb +367 -162
 - data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +311 -327
 - data/lib/active_record/connection_adapters/column.rb +33 -11
 - data/lib/active_record/connection_adapters/deduplicable.rb +29 -0
 - data/lib/active_record/connection_adapters/legacy_pool_manager.rb +35 -0
 - data/lib/active_record/connection_adapters/mysql/column.rb +1 -1
 - data/lib/active_record/connection_adapters/mysql/database_statements.rb +113 -45
 - data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +1 -2
 - data/lib/active_record/connection_adapters/mysql/quoting.rb +71 -5
 - data/lib/active_record/connection_adapters/mysql/schema_creation.rb +34 -10
 - data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +48 -32
 - data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +25 -8
 - data/lib/active_record/connection_adapters/mysql/schema_statements.rb +143 -19
 - data/lib/active_record/connection_adapters/mysql/type_metadata.rb +14 -9
 - data/lib/active_record/connection_adapters/mysql2_adapter.rb +63 -22
 - data/lib/active_record/connection_adapters/pool_config.rb +73 -0
 - data/lib/active_record/connection_adapters/pool_manager.rb +47 -0
 - data/lib/active_record/connection_adapters/postgresql/column.rb +53 -28
 - data/lib/active_record/connection_adapters/postgresql/database_statements.rb +56 -63
 - data/lib/active_record/connection_adapters/postgresql/oid/array.rb +1 -2
 - data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +1 -4
 - data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +3 -5
 - data/lib/active_record/connection_adapters/postgresql/oid/date.rb +10 -2
 - data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +15 -2
 - data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +0 -1
 - data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +54 -16
 - data/lib/active_record/connection_adapters/postgresql/oid/interval.rb +49 -0
 - data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +3 -4
 - data/lib/active_record/connection_adapters/postgresql/oid/macaddr.rb +25 -0
 - data/lib/active_record/connection_adapters/postgresql/oid/oid.rb +1 -1
 - data/lib/active_record/connection_adapters/postgresql/oid/point.rb +3 -4
 - data/lib/active_record/connection_adapters/postgresql/oid/range.rb +25 -7
 - data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +1 -1
 - data/lib/active_record/connection_adapters/postgresql/oid/timestamp.rb +15 -0
 - data/lib/active_record/connection_adapters/postgresql/oid/timestamp_with_time_zone.rb +30 -0
 - data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +26 -12
 - data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +15 -3
 - data/lib/active_record/connection_adapters/postgresql/oid.rb +4 -0
 - data/lib/active_record/connection_adapters/postgresql/quoting.rb +89 -52
 - data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +34 -2
 - data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +39 -4
 - data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +128 -91
 - data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +25 -1
 - data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +149 -113
 - data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +31 -26
 - data/lib/active_record/connection_adapters/postgresql/utils.rb +0 -1
 - data/lib/active_record/connection_adapters/postgresql_adapter.rb +386 -182
 - data/lib/active_record/connection_adapters/schema_cache.rb +161 -22
 - data/lib/active_record/connection_adapters/sql_type_metadata.rb +17 -6
 - data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +152 -0
 - data/lib/active_record/connection_adapters/sqlite3/quoting.rb +65 -18
 - data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +5 -1
 - data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +92 -26
 - data/lib/active_record/connection_adapters/sqlite3_adapter.rb +251 -204
 - data/lib/active_record/connection_adapters/statement_pool.rb +0 -1
 - data/lib/active_record/connection_adapters.rb +53 -0
 - data/lib/active_record/connection_handling.rb +292 -38
 - data/lib/active_record/core.rb +385 -158
 - data/lib/active_record/counter_cache.rb +8 -30
 - data/lib/active_record/database_configurations/connection_url_resolver.rb +100 -0
 - data/lib/active_record/database_configurations/database_config.rb +83 -0
 - data/lib/active_record/database_configurations/hash_config.rb +154 -0
 - data/lib/active_record/database_configurations/url_config.rb +53 -0
 - data/lib/active_record/database_configurations.rb +256 -0
 - data/lib/active_record/delegated_type.rb +250 -0
 - data/lib/active_record/destroy_association_async_job.rb +36 -0
 - data/lib/active_record/disable_joins_association_relation.rb +39 -0
 - data/lib/active_record/dynamic_matchers.rb +4 -5
 - data/lib/active_record/encryption/cipher/aes256_gcm.rb +98 -0
 - data/lib/active_record/encryption/cipher.rb +53 -0
 - data/lib/active_record/encryption/config.rb +44 -0
 - data/lib/active_record/encryption/configurable.rb +61 -0
 - data/lib/active_record/encryption/context.rb +35 -0
 - data/lib/active_record/encryption/contexts.rb +72 -0
 - data/lib/active_record/encryption/derived_secret_key_provider.rb +12 -0
 - data/lib/active_record/encryption/deterministic_key_provider.rb +14 -0
 - data/lib/active_record/encryption/encryptable_record.rb +208 -0
 - data/lib/active_record/encryption/encrypted_attribute_type.rb +140 -0
 - data/lib/active_record/encryption/encrypted_fixtures.rb +38 -0
 - data/lib/active_record/encryption/encrypting_only_encryptor.rb +12 -0
 - data/lib/active_record/encryption/encryptor.rb +155 -0
 - data/lib/active_record/encryption/envelope_encryption_key_provider.rb +55 -0
 - data/lib/active_record/encryption/errors.rb +15 -0
 - data/lib/active_record/encryption/extended_deterministic_queries.rb +160 -0
 - data/lib/active_record/encryption/extended_deterministic_uniqueness_validator.rb +28 -0
 - data/lib/active_record/encryption/key.rb +28 -0
 - data/lib/active_record/encryption/key_generator.rb +42 -0
 - data/lib/active_record/encryption/key_provider.rb +46 -0
 - data/lib/active_record/encryption/message.rb +33 -0
 - data/lib/active_record/encryption/message_serializer.rb +90 -0
 - data/lib/active_record/encryption/null_encryptor.rb +21 -0
 - data/lib/active_record/encryption/properties.rb +76 -0
 - data/lib/active_record/encryption/read_only_null_encryptor.rb +24 -0
 - data/lib/active_record/encryption/scheme.rb +99 -0
 - data/lib/active_record/encryption.rb +55 -0
 - data/lib/active_record/enum.rb +130 -51
 - data/lib/active_record/errors.rb +129 -23
 - data/lib/active_record/explain.rb +10 -6
 - data/lib/active_record/explain_registry.rb +11 -6
 - data/lib/active_record/explain_subscriber.rb +1 -1
 - data/lib/active_record/fixture_set/file.rb +22 -15
 - data/lib/active_record/fixture_set/model_metadata.rb +32 -0
 - data/lib/active_record/fixture_set/render_context.rb +17 -0
 - data/lib/active_record/fixture_set/table_row.rb +187 -0
 - data/lib/active_record/fixture_set/table_rows.rb +46 -0
 - data/lib/active_record/fixtures.rb +206 -490
 - data/lib/active_record/future_result.rb +139 -0
 - data/lib/active_record/gem_version.rb +3 -3
 - data/lib/active_record/inheritance.rb +104 -37
 - data/lib/active_record/insert_all.rb +278 -0
 - data/lib/active_record/integration.rb +69 -18
 - data/lib/active_record/internal_metadata.rb +24 -9
 - data/lib/active_record/legacy_yaml_adapter.rb +3 -36
 - data/lib/active_record/locking/optimistic.rb +41 -26
 - data/lib/active_record/locking/pessimistic.rb +18 -8
 - data/lib/active_record/log_subscriber.rb +46 -35
 - data/lib/active_record/middleware/database_selector/resolver/session.rb +48 -0
 - data/lib/active_record/middleware/database_selector/resolver.rb +88 -0
 - data/lib/active_record/middleware/database_selector.rb +82 -0
 - data/lib/active_record/middleware/shard_selector.rb +60 -0
 - data/lib/active_record/migration/command_recorder.rb +96 -44
 - data/lib/active_record/migration/compatibility.rb +246 -64
 - data/lib/active_record/migration/join_table.rb +1 -2
 - data/lib/active_record/migration.rb +266 -187
 - data/lib/active_record/model_schema.rb +165 -52
 - data/lib/active_record/nested_attributes.rb +17 -19
 - data/lib/active_record/no_touching.rb +11 -4
 - data/lib/active_record/null_relation.rb +2 -7
 - data/lib/active_record/persistence.rb +467 -92
 - data/lib/active_record/query_cache.rb +21 -4
 - data/lib/active_record/query_logs.rb +138 -0
 - data/lib/active_record/querying.rb +51 -24
 - data/lib/active_record/railtie.rb +224 -57
 - data/lib/active_record/railties/console_sandbox.rb +2 -4
 - data/lib/active_record/railties/controller_runtime.rb +31 -36
 - data/lib/active_record/railties/databases.rake +369 -101
 - data/lib/active_record/readonly_attributes.rb +15 -0
 - data/lib/active_record/reflection.rb +170 -137
 - data/lib/active_record/relation/batches/batch_enumerator.rb +44 -14
 - data/lib/active_record/relation/batches.rb +46 -37
 - data/lib/active_record/relation/calculations.rb +168 -96
 - data/lib/active_record/relation/delegation.rb +37 -52
 - data/lib/active_record/relation/finder_methods.rb +79 -58
 - data/lib/active_record/relation/from_clause.rb +5 -1
 - data/lib/active_record/relation/merger.rb +50 -51
 - data/lib/active_record/relation/predicate_builder/array_handler.rb +13 -13
 - data/lib/active_record/relation/predicate_builder/association_query_value.rb +5 -9
 - data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +1 -2
 - data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +11 -10
 - data/lib/active_record/relation/predicate_builder/range_handler.rb +3 -23
 - data/lib/active_record/relation/predicate_builder/relation_handler.rb +1 -1
 - data/lib/active_record/relation/predicate_builder.rb +58 -46
 - data/lib/active_record/relation/query_attribute.rb +9 -10
 - data/lib/active_record/relation/query_methods.rb +685 -208
 - data/lib/active_record/relation/record_fetch_warning.rb +9 -11
 - data/lib/active_record/relation/spawn_methods.rb +10 -10
 - data/lib/active_record/relation/where_clause.rb +108 -64
 - data/lib/active_record/relation.rb +515 -151
 - data/lib/active_record/result.rb +78 -42
 - data/lib/active_record/runtime_registry.rb +9 -13
 - data/lib/active_record/sanitization.rb +29 -44
 - data/lib/active_record/schema.rb +37 -31
 - data/lib/active_record/schema_dumper.rb +74 -23
 - data/lib/active_record/schema_migration.rb +7 -9
 - data/lib/active_record/scoping/default.rb +62 -17
 - data/lib/active_record/scoping/named.rb +17 -32
 - data/lib/active_record/scoping.rb +70 -41
 - data/lib/active_record/secure_token.rb +16 -8
 - data/lib/active_record/serialization.rb +6 -4
 - data/lib/active_record/signed_id.rb +116 -0
 - data/lib/active_record/statement_cache.rb +49 -6
 - data/lib/active_record/store.rb +88 -9
 - data/lib/active_record/suppressor.rb +13 -17
 - data/lib/active_record/table_metadata.rb +42 -43
 - data/lib/active_record/tasks/database_tasks.rb +352 -94
 - data/lib/active_record/tasks/mysql_database_tasks.rb +37 -39
 - data/lib/active_record/tasks/postgresql_database_tasks.rb +41 -39
 - data/lib/active_record/tasks/sqlite_database_tasks.rb +14 -17
 - data/lib/active_record/test_databases.rb +24 -0
 - data/lib/active_record/test_fixtures.rb +287 -0
 - data/lib/active_record/timestamp.rb +44 -34
 - data/lib/active_record/touch_later.rb +23 -22
 - data/lib/active_record/transactions.rb +67 -128
 - data/lib/active_record/translation.rb +3 -3
 - data/lib/active_record/type/adapter_specific_registry.rb +34 -19
 - data/lib/active_record/type/hash_lookup_type_map.rb +34 -2
 - data/lib/active_record/type/internal/timezone.rb +2 -2
 - data/lib/active_record/type/serialized.rb +7 -4
 - data/lib/active_record/type/time.rb +10 -0
 - data/lib/active_record/type/type_map.rb +17 -21
 - data/lib/active_record/type/unsigned_integer.rb +0 -1
 - data/lib/active_record/type.rb +9 -5
 - data/lib/active_record/type_caster/connection.rb +15 -15
 - data/lib/active_record/type_caster/map.rb +8 -8
 - data/lib/active_record/validations/associated.rb +2 -3
 - data/lib/active_record/validations/numericality.rb +35 -0
 - data/lib/active_record/validations/uniqueness.rb +39 -31
 - data/lib/active_record/validations.rb +4 -3
 - data/lib/active_record.rb +209 -32
 - data/lib/arel/alias_predication.rb +9 -0
 - data/lib/arel/attributes/attribute.rb +33 -0
 - data/lib/arel/collectors/bind.rb +29 -0
 - data/lib/arel/collectors/composite.rb +39 -0
 - data/lib/arel/collectors/plain_string.rb +20 -0
 - data/lib/arel/collectors/sql_string.rb +27 -0
 - data/lib/arel/collectors/substitute_binds.rb +35 -0
 - data/lib/arel/crud.rb +48 -0
 - data/lib/arel/delete_manager.rb +32 -0
 - data/lib/arel/errors.rb +9 -0
 - data/lib/arel/expressions.rb +29 -0
 - data/lib/arel/factory_methods.rb +49 -0
 - data/lib/arel/filter_predications.rb +9 -0
 - data/lib/arel/insert_manager.rb +48 -0
 - data/lib/arel/math.rb +45 -0
 - data/lib/arel/nodes/and.rb +32 -0
 - data/lib/arel/nodes/ascending.rb +23 -0
 - data/lib/arel/nodes/binary.rb +126 -0
 - data/lib/arel/nodes/bind_param.rb +44 -0
 - data/lib/arel/nodes/case.rb +55 -0
 - data/lib/arel/nodes/casted.rb +62 -0
 - data/lib/arel/nodes/comment.rb +29 -0
 - data/lib/arel/nodes/count.rb +12 -0
 - data/lib/arel/nodes/delete_statement.rb +44 -0
 - data/lib/arel/nodes/descending.rb +23 -0
 - data/lib/arel/nodes/equality.rb +15 -0
 - data/lib/arel/nodes/extract.rb +24 -0
 - data/lib/arel/nodes/false.rb +16 -0
 - data/lib/arel/nodes/filter.rb +10 -0
 - data/lib/arel/nodes/full_outer_join.rb +8 -0
 - data/lib/arel/nodes/function.rb +45 -0
 - data/lib/arel/nodes/grouping.rb +11 -0
 - data/lib/arel/nodes/homogeneous_in.rb +76 -0
 - data/lib/arel/nodes/in.rb +15 -0
 - data/lib/arel/nodes/infix_operation.rb +92 -0
 - data/lib/arel/nodes/inner_join.rb +8 -0
 - data/lib/arel/nodes/insert_statement.rb +37 -0
 - data/lib/arel/nodes/join_source.rb +20 -0
 - data/lib/arel/nodes/matches.rb +18 -0
 - data/lib/arel/nodes/named_function.rb +23 -0
 - data/lib/arel/nodes/node.rb +51 -0
 - data/lib/arel/nodes/node_expression.rb +13 -0
 - data/lib/arel/nodes/ordering.rb +27 -0
 - data/lib/arel/nodes/outer_join.rb +8 -0
 - data/lib/arel/nodes/over.rb +15 -0
 - data/lib/arel/nodes/regexp.rb +16 -0
 - data/lib/arel/nodes/right_outer_join.rb +8 -0
 - data/lib/arel/nodes/select_core.rb +67 -0
 - data/lib/arel/nodes/select_statement.rb +41 -0
 - data/lib/arel/nodes/sql_literal.rb +19 -0
 - data/lib/arel/nodes/string_join.rb +11 -0
 - data/lib/arel/nodes/table_alias.rb +31 -0
 - data/lib/arel/nodes/terminal.rb +16 -0
 - data/lib/arel/nodes/true.rb +16 -0
 - data/lib/arel/nodes/unary.rb +44 -0
 - data/lib/arel/nodes/unary_operation.rb +20 -0
 - data/lib/arel/nodes/unqualified_column.rb +22 -0
 - data/lib/arel/nodes/update_statement.rb +46 -0
 - data/lib/arel/nodes/values_list.rb +9 -0
 - data/lib/arel/nodes/window.rb +126 -0
 - data/lib/arel/nodes/with.rb +11 -0
 - data/lib/arel/nodes.rb +71 -0
 - data/lib/arel/order_predications.rb +13 -0
 - data/lib/arel/predications.rb +258 -0
 - data/lib/arel/select_manager.rb +276 -0
 - data/lib/arel/table.rb +117 -0
 - data/lib/arel/tree_manager.rb +60 -0
 - data/lib/arel/update_manager.rb +48 -0
 - data/lib/arel/visitors/dot.rb +298 -0
 - data/lib/arel/visitors/mysql.rb +99 -0
 - data/lib/arel/visitors/postgresql.rb +110 -0
 - data/lib/arel/visitors/sqlite.rb +38 -0
 - data/lib/arel/visitors/to_sql.rb +955 -0
 - data/lib/arel/visitors/visitor.rb +45 -0
 - data/lib/arel/visitors.rb +13 -0
 - data/lib/arel/window_predications.rb +9 -0
 - data/lib/arel.rb +55 -0
 - data/lib/rails/generators/active_record/application_record/application_record_generator.rb +0 -1
 - data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +1 -1
 - data/lib/rails/generators/active_record/migration/migration_generator.rb +3 -5
 - data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +3 -1
 - data/lib/rails/generators/active_record/migration/templates/migration.rb.tt +7 -5
 - data/lib/rails/generators/active_record/migration.rb +19 -2
 - data/lib/rails/generators/active_record/model/model_generator.rb +39 -2
 - data/lib/rails/generators/active_record/model/templates/abstract_base_class.rb.tt +7 -0
 - data/lib/rails/generators/active_record/model/templates/model.rb.tt +10 -1
 - data/lib/rails/generators/active_record/model/templates/module.rb.tt +2 -2
 - data/lib/rails/generators/active_record/multi_db/multi_db_generator.rb +16 -0
 - data/lib/rails/generators/active_record/multi_db/templates/multi_db.rb.tt +44 -0
 - metadata +162 -32
 - data/lib/active_record/attribute_decorators.rb +0 -90
 - data/lib/active_record/collection_cache_key.rb +0 -53
 - data/lib/active_record/connection_adapters/connection_specification.rb +0 -287
 - data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +0 -33
 - data/lib/active_record/define_callbacks.rb +0 -22
 - data/lib/active_record/relation/predicate_builder/base_handler.rb +0 -19
 - data/lib/active_record/relation/where_clause_factory.rb +0 -34
 
| 
         @@ -2,6 +2,15 @@ 
     | 
|
| 
       2 
2 
     | 
    
         | 
| 
       3 
3 
     | 
    
         
             
            module ActiveRecord
         
     | 
| 
       4 
4 
     | 
    
         
             
              module Scoping
         
     | 
| 
      
 5 
     | 
    
         
            +
                class DefaultScope # :nodoc:
         
     | 
| 
      
 6 
     | 
    
         
            +
                  attr_reader :scope, :all_queries
         
     | 
| 
      
 7 
     | 
    
         
            +
             
     | 
| 
      
 8 
     | 
    
         
            +
                  def initialize(scope, all_queries = nil)
         
     | 
| 
      
 9 
     | 
    
         
            +
                    @scope = scope
         
     | 
| 
      
 10 
     | 
    
         
            +
                    @all_queries = all_queries
         
     | 
| 
      
 11 
     | 
    
         
            +
                  end
         
     | 
| 
      
 12 
     | 
    
         
            +
                end
         
     | 
| 
      
 13 
     | 
    
         
            +
             
     | 
| 
       5 
14 
     | 
    
         
             
                module Default
         
     | 
| 
       6 
15 
     | 
    
         
             
                  extend ActiveSupport::Concern
         
     | 
| 
       7 
16 
     | 
    
         | 
| 
         @@ -30,8 +39,8 @@ module ActiveRecord 
     | 
|
| 
       30 
39 
     | 
    
         
             
                    #   Post.unscoped {
         
     | 
| 
       31 
40 
     | 
    
         
             
                    #     Post.limit(10) # Fires "SELECT * FROM posts LIMIT 10"
         
     | 
| 
       32 
41 
     | 
    
         
             
                    #   }
         
     | 
| 
       33 
     | 
    
         
            -
                    def unscoped
         
     | 
| 
       34 
     | 
    
         
            -
                      block_given? ? relation.scoping  
     | 
| 
      
 42 
     | 
    
         
            +
                    def unscoped(&block)
         
     | 
| 
      
 43 
     | 
    
         
            +
                      block_given? ? relation.scoping(&block) : relation
         
     | 
| 
       35 
44 
     | 
    
         
             
                    end
         
     | 
| 
       36 
45 
     | 
    
         | 
| 
       37 
46 
     | 
    
         
             
                    # Are there attributes associated with this scope?
         
     | 
| 
         @@ -39,12 +48,22 @@ module ActiveRecord 
     | 
|
| 
       39 
48 
     | 
    
         
             
                      super || default_scopes.any? || respond_to?(:default_scope)
         
     | 
| 
       40 
49 
     | 
    
         
             
                    end
         
     | 
| 
       41 
50 
     | 
    
         | 
| 
       42 
     | 
    
         
            -
                    def before_remove_const  
     | 
| 
      
 51 
     | 
    
         
            +
                    def before_remove_const # :nodoc:
         
     | 
| 
       43 
52 
     | 
    
         
             
                      self.current_scope = nil
         
     | 
| 
       44 
53 
     | 
    
         
             
                    end
         
     | 
| 
       45 
54 
     | 
    
         | 
| 
       46 
     | 
    
         
            -
                     
     | 
| 
      
 55 
     | 
    
         
            +
                    # Checks if the model has any default scopes. If all_queries
         
     | 
| 
      
 56 
     | 
    
         
            +
                    # is set to true, the method will check if there are any
         
     | 
| 
      
 57 
     | 
    
         
            +
                    # default_scopes for the model  where +all_queries+ is true.
         
     | 
| 
      
 58 
     | 
    
         
            +
                    def default_scopes?(all_queries: false)
         
     | 
| 
      
 59 
     | 
    
         
            +
                      if all_queries
         
     | 
| 
      
 60 
     | 
    
         
            +
                        self.default_scopes.any?(&:all_queries)
         
     | 
| 
      
 61 
     | 
    
         
            +
                      else
         
     | 
| 
      
 62 
     | 
    
         
            +
                        self.default_scopes.any?
         
     | 
| 
      
 63 
     | 
    
         
            +
                      end
         
     | 
| 
      
 64 
     | 
    
         
            +
                    end
         
     | 
| 
       47 
65 
     | 
    
         | 
| 
      
 66 
     | 
    
         
            +
                    private
         
     | 
| 
       48 
67 
     | 
    
         
             
                      # Use this macro in your model to set a default scope for all operations on
         
     | 
| 
       49 
68 
     | 
    
         
             
                      # the model.
         
     | 
| 
       50 
69 
     | 
    
         
             
                      #
         
     | 
| 
         @@ -55,11 +74,26 @@ module ActiveRecord 
     | 
|
| 
       55 
74 
     | 
    
         
             
                      #   Article.all # => SELECT * FROM articles WHERE published = true
         
     | 
| 
       56 
75 
     | 
    
         
             
                      #
         
     | 
| 
       57 
76 
     | 
    
         
             
                      # The #default_scope is also applied while creating/building a record.
         
     | 
| 
       58 
     | 
    
         
            -
                      # It is not applied while updating a record.
         
     | 
| 
      
 77 
     | 
    
         
            +
                      # It is not applied while updating or deleting a record.
         
     | 
| 
       59 
78 
     | 
    
         
             
                      #
         
     | 
| 
       60 
79 
     | 
    
         
             
                      #   Article.new.published    # => true
         
     | 
| 
       61 
80 
     | 
    
         
             
                      #   Article.create.published # => true
         
     | 
| 
       62 
81 
     | 
    
         
             
                      #
         
     | 
| 
      
 82 
     | 
    
         
            +
                      # To apply a #default_scope when updating or deleting a record, add
         
     | 
| 
      
 83 
     | 
    
         
            +
                      # <tt>all_queries: true</tt>:
         
     | 
| 
      
 84 
     | 
    
         
            +
                      #
         
     | 
| 
      
 85 
     | 
    
         
            +
                      #   class Article < ActiveRecord::Base
         
     | 
| 
      
 86 
     | 
    
         
            +
                      #     default_scope { where(blog_id: 1) }, all_queries: true
         
     | 
| 
      
 87 
     | 
    
         
            +
                      #   end
         
     | 
| 
      
 88 
     | 
    
         
            +
                      #
         
     | 
| 
      
 89 
     | 
    
         
            +
                      # Applying a default scope to all queries will ensure that records
         
     | 
| 
      
 90 
     | 
    
         
            +
                      # are always queried by the additional conditions. Note that only
         
     | 
| 
      
 91 
     | 
    
         
            +
                      # where clauses apply, as it does not make sense to add order to
         
     | 
| 
      
 92 
     | 
    
         
            +
                      # queries that return a single object by primary key.
         
     | 
| 
      
 93 
     | 
    
         
            +
                      #
         
     | 
| 
      
 94 
     | 
    
         
            +
                      #   Article.find(1).destroy
         
     | 
| 
      
 95 
     | 
    
         
            +
                      #   => DELETE ... FROM `articles` where ID = 1 AND blog_id = 1;
         
     | 
| 
      
 96 
     | 
    
         
            +
                      #
         
     | 
| 
       63 
97 
     | 
    
         
             
                      # (You can also pass any object which responds to +call+ to the
         
     | 
| 
       64 
98 
     | 
    
         
             
                      # +default_scope+ macro, and it will be called when building the
         
     | 
| 
       65 
99 
     | 
    
         
             
                      # default scope.)
         
     | 
| 
         @@ -86,7 +120,7 @@ module ActiveRecord 
     | 
|
| 
       86 
120 
     | 
    
         
             
                      #       # Should return a scope, you can call 'super' here etc.
         
     | 
| 
       87 
121 
     | 
    
         
             
                      #     end
         
     | 
| 
       88 
122 
     | 
    
         
             
                      #   end
         
     | 
| 
       89 
     | 
    
         
            -
                      def default_scope(scope = nil, &block) # :doc:
         
     | 
| 
      
 123 
     | 
    
         
            +
                      def default_scope(scope = nil, all_queries: nil, &block) # :doc:
         
     | 
| 
       90 
124 
     | 
    
         
             
                        scope = block if block_given?
         
     | 
| 
       91 
125 
     | 
    
         | 
| 
       92 
126 
     | 
    
         
             
                        if scope.is_a?(Relation) || !scope.respond_to?(:call)
         
     | 
| 
         @@ -97,10 +131,12 @@ module ActiveRecord 
     | 
|
| 
       97 
131 
     | 
    
         
             
                            "self.default_scope.)"
         
     | 
| 
       98 
132 
     | 
    
         
             
                        end
         
     | 
| 
       99 
133 
     | 
    
         | 
| 
       100 
     | 
    
         
            -
                         
     | 
| 
      
 134 
     | 
    
         
            +
                        default_scope = DefaultScope.new(scope, all_queries)
         
     | 
| 
      
 135 
     | 
    
         
            +
             
     | 
| 
      
 136 
     | 
    
         
            +
                        self.default_scopes += [default_scope]
         
     | 
| 
       101 
137 
     | 
    
         
             
                      end
         
     | 
| 
       102 
138 
     | 
    
         | 
| 
       103 
     | 
    
         
            -
                      def build_default_scope( 
     | 
| 
      
 139 
     | 
    
         
            +
                      def build_default_scope(relation = relation(), all_queries: nil)
         
     | 
| 
       104 
140 
     | 
    
         
             
                        return if abstract_class?
         
     | 
| 
       105 
141 
     | 
    
         | 
| 
       106 
142 
     | 
    
         
             
                        if default_scope_override.nil?
         
     | 
| 
         @@ -110,27 +146,36 @@ module ActiveRecord 
     | 
|
| 
       110 
146 
     | 
    
         
             
                        if default_scope_override
         
     | 
| 
       111 
147 
     | 
    
         
             
                          # The user has defined their own default scope method, so call that
         
     | 
| 
       112 
148 
     | 
    
         
             
                          evaluate_default_scope do
         
     | 
| 
       113 
     | 
    
         
            -
                             
     | 
| 
       114 
     | 
    
         
            -
                              (base_rel ||= relation).merge!(scope)
         
     | 
| 
       115 
     | 
    
         
            -
                            end
         
     | 
| 
      
 149 
     | 
    
         
            +
                            relation.scoping { default_scope }
         
     | 
| 
       116 
150 
     | 
    
         
             
                          end
         
     | 
| 
       117 
151 
     | 
    
         
             
                        elsif default_scopes.any?
         
     | 
| 
       118 
     | 
    
         
            -
                          base_rel ||= relation
         
     | 
| 
       119 
152 
     | 
    
         
             
                          evaluate_default_scope do
         
     | 
| 
       120 
     | 
    
         
            -
                            default_scopes.inject( 
     | 
| 
       121 
     | 
    
         
            -
                               
     | 
| 
       122 
     | 
    
         
            -
             
     | 
| 
      
 153 
     | 
    
         
            +
                            default_scopes.inject(relation) do |default_scope, scope_obj|
         
     | 
| 
      
 154 
     | 
    
         
            +
                              if execute_scope?(all_queries, scope_obj)
         
     | 
| 
      
 155 
     | 
    
         
            +
                                scope = scope_obj.scope.respond_to?(:to_proc) ? scope_obj.scope : scope_obj.scope.method(:call)
         
     | 
| 
      
 156 
     | 
    
         
            +
             
     | 
| 
      
 157 
     | 
    
         
            +
                                default_scope.instance_exec(&scope) || default_scope
         
     | 
| 
      
 158 
     | 
    
         
            +
                              end
         
     | 
| 
       123 
159 
     | 
    
         
             
                            end
         
     | 
| 
       124 
160 
     | 
    
         
             
                          end
         
     | 
| 
       125 
161 
     | 
    
         
             
                        end
         
     | 
| 
       126 
162 
     | 
    
         
             
                      end
         
     | 
| 
       127 
163 
     | 
    
         | 
| 
      
 164 
     | 
    
         
            +
                      # If all_queries is nil, only execute on select and insert queries.
         
     | 
| 
      
 165 
     | 
    
         
            +
                      #
         
     | 
| 
      
 166 
     | 
    
         
            +
                      # If all_queries is true, check if the default_scope object has
         
     | 
| 
      
 167 
     | 
    
         
            +
                      # all_queries set, then execute on all queries; select, insert, update
         
     | 
| 
      
 168 
     | 
    
         
            +
                      # and delete.
         
     | 
| 
      
 169 
     | 
    
         
            +
                      def execute_scope?(all_queries, default_scope_obj)
         
     | 
| 
      
 170 
     | 
    
         
            +
                        all_queries.nil? || all_queries && default_scope_obj.all_queries
         
     | 
| 
      
 171 
     | 
    
         
            +
                      end
         
     | 
| 
      
 172 
     | 
    
         
            +
             
     | 
| 
       128 
173 
     | 
    
         
             
                      def ignore_default_scope?
         
     | 
| 
       129 
     | 
    
         
            -
                        ScopeRegistry. 
     | 
| 
      
 174 
     | 
    
         
            +
                        ScopeRegistry.ignore_default_scope(base_class)
         
     | 
| 
       130 
175 
     | 
    
         
             
                      end
         
     | 
| 
       131 
176 
     | 
    
         | 
| 
       132 
177 
     | 
    
         
             
                      def ignore_default_scope=(ignore)
         
     | 
| 
       133 
     | 
    
         
            -
                        ScopeRegistry. 
     | 
| 
      
 178 
     | 
    
         
            +
                        ScopeRegistry.set_ignore_default_scope(base_class, ignore)
         
     | 
| 
       134 
179 
     | 
    
         
             
                      end
         
     | 
| 
       135 
180 
     | 
    
         | 
| 
       136 
181 
     | 
    
         
             
                      # The ignore_default_scope flag is used to prevent an infinite recursion
         
     | 
| 
         @@ -1,9 +1,5 @@ 
     | 
|
| 
       1 
1 
     | 
    
         
             
            # frozen_string_literal: true
         
     | 
| 
       2 
2 
     | 
    
         | 
| 
       3 
     | 
    
         
            -
            require "active_support/core_ext/array"
         
     | 
| 
       4 
     | 
    
         
            -
            require "active_support/core_ext/hash/except"
         
     | 
| 
       5 
     | 
    
         
            -
            require "active_support/core_ext/kernel/singleton_class"
         
     | 
| 
       6 
     | 
    
         
            -
             
     | 
| 
       7 
3 
     | 
    
         
             
            module ActiveRecord
         
     | 
| 
       8 
4 
     | 
    
         
             
              # = Active Record \Named \Scopes
         
     | 
| 
       9 
5 
     | 
    
         
             
              module Scoping
         
     | 
| 
         @@ -24,13 +20,13 @@ module ActiveRecord 
     | 
|
| 
       24 
20 
     | 
    
         
             
                    # You can define a scope that applies to all finders using
         
     | 
| 
       25 
21 
     | 
    
         
             
                    # {default_scope}[rdoc-ref:Scoping::Default::ClassMethods#default_scope].
         
     | 
| 
       26 
22 
     | 
    
         
             
                    def all
         
     | 
| 
       27 
     | 
    
         
            -
                       
     | 
| 
      
 23 
     | 
    
         
            +
                      scope = current_scope
         
     | 
| 
       28 
24 
     | 
    
         | 
| 
       29 
     | 
    
         
            -
                      if  
     | 
| 
       30 
     | 
    
         
            -
                        if self ==  
     | 
| 
       31 
     | 
    
         
            -
                           
     | 
| 
      
 25 
     | 
    
         
            +
                      if scope
         
     | 
| 
      
 26 
     | 
    
         
            +
                        if self == scope.klass
         
     | 
| 
      
 27 
     | 
    
         
            +
                          scope.clone
         
     | 
| 
       32 
28 
     | 
    
         
             
                        else
         
     | 
| 
       33 
     | 
    
         
            -
                          relation.merge!( 
     | 
| 
      
 29 
     | 
    
         
            +
                          relation.merge!(scope)
         
     | 
| 
       34 
30 
     | 
    
         
             
                        end
         
     | 
| 
       35 
31 
     | 
    
         
             
                      else
         
     | 
| 
       36 
32 
     | 
    
         
             
                        default_scoped
         
     | 
| 
         @@ -38,21 +34,20 @@ module ActiveRecord 
     | 
|
| 
       38 
34 
     | 
    
         
             
                    end
         
     | 
| 
       39 
35 
     | 
    
         | 
| 
       40 
36 
     | 
    
         
             
                    def scope_for_association(scope = relation) # :nodoc:
         
     | 
| 
       41 
     | 
    
         
            -
                       
     | 
| 
       42 
     | 
    
         
            -
             
     | 
| 
       43 
     | 
    
         
            -
                      if current_scope && current_scope.empty_scope?
         
     | 
| 
      
 37 
     | 
    
         
            +
                      if current_scope&.empty_scope?
         
     | 
| 
       44 
38 
     | 
    
         
             
                        scope
         
     | 
| 
       45 
39 
     | 
    
         
             
                      else
         
     | 
| 
       46 
40 
     | 
    
         
             
                        default_scoped(scope)
         
     | 
| 
       47 
41 
     | 
    
         
             
                      end
         
     | 
| 
       48 
42 
     | 
    
         
             
                    end
         
     | 
| 
       49 
43 
     | 
    
         | 
| 
       50 
     | 
    
         
            -
                     
     | 
| 
       51 
     | 
    
         
            -
             
     | 
| 
      
 44 
     | 
    
         
            +
                    # Returns a scope for the model with default scopes.
         
     | 
| 
      
 45 
     | 
    
         
            +
                    def default_scoped(scope = relation, all_queries: nil)
         
     | 
| 
      
 46 
     | 
    
         
            +
                      build_default_scope(scope, all_queries: all_queries) || scope
         
     | 
| 
       52 
47 
     | 
    
         
             
                    end
         
     | 
| 
       53 
48 
     | 
    
         | 
| 
       54 
49 
     | 
    
         
             
                    def default_extensions # :nodoc:
         
     | 
| 
       55 
     | 
    
         
            -
                      if scope =  
     | 
| 
      
 50 
     | 
    
         
            +
                      if scope = scope_for_association || build_default_scope
         
     | 
| 
       56 
51 
     | 
    
         
             
                        scope.extensions
         
     | 
| 
       57 
52 
     | 
    
         
             
                      else
         
     | 
| 
       58 
53 
     | 
    
         
             
                        []
         
     | 
| 
         @@ -77,10 +72,6 @@ module ActiveRecord 
     | 
|
| 
       77 
72 
     | 
    
         
             
                    # <tt>Shirt.dry_clean_only</tt>. <tt>Shirt.red</tt>, in effect,
         
     | 
| 
       78 
73 
     | 
    
         
             
                    # represents the query <tt>Shirt.where(color: 'red')</tt>.
         
     | 
| 
       79 
74 
     | 
    
         
             
                    #
         
     | 
| 
       80 
     | 
    
         
            -
                    # You should always pass a callable object to the scopes defined
         
     | 
| 
       81 
     | 
    
         
            -
                    # with #scope. This ensures that the scope is re-evaluated each
         
     | 
| 
       82 
     | 
    
         
            -
                    # time it is called.
         
     | 
| 
       83 
     | 
    
         
            -
                    #
         
     | 
| 
       84 
75 
     | 
    
         
             
                    # Note that this is simply 'syntactic sugar' for defining an actual
         
     | 
| 
       85 
76 
     | 
    
         
             
                    # class method:
         
     | 
| 
       86 
77 
     | 
    
         
             
                    #
         
     | 
| 
         @@ -177,35 +168,29 @@ module ActiveRecord 
     | 
|
| 
       177 
168 
     | 
    
         
             
                          "an instance method with the same name."
         
     | 
| 
       178 
169 
     | 
    
         
             
                      end
         
     | 
| 
       179 
170 
     | 
    
         | 
| 
       180 
     | 
    
         
            -
                      valid_scope_name?(name)
         
     | 
| 
       181 
171 
     | 
    
         
             
                      extension = Module.new(&block) if block
         
     | 
| 
       182 
172 
     | 
    
         | 
| 
       183 
173 
     | 
    
         
             
                      if body.respond_to?(:to_proc)
         
     | 
| 
       184 
     | 
    
         
            -
                        singleton_class. 
     | 
| 
       185 
     | 
    
         
            -
                          scope = all
         
     | 
| 
       186 
     | 
    
         
            -
                          scope = scope._exec_scope(*args, &body)
         
     | 
| 
      
 174 
     | 
    
         
            +
                        singleton_class.define_method(name) do |*args|
         
     | 
| 
      
 175 
     | 
    
         
            +
                          scope = all._exec_scope(*args, &body)
         
     | 
| 
       187 
176 
     | 
    
         
             
                          scope = scope.extending(extension) if extension
         
     | 
| 
       188 
177 
     | 
    
         
             
                          scope
         
     | 
| 
       189 
178 
     | 
    
         
             
                        end
         
     | 
| 
       190 
179 
     | 
    
         
             
                      else
         
     | 
| 
       191 
     | 
    
         
            -
                        singleton_class. 
     | 
| 
       192 
     | 
    
         
            -
                          scope = all
         
     | 
| 
       193 
     | 
    
         
            -
                          scope = scope.scoping { body.call(*args) || scope }
         
     | 
| 
      
 180 
     | 
    
         
            +
                        singleton_class.define_method(name) do |*args|
         
     | 
| 
      
 181 
     | 
    
         
            +
                          scope = body.call(*args) || all
         
     | 
| 
       194 
182 
     | 
    
         
             
                          scope = scope.extending(extension) if extension
         
     | 
| 
       195 
183 
     | 
    
         
             
                          scope
         
     | 
| 
       196 
184 
     | 
    
         
             
                        end
         
     | 
| 
       197 
185 
     | 
    
         
             
                      end
         
     | 
| 
      
 186 
     | 
    
         
            +
                      singleton_class.send(:ruby2_keywords, name)
         
     | 
| 
       198 
187 
     | 
    
         | 
| 
       199 
188 
     | 
    
         
             
                      generate_relation_method(name)
         
     | 
| 
       200 
189 
     | 
    
         
             
                    end
         
     | 
| 
       201 
190 
     | 
    
         | 
| 
       202 
191 
     | 
    
         
             
                    private
         
     | 
| 
       203 
     | 
    
         
            -
             
     | 
| 
       204 
     | 
    
         
            -
             
     | 
| 
       205 
     | 
    
         
            -
                        if respond_to?(name, true) && logger
         
     | 
| 
       206 
     | 
    
         
            -
                          logger.warn "Creating scope :#{name}. " \
         
     | 
| 
       207 
     | 
    
         
            -
                            "Overwriting existing method #{self.name}.#{name}."
         
     | 
| 
       208 
     | 
    
         
            -
                        end
         
     | 
| 
      
 192 
     | 
    
         
            +
                      def singleton_method_added(name)
         
     | 
| 
      
 193 
     | 
    
         
            +
                        generate_relation_method(name) if Kernel.respond_to?(name) && !ActiveRecord::Relation.method_defined?(name)
         
     | 
| 
       209 
194 
     | 
    
         
             
                      end
         
     | 
| 
       210 
195 
     | 
    
         
             
                  end
         
     | 
| 
       211 
196 
     | 
    
         
             
                end
         
     | 
| 
         @@ -1,6 +1,6 @@ 
     | 
|
| 
       1 
1 
     | 
    
         
             
            # frozen_string_literal: true
         
     | 
| 
       2 
2 
     | 
    
         | 
| 
       3 
     | 
    
         
            -
            require "active_support/ 
     | 
| 
      
 3 
     | 
    
         
            +
            require "active_support/core_ext/module/delegation"
         
     | 
| 
       4 
4 
     | 
    
         | 
| 
       5 
5 
     | 
    
         
             
            module ActiveRecord
         
     | 
| 
       6 
6 
     | 
    
         
             
              module Scoping
         
     | 
| 
         @@ -12,14 +12,6 @@ module ActiveRecord 
     | 
|
| 
       12 
12 
     | 
    
         
             
                end
         
     | 
| 
       13 
13 
     | 
    
         | 
| 
       14 
14 
     | 
    
         
             
                module ClassMethods # :nodoc:
         
     | 
| 
       15 
     | 
    
         
            -
                  def current_scope(skip_inherited_scope = false)
         
     | 
| 
       16 
     | 
    
         
            -
                    ScopeRegistry.value_for(:current_scope, self, skip_inherited_scope)
         
     | 
| 
       17 
     | 
    
         
            -
                  end
         
     | 
| 
       18 
     | 
    
         
            -
             
     | 
| 
       19 
     | 
    
         
            -
                  def current_scope=(scope)
         
     | 
| 
       20 
     | 
    
         
            -
                    ScopeRegistry.set_value_for(:current_scope, self, scope)
         
     | 
| 
       21 
     | 
    
         
            -
                  end
         
     | 
| 
       22 
     | 
    
         
            -
             
     | 
| 
       23 
15 
     | 
    
         
             
                  # Collects attributes from scopes that should be applied when creating
         
     | 
| 
       24 
16 
     | 
    
         
             
                  # an AR instance for the particular class this is called on.
         
     | 
| 
       25 
17 
     | 
    
         
             
                  def scope_attributes
         
     | 
| 
         @@ -30,6 +22,26 @@ module ActiveRecord 
     | 
|
| 
       30 
22 
     | 
    
         
             
                  def scope_attributes?
         
     | 
| 
       31 
23 
     | 
    
         
             
                    current_scope
         
     | 
| 
       32 
24 
     | 
    
         
             
                  end
         
     | 
| 
      
 25 
     | 
    
         
            +
             
     | 
| 
      
 26 
     | 
    
         
            +
                  def current_scope(skip_inherited_scope = false)
         
     | 
| 
      
 27 
     | 
    
         
            +
                    ScopeRegistry.current_scope(self, skip_inherited_scope)
         
     | 
| 
      
 28 
     | 
    
         
            +
                  end
         
     | 
| 
      
 29 
     | 
    
         
            +
             
     | 
| 
      
 30 
     | 
    
         
            +
                  def current_scope=(scope)
         
     | 
| 
      
 31 
     | 
    
         
            +
                    ScopeRegistry.set_current_scope(self, scope)
         
     | 
| 
      
 32 
     | 
    
         
            +
                  end
         
     | 
| 
      
 33 
     | 
    
         
            +
             
     | 
| 
      
 34 
     | 
    
         
            +
                  def global_current_scope(skip_inherited_scope = false)
         
     | 
| 
      
 35 
     | 
    
         
            +
                    ScopeRegistry.global_current_scope(self, skip_inherited_scope)
         
     | 
| 
      
 36 
     | 
    
         
            +
                  end
         
     | 
| 
      
 37 
     | 
    
         
            +
             
     | 
| 
      
 38 
     | 
    
         
            +
                  def global_current_scope=(scope)
         
     | 
| 
      
 39 
     | 
    
         
            +
                    ScopeRegistry.set_global_current_scope(self, scope)
         
     | 
| 
      
 40 
     | 
    
         
            +
                  end
         
     | 
| 
      
 41 
     | 
    
         
            +
             
     | 
| 
      
 42 
     | 
    
         
            +
                  def scope_registry
         
     | 
| 
      
 43 
     | 
    
         
            +
                    ScopeRegistry.instance
         
     | 
| 
      
 44 
     | 
    
         
            +
                  end
         
     | 
| 
       33 
45 
     | 
    
         
             
                end
         
     | 
| 
       34 
46 
     | 
    
         | 
| 
       35 
47 
     | 
    
         
             
                def populate_with_current_scope_attributes # :nodoc:
         
     | 
| 
         @@ -45,8 +57,8 @@ module ActiveRecord 
     | 
|
| 
       45 
57 
     | 
    
         
             
                end
         
     | 
| 
       46 
58 
     | 
    
         | 
| 
       47 
59 
     | 
    
         
             
                # This class stores the +:current_scope+ and +:ignore_default_scope+ values
         
     | 
| 
       48 
     | 
    
         
            -
                # for different classes. The registry is stored as a thread  
     | 
| 
       49 
     | 
    
         
            -
                #  
     | 
| 
      
 60 
     | 
    
         
            +
                # for different classes. The registry is stored as either a thread or fiber
         
     | 
| 
      
 61 
     | 
    
         
            +
                # local depending on the application configuration.
         
     | 
| 
       50 
62 
     | 
    
         
             
                #
         
     | 
| 
       51 
63 
     | 
    
         
             
                # This class allows you to store and get the scope values on different
         
     | 
| 
       52 
64 
     | 
    
         
             
                # classes and different types of scopes. For example, if you are attempting
         
     | 
| 
         @@ -54,53 +66,70 @@ module ActiveRecord 
     | 
|
| 
       54 
66 
     | 
    
         
             
                # following code:
         
     | 
| 
       55 
67 
     | 
    
         
             
                #
         
     | 
| 
       56 
68 
     | 
    
         
             
                #   registry = ActiveRecord::Scoping::ScopeRegistry
         
     | 
| 
       57 
     | 
    
         
            -
                #   registry. 
     | 
| 
      
 69 
     | 
    
         
            +
                #   registry.set_current_scope(Board, some_new_scope)
         
     | 
| 
       58 
70 
     | 
    
         
             
                #
         
     | 
| 
       59 
71 
     | 
    
         
             
                # Now when you run:
         
     | 
| 
       60 
72 
     | 
    
         
             
                #
         
     | 
| 
       61 
     | 
    
         
            -
                #   registry. 
     | 
| 
       62 
     | 
    
         
            -
                #
         
     | 
| 
       63 
     | 
    
         
            -
                # You will obtain whatever was defined in +some_new_scope+. The #value_for
         
     | 
| 
       64 
     | 
    
         
            -
                # and #set_value_for methods are delegated to the current ScopeRegistry
         
     | 
| 
       65 
     | 
    
         
            -
                # object, so the above example code can also be called as:
         
     | 
| 
      
 73 
     | 
    
         
            +
                #   registry.current_scope(Board)
         
     | 
| 
       66 
74 
     | 
    
         
             
                #
         
     | 
| 
       67 
     | 
    
         
            -
                # 
     | 
| 
       68 
     | 
    
         
            -
                #       Board, some_new_scope)
         
     | 
| 
      
 75 
     | 
    
         
            +
                # You will obtain whatever was defined in +some_new_scope+.
         
     | 
| 
       69 
76 
     | 
    
         
             
                class ScopeRegistry # :nodoc:
         
     | 
| 
       70 
     | 
    
         
            -
                   
     | 
| 
      
 77 
     | 
    
         
            +
                  class << self
         
     | 
| 
      
 78 
     | 
    
         
            +
                    delegate :current_scope, :set_current_scope, :ignore_default_scope, :set_ignore_default_scope,
         
     | 
| 
      
 79 
     | 
    
         
            +
                      :global_current_scope, :set_global_current_scope, to: :instance
         
     | 
| 
       71 
80 
     | 
    
         | 
| 
       72 
     | 
    
         
            -
             
     | 
| 
      
 81 
     | 
    
         
            +
                    def instance
         
     | 
| 
      
 82 
     | 
    
         
            +
                      ActiveSupport::IsolatedExecutionState[:active_record_scope_registry] ||= new
         
     | 
| 
      
 83 
     | 
    
         
            +
                    end
         
     | 
| 
      
 84 
     | 
    
         
            +
                  end
         
     | 
| 
       73 
85 
     | 
    
         | 
| 
       74 
86 
     | 
    
         
             
                  def initialize
         
     | 
| 
       75 
     | 
    
         
            -
                    @ 
     | 
| 
      
 87 
     | 
    
         
            +
                    @current_scope        = {}
         
     | 
| 
      
 88 
     | 
    
         
            +
                    @ignore_default_scope = {}
         
     | 
| 
      
 89 
     | 
    
         
            +
                    @global_current_scope = {}
         
     | 
| 
       76 
90 
     | 
    
         
             
                  end
         
     | 
| 
       77 
91 
     | 
    
         | 
| 
       78 
     | 
    
         
            -
                   
     | 
| 
       79 
     | 
    
         
            -
             
     | 
| 
       80 
     | 
    
         
            -
                    raise_invalid_scope_type!(scope_type)
         
     | 
| 
       81 
     | 
    
         
            -
                    return @registry[scope_type][model.name] if skip_inherited_scope
         
     | 
| 
       82 
     | 
    
         
            -
                    klass = model
         
     | 
| 
       83 
     | 
    
         
            -
                    base = model.base_class
         
     | 
| 
       84 
     | 
    
         
            -
                    while klass <= base
         
     | 
| 
       85 
     | 
    
         
            -
                      value = @registry[scope_type][klass.name]
         
     | 
| 
       86 
     | 
    
         
            -
                      return value if value
         
     | 
| 
       87 
     | 
    
         
            -
                      klass = klass.superclass
         
     | 
| 
       88 
     | 
    
         
            -
                    end
         
     | 
| 
      
 92 
     | 
    
         
            +
                  def current_scope(model, skip_inherited_scope = false)
         
     | 
| 
      
 93 
     | 
    
         
            +
                    value_for(@current_scope, model, skip_inherited_scope)
         
     | 
| 
       89 
94 
     | 
    
         
             
                  end
         
     | 
| 
       90 
95 
     | 
    
         | 
| 
       91 
     | 
    
         
            -
                   
     | 
| 
       92 
     | 
    
         
            -
             
     | 
| 
       93 
     | 
    
         
            -
                    raise_invalid_scope_type!(scope_type)
         
     | 
| 
       94 
     | 
    
         
            -
                    @registry[scope_type][model.name] = value
         
     | 
| 
      
 96 
     | 
    
         
            +
                  def set_current_scope(model, value)
         
     | 
| 
      
 97 
     | 
    
         
            +
                    set_value_for(@current_scope, model, value)
         
     | 
| 
       95 
98 
     | 
    
         
             
                  end
         
     | 
| 
       96 
99 
     | 
    
         | 
| 
       97 
     | 
    
         
            -
                   
     | 
| 
      
 100 
     | 
    
         
            +
                  def ignore_default_scope(model, skip_inherited_scope = false)
         
     | 
| 
      
 101 
     | 
    
         
            +
                    value_for(@ignore_default_scope, model, skip_inherited_scope)
         
     | 
| 
      
 102 
     | 
    
         
            +
                  end
         
     | 
| 
      
 103 
     | 
    
         
            +
             
     | 
| 
      
 104 
     | 
    
         
            +
                  def set_ignore_default_scope(model, value)
         
     | 
| 
      
 105 
     | 
    
         
            +
                    set_value_for(@ignore_default_scope, model, value)
         
     | 
| 
      
 106 
     | 
    
         
            +
                  end
         
     | 
| 
      
 107 
     | 
    
         
            +
             
     | 
| 
      
 108 
     | 
    
         
            +
                  def global_current_scope(model, skip_inherited_scope = false)
         
     | 
| 
      
 109 
     | 
    
         
            +
                    value_for(@global_current_scope, model, skip_inherited_scope)
         
     | 
| 
      
 110 
     | 
    
         
            +
                  end
         
     | 
| 
      
 111 
     | 
    
         
            +
             
     | 
| 
      
 112 
     | 
    
         
            +
                  def set_global_current_scope(model, value)
         
     | 
| 
      
 113 
     | 
    
         
            +
                    set_value_for(@global_current_scope, model, value)
         
     | 
| 
      
 114 
     | 
    
         
            +
                  end
         
     | 
| 
       98 
115 
     | 
    
         | 
| 
       99 
     | 
    
         
            -
             
     | 
| 
       100 
     | 
    
         
            -
             
     | 
| 
       101 
     | 
    
         
            -
             
     | 
| 
      
 116 
     | 
    
         
            +
                  private
         
     | 
| 
      
 117 
     | 
    
         
            +
                    # Obtains the value for a given +scope_type+ and +model+.
         
     | 
| 
      
 118 
     | 
    
         
            +
                    def value_for(scope_type, model, skip_inherited_scope = false)
         
     | 
| 
      
 119 
     | 
    
         
            +
                      return scope_type[model.name] if skip_inherited_scope
         
     | 
| 
      
 120 
     | 
    
         
            +
                      klass = model
         
     | 
| 
      
 121 
     | 
    
         
            +
                      base = model.base_class
         
     | 
| 
      
 122 
     | 
    
         
            +
                      while klass <= base
         
     | 
| 
      
 123 
     | 
    
         
            +
                        value = scope_type[klass.name]
         
     | 
| 
      
 124 
     | 
    
         
            +
                        return value if value
         
     | 
| 
      
 125 
     | 
    
         
            +
                        klass = klass.superclass
         
     | 
| 
       102 
126 
     | 
    
         
             
                      end
         
     | 
| 
       103 
127 
     | 
    
         
             
                    end
         
     | 
| 
      
 128 
     | 
    
         
            +
             
     | 
| 
      
 129 
     | 
    
         
            +
                    # Sets the +value+ for a given +scope_type+ and +model+.
         
     | 
| 
      
 130 
     | 
    
         
            +
                    def set_value_for(scope_type, model, value)
         
     | 
| 
      
 131 
     | 
    
         
            +
                      scope_type[model.name] = value
         
     | 
| 
      
 132 
     | 
    
         
            +
                    end
         
     | 
| 
       104 
133 
     | 
    
         
             
                end
         
     | 
| 
       105 
134 
     | 
    
         
             
              end
         
     | 
| 
       106 
135 
     | 
    
         
             
            end
         
     | 
| 
         @@ -2,6 +2,10 @@ 
     | 
|
| 
       2 
2 
     | 
    
         | 
| 
       3 
3 
     | 
    
         
             
            module ActiveRecord
         
     | 
| 
       4 
4 
     | 
    
         
             
              module SecureToken
         
     | 
| 
      
 5 
     | 
    
         
            +
                class MinimumLengthError < StandardError; end
         
     | 
| 
      
 6 
     | 
    
         
            +
             
     | 
| 
      
 7 
     | 
    
         
            +
                MINIMUM_TOKEN_LENGTH = 24
         
     | 
| 
      
 8 
     | 
    
         
            +
             
     | 
| 
       5 
9 
     | 
    
         
             
                extend ActiveSupport::Concern
         
     | 
| 
       6 
10 
     | 
    
         | 
| 
       7 
11 
     | 
    
         
             
                module ClassMethods
         
     | 
| 
         @@ -10,30 +14,34 @@ module ActiveRecord 
     | 
|
| 
       10 
14 
     | 
    
         
             
                  #   # Schema: User(token:string, auth_token:string)
         
     | 
| 
       11 
15 
     | 
    
         
             
                  #   class User < ActiveRecord::Base
         
     | 
| 
       12 
16 
     | 
    
         
             
                  #     has_secure_token
         
     | 
| 
       13 
     | 
    
         
            -
                  #     has_secure_token :auth_token
         
     | 
| 
      
 17 
     | 
    
         
            +
                  #     has_secure_token :auth_token, length: 36
         
     | 
| 
       14 
18 
     | 
    
         
             
                  #   end
         
     | 
| 
       15 
19 
     | 
    
         
             
                  #
         
     | 
| 
       16 
20 
     | 
    
         
             
                  #   user = User.new
         
     | 
| 
       17 
21 
     | 
    
         
             
                  #   user.save
         
     | 
| 
       18 
22 
     | 
    
         
             
                  #   user.token # => "pX27zsMN2ViQKta1bGfLmVJE"
         
     | 
| 
       19 
     | 
    
         
            -
                  #   user.auth_token # => " 
     | 
| 
      
 23 
     | 
    
         
            +
                  #   user.auth_token # => "tU9bLuZseefXQ4yQxQo8wjtBvsAfPc78os6R"
         
     | 
| 
       20 
24 
     | 
    
         
             
                  #   user.regenerate_token # => true
         
     | 
| 
       21 
25 
     | 
    
         
             
                  #   user.regenerate_auth_token # => true
         
     | 
| 
       22 
26 
     | 
    
         
             
                  #
         
     | 
| 
       23 
     | 
    
         
            -
                  # <tt>SecureRandom::base58</tt> is used to generate  
     | 
| 
      
 27 
     | 
    
         
            +
                  # <tt>SecureRandom::base58</tt> is used to generate at minimum a 24-character unique token, so collisions are highly unlikely.
         
     | 
| 
       24 
28 
     | 
    
         
             
                  #
         
     | 
| 
       25 
29 
     | 
    
         
             
                  # Note that it's still possible to generate a race condition in the database in the same way that
         
     | 
| 
       26 
30 
     | 
    
         
             
                  # {validates_uniqueness_of}[rdoc-ref:Validations::ClassMethods#validates_uniqueness_of] can.
         
     | 
| 
       27 
31 
     | 
    
         
             
                  # You're encouraged to add a unique index in the database to deal with this even more unlikely scenario.
         
     | 
| 
       28 
     | 
    
         
            -
                  def has_secure_token(attribute = :token)
         
     | 
| 
      
 32 
     | 
    
         
            +
                  def has_secure_token(attribute = :token, length: MINIMUM_TOKEN_LENGTH)
         
     | 
| 
      
 33 
     | 
    
         
            +
                    if length < MINIMUM_TOKEN_LENGTH
         
     | 
| 
      
 34 
     | 
    
         
            +
                      raise MinimumLengthError, "Token requires a minimum length of #{MINIMUM_TOKEN_LENGTH} characters."
         
     | 
| 
      
 35 
     | 
    
         
            +
                    end
         
     | 
| 
      
 36 
     | 
    
         
            +
             
     | 
| 
       29 
37 
     | 
    
         
             
                    # Load securerandom only when has_secure_token is used.
         
     | 
| 
       30 
38 
     | 
    
         
             
                    require "active_support/core_ext/securerandom"
         
     | 
| 
       31 
     | 
    
         
            -
                    define_method("regenerate_#{attribute}") { update! attribute => self.class.generate_unique_secure_token }
         
     | 
| 
       32 
     | 
    
         
            -
                    before_create { send("#{attribute}=", self.class.generate_unique_secure_token) unless send("#{attribute}?") }
         
     | 
| 
      
 39 
     | 
    
         
            +
                    define_method("regenerate_#{attribute}") { update! attribute => self.class.generate_unique_secure_token(length: length) }
         
     | 
| 
      
 40 
     | 
    
         
            +
                    before_create { send("#{attribute}=", self.class.generate_unique_secure_token(length: length)) unless send("#{attribute}?") }
         
     | 
| 
       33 
41 
     | 
    
         
             
                  end
         
     | 
| 
       34 
42 
     | 
    
         | 
| 
       35 
     | 
    
         
            -
                  def generate_unique_secure_token
         
     | 
| 
       36 
     | 
    
         
            -
                    SecureRandom.base58( 
     | 
| 
      
 43 
     | 
    
         
            +
                  def generate_unique_secure_token(length: MINIMUM_TOKEN_LENGTH)
         
     | 
| 
      
 44 
     | 
    
         
            +
                    SecureRandom.base58(length)
         
     | 
| 
       37 
45 
     | 
    
         
             
                  end
         
     | 
| 
       38 
46 
     | 
    
         
             
                end
         
     | 
| 
       39 
47 
     | 
    
         
             
              end
         
     | 
| 
         @@ -1,6 +1,6 @@ 
     | 
|
| 
       1 
1 
     | 
    
         
             
            # frozen_string_literal: true
         
     | 
| 
       2 
2 
     | 
    
         | 
| 
       3 
     | 
    
         
            -
            module ActiveRecord  
     | 
| 
      
 3 
     | 
    
         
            +
            module ActiveRecord # :nodoc:
         
     | 
| 
       4 
4 
     | 
    
         
             
              # = Active Record \Serialization
         
     | 
| 
       5 
5 
     | 
    
         
             
              module Serialization
         
     | 
| 
       6 
6 
     | 
    
         
             
                extend ActiveSupport::Concern
         
     | 
| 
         @@ -11,10 +11,12 @@ module ActiveRecord #:nodoc: 
     | 
|
| 
       11 
11 
     | 
    
         
             
                end
         
     | 
| 
       12 
12 
     | 
    
         | 
| 
       13 
13 
     | 
    
         
             
                def serializable_hash(options = nil)
         
     | 
| 
       14 
     | 
    
         
            -
                   
     | 
| 
      
 14 
     | 
    
         
            +
                  if self.class._has_attribute?(self.class.inheritance_column)
         
     | 
| 
      
 15 
     | 
    
         
            +
                    options = options ? options.dup : {}
         
     | 
| 
       15 
16 
     | 
    
         | 
| 
       16 
     | 
    
         
            -
             
     | 
| 
       17 
     | 
    
         
            -
             
     | 
| 
      
 17 
     | 
    
         
            +
                    options[:except] = Array(options[:except]).map(&:to_s)
         
     | 
| 
      
 18 
     | 
    
         
            +
                    options[:except] |= Array(self.class.inheritance_column)
         
     | 
| 
      
 19 
     | 
    
         
            +
                  end
         
     | 
| 
       18 
20 
     | 
    
         | 
| 
       19 
21 
     | 
    
         
             
                  super(options)
         
     | 
| 
       20 
22 
     | 
    
         
             
                end
         
     | 
| 
         @@ -0,0 +1,116 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            # frozen_string_literal: true
         
     | 
| 
      
 2 
     | 
    
         
            +
             
     | 
| 
      
 3 
     | 
    
         
            +
            module ActiveRecord
         
     | 
| 
      
 4 
     | 
    
         
            +
              # = Active Record Signed Id
         
     | 
| 
      
 5 
     | 
    
         
            +
              module SignedId
         
     | 
| 
      
 6 
     | 
    
         
            +
                extend ActiveSupport::Concern
         
     | 
| 
      
 7 
     | 
    
         
            +
             
     | 
| 
      
 8 
     | 
    
         
            +
                included do
         
     | 
| 
      
 9 
     | 
    
         
            +
                  ##
         
     | 
| 
      
 10 
     | 
    
         
            +
                  # :singleton-method:
         
     | 
| 
      
 11 
     | 
    
         
            +
                  # Set the secret used for the signed id verifier instance when using Active Record outside of Rails.
         
     | 
| 
      
 12 
     | 
    
         
            +
                  # Within Rails, this is automatically set using the Rails application key generator.
         
     | 
| 
      
 13 
     | 
    
         
            +
                  class_attribute :signed_id_verifier_secret, instance_writer: false
         
     | 
| 
      
 14 
     | 
    
         
            +
                end
         
     | 
| 
      
 15 
     | 
    
         
            +
             
     | 
| 
      
 16 
     | 
    
         
            +
                module ClassMethods
         
     | 
| 
      
 17 
     | 
    
         
            +
                  # Lets you find a record based on a signed id that's safe to put into the world without risk of tampering.
         
     | 
| 
      
 18 
     | 
    
         
            +
                  # This is particularly useful for things like password reset or email verification, where you want
         
     | 
| 
      
 19 
     | 
    
         
            +
                  # the bearer of the signed id to be able to interact with the underlying record, but usually only within
         
     | 
| 
      
 20 
     | 
    
         
            +
                  # a certain time period.
         
     | 
| 
      
 21 
     | 
    
         
            +
                  #
         
     | 
| 
      
 22 
     | 
    
         
            +
                  # You set the time period that the signed id is valid for during generation, using the instance method
         
     | 
| 
      
 23 
     | 
    
         
            +
                  # <tt>signed_id(expires_in: 15.minutes)</tt>. If the time has elapsed before a signed find is attempted,
         
     | 
| 
      
 24 
     | 
    
         
            +
                  # the signed id will no longer be valid, and nil is returned.
         
     | 
| 
      
 25 
     | 
    
         
            +
                  #
         
     | 
| 
      
 26 
     | 
    
         
            +
                  # It's possible to further restrict the use of a signed id with a purpose. This helps when you have a
         
     | 
| 
      
 27 
     | 
    
         
            +
                  # general base model, like a User, which might have signed ids for several things, like password reset
         
     | 
| 
      
 28 
     | 
    
         
            +
                  # or email verification. The purpose that was set during generation must match the purpose set when
         
     | 
| 
      
 29 
     | 
    
         
            +
                  # finding. If there's a mismatch, nil is again returned.
         
     | 
| 
      
 30 
     | 
    
         
            +
                  #
         
     | 
| 
      
 31 
     | 
    
         
            +
                  # ==== Examples
         
     | 
| 
      
 32 
     | 
    
         
            +
                  #
         
     | 
| 
      
 33 
     | 
    
         
            +
                  #   signed_id = User.first.signed_id expires_in: 15.minutes, purpose: :password_reset
         
     | 
| 
      
 34 
     | 
    
         
            +
                  #
         
     | 
| 
      
 35 
     | 
    
         
            +
                  #   User.find_signed signed_id # => nil, since the purpose does not match
         
     | 
| 
      
 36 
     | 
    
         
            +
                  #
         
     | 
| 
      
 37 
     | 
    
         
            +
                  #   travel 16.minutes
         
     | 
| 
      
 38 
     | 
    
         
            +
                  #   User.find_signed signed_id, purpose: :password_reset # => nil, since the signed id has expired
         
     | 
| 
      
 39 
     | 
    
         
            +
                  #
         
     | 
| 
      
 40 
     | 
    
         
            +
                  #   travel_back
         
     | 
| 
      
 41 
     | 
    
         
            +
                  #   User.find_signed signed_id, purpose: :password_reset # => User.first
         
     | 
| 
      
 42 
     | 
    
         
            +
                  def find_signed(signed_id, purpose: nil)
         
     | 
| 
      
 43 
     | 
    
         
            +
                    raise UnknownPrimaryKey.new(self) if primary_key.nil?
         
     | 
| 
      
 44 
     | 
    
         
            +
             
     | 
| 
      
 45 
     | 
    
         
            +
                    if id = signed_id_verifier.verified(signed_id, purpose: combine_signed_id_purposes(purpose))
         
     | 
| 
      
 46 
     | 
    
         
            +
                      find_by primary_key => id
         
     | 
| 
      
 47 
     | 
    
         
            +
                    end
         
     | 
| 
      
 48 
     | 
    
         
            +
                  end
         
     | 
| 
      
 49 
     | 
    
         
            +
             
     | 
| 
      
 50 
     | 
    
         
            +
                  # Works like +find_signed+, but will raise an +ActiveSupport::MessageVerifier::InvalidSignature+
         
     | 
| 
      
 51 
     | 
    
         
            +
                  # exception if the +signed_id+ has either expired, has a purpose mismatch, is for another record,
         
     | 
| 
      
 52 
     | 
    
         
            +
                  # or has been tampered with. It will also raise an +ActiveRecord::RecordNotFound+ exception if
         
     | 
| 
      
 53 
     | 
    
         
            +
                  # the valid signed id can't find a record.
         
     | 
| 
      
 54 
     | 
    
         
            +
                  #
         
     | 
| 
      
 55 
     | 
    
         
            +
                  # === Examples
         
     | 
| 
      
 56 
     | 
    
         
            +
                  #
         
     | 
| 
      
 57 
     | 
    
         
            +
                  #   User.find_signed! "bad data" # => ActiveSupport::MessageVerifier::InvalidSignature
         
     | 
| 
      
 58 
     | 
    
         
            +
                  #
         
     | 
| 
      
 59 
     | 
    
         
            +
                  #   signed_id = User.first.signed_id
         
     | 
| 
      
 60 
     | 
    
         
            +
                  #   User.first.destroy
         
     | 
| 
      
 61 
     | 
    
         
            +
                  #   User.find_signed! signed_id # => ActiveRecord::RecordNotFound
         
     | 
| 
      
 62 
     | 
    
         
            +
                  def find_signed!(signed_id, purpose: nil)
         
     | 
| 
      
 63 
     | 
    
         
            +
                    if id = signed_id_verifier.verify(signed_id, purpose: combine_signed_id_purposes(purpose))
         
     | 
| 
      
 64 
     | 
    
         
            +
                      find(id)
         
     | 
| 
      
 65 
     | 
    
         
            +
                    end
         
     | 
| 
      
 66 
     | 
    
         
            +
                  end
         
     | 
| 
      
 67 
     | 
    
         
            +
             
     | 
| 
      
 68 
     | 
    
         
            +
                  # The verifier instance that all signed ids are generated and verified from. By default, it'll be initialized
         
     | 
| 
      
 69 
     | 
    
         
            +
                  # with the class-level +signed_id_verifier_secret+, which within Rails comes from the
         
     | 
| 
      
 70 
     | 
    
         
            +
                  # Rails.application.key_generator. By default, it's SHA256 for the digest and JSON for the serialization.
         
     | 
| 
      
 71 
     | 
    
         
            +
                  def signed_id_verifier
         
     | 
| 
      
 72 
     | 
    
         
            +
                    @signed_id_verifier ||= begin
         
     | 
| 
      
 73 
     | 
    
         
            +
                      secret = signed_id_verifier_secret
         
     | 
| 
      
 74 
     | 
    
         
            +
                      secret = secret.call if secret.respond_to?(:call)
         
     | 
| 
      
 75 
     | 
    
         
            +
             
     | 
| 
      
 76 
     | 
    
         
            +
                      if secret.nil?
         
     | 
| 
      
 77 
     | 
    
         
            +
                        raise ArgumentError, "You must set ActiveRecord::Base.signed_id_verifier_secret to use signed ids"
         
     | 
| 
      
 78 
     | 
    
         
            +
                      else
         
     | 
| 
      
 79 
     | 
    
         
            +
                        ActiveSupport::MessageVerifier.new secret, digest: "SHA256", serializer: JSON
         
     | 
| 
      
 80 
     | 
    
         
            +
                      end
         
     | 
| 
      
 81 
     | 
    
         
            +
                    end
         
     | 
| 
      
 82 
     | 
    
         
            +
                  end
         
     | 
| 
      
 83 
     | 
    
         
            +
             
     | 
| 
      
 84 
     | 
    
         
            +
                  # Allows you to pass in a custom verifier used for the signed ids. This also allows you to use different
         
     | 
| 
      
 85 
     | 
    
         
            +
                  # verifiers for different classes. This is also helpful if you need to rotate keys, as you can prepare
         
     | 
| 
      
 86 
     | 
    
         
            +
                  # your custom verifier for that in advance. See +ActiveSupport::MessageVerifier+ for details.
         
     | 
| 
      
 87 
     | 
    
         
            +
                  def signed_id_verifier=(verifier)
         
     | 
| 
      
 88 
     | 
    
         
            +
                    @signed_id_verifier = verifier
         
     | 
| 
      
 89 
     | 
    
         
            +
                  end
         
     | 
| 
      
 90 
     | 
    
         
            +
             
     | 
| 
      
 91 
     | 
    
         
            +
                  # :nodoc:
         
     | 
| 
      
 92 
     | 
    
         
            +
                  def combine_signed_id_purposes(purpose)
         
     | 
| 
      
 93 
     | 
    
         
            +
                    [ base_class.name.underscore, purpose.to_s ].compact_blank.join("/")
         
     | 
| 
      
 94 
     | 
    
         
            +
                  end
         
     | 
| 
      
 95 
     | 
    
         
            +
                end
         
     | 
| 
      
 96 
     | 
    
         
            +
             
     | 
| 
      
 97 
     | 
    
         
            +
             
     | 
| 
      
 98 
     | 
    
         
            +
                # Returns a signed id that's generated using a preconfigured +ActiveSupport::MessageVerifier+ instance.
         
     | 
| 
      
 99 
     | 
    
         
            +
                # This signed id is tamper proof, so it's safe to send in an email or otherwise share with the outside world.
         
     | 
| 
      
 100 
     | 
    
         
            +
                # It can further more be set to expire (the default is not to expire), and scoped down with a specific purpose.
         
     | 
| 
      
 101 
     | 
    
         
            +
                # If the expiration date has been exceeded before +find_signed+ is called, the id won't find the designated
         
     | 
| 
      
 102 
     | 
    
         
            +
                # record. If a purpose is set, this too must match.
         
     | 
| 
      
 103 
     | 
    
         
            +
                #
         
     | 
| 
      
 104 
     | 
    
         
            +
                # If you accidentally let a signed id out in the wild that you wish to retract sooner than its expiration date
         
     | 
| 
      
 105 
     | 
    
         
            +
                # (or maybe you forgot to set an expiration date while meaning to!), you can use the purpose to essentially
         
     | 
| 
      
 106 
     | 
    
         
            +
                # version the signed_id, like so:
         
     | 
| 
      
 107 
     | 
    
         
            +
                #
         
     | 
| 
      
 108 
     | 
    
         
            +
                #   user.signed_id purpose: :v2
         
     | 
| 
      
 109 
     | 
    
         
            +
                #
         
     | 
| 
      
 110 
     | 
    
         
            +
                # And you then change your +find_signed+ calls to require this new purpose. Any old signed ids that were not
         
     | 
| 
      
 111 
     | 
    
         
            +
                # created with the purpose will no longer find the record.
         
     | 
| 
      
 112 
     | 
    
         
            +
                def signed_id(expires_in: nil, purpose: nil)
         
     | 
| 
      
 113 
     | 
    
         
            +
                  self.class.signed_id_verifier.generate id, expires_in: expires_in, purpose: self.class.combine_signed_id_purposes(purpose)
         
     | 
| 
      
 114 
     | 
    
         
            +
                end
         
     | 
| 
      
 115 
     | 
    
         
            +
              end
         
     | 
| 
      
 116 
     | 
    
         
            +
            end
         
     |