activerecord 3.2.22.5 → 5.2.8
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 +5 -5
- data/CHANGELOG.md +657 -621
- data/MIT-LICENSE +2 -2
- data/README.rdoc +41 -46
- data/examples/performance.rb +55 -42
- data/examples/simple.rb +6 -5
- data/lib/active_record/aggregations.rb +264 -236
- data/lib/active_record/association_relation.rb +40 -0
- data/lib/active_record/associations/alias_tracker.rb +47 -42
- data/lib/active_record/associations/association.rb +127 -75
- data/lib/active_record/associations/association_scope.rb +126 -92
- data/lib/active_record/associations/belongs_to_association.rb +78 -27
- data/lib/active_record/associations/belongs_to_polymorphic_association.rb +9 -4
- data/lib/active_record/associations/builder/association.rb +117 -32
- data/lib/active_record/associations/builder/belongs_to.rb +135 -60
- data/lib/active_record/associations/builder/collection_association.rb +61 -54
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +120 -42
- data/lib/active_record/associations/builder/has_many.rb +10 -64
- data/lib/active_record/associations/builder/has_one.rb +19 -51
- data/lib/active_record/associations/builder/singular_association.rb +28 -18
- data/lib/active_record/associations/collection_association.rb +226 -293
- data/lib/active_record/associations/collection_proxy.rb +1067 -69
- data/lib/active_record/associations/foreign_association.rb +13 -0
- data/lib/active_record/associations/has_many_association.rb +83 -47
- data/lib/active_record/associations/has_many_through_association.rb +98 -65
- data/lib/active_record/associations/has_one_association.rb +57 -20
- data/lib/active_record/associations/has_one_through_association.rb +18 -9
- data/lib/active_record/associations/join_dependency/join_association.rb +48 -126
- data/lib/active_record/associations/join_dependency/join_base.rb +11 -12
- data/lib/active_record/associations/join_dependency/join_part.rb +35 -42
- data/lib/active_record/associations/join_dependency.rb +212 -164
- data/lib/active_record/associations/preloader/association.rb +95 -89
- data/lib/active_record/associations/preloader/through_association.rb +84 -44
- data/lib/active_record/associations/preloader.rb +123 -111
- data/lib/active_record/associations/singular_association.rb +33 -24
- data/lib/active_record/associations/through_association.rb +60 -26
- data/lib/active_record/associations.rb +1759 -1506
- data/lib/active_record/attribute_assignment.rb +60 -193
- data/lib/active_record/attribute_decorators.rb +90 -0
- data/lib/active_record/attribute_methods/before_type_cast.rb +55 -8
- data/lib/active_record/attribute_methods/dirty.rb +113 -74
- data/lib/active_record/attribute_methods/primary_key.rb +106 -77
- data/lib/active_record/attribute_methods/query.rb +8 -5
- data/lib/active_record/attribute_methods/read.rb +63 -114
- data/lib/active_record/attribute_methods/serialization.rb +60 -90
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +69 -43
- data/lib/active_record/attribute_methods/write.rb +43 -45
- data/lib/active_record/attribute_methods.rb +366 -149
- data/lib/active_record/attributes.rb +266 -0
- data/lib/active_record/autosave_association.rb +312 -225
- data/lib/active_record/base.rb +114 -505
- data/lib/active_record/callbacks.rb +145 -67
- data/lib/active_record/coders/json.rb +15 -0
- data/lib/active_record/coders/yaml_column.rb +32 -23
- data/lib/active_record/collection_cache_key.rb +53 -0
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +883 -284
- data/lib/active_record/connection_adapters/abstract/database_limits.rb +16 -2
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +350 -200
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +82 -27
- data/lib/active_record/connection_adapters/abstract/quoting.rb +150 -65
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +23 -0
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +146 -0
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +477 -284
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +95 -0
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +1100 -310
- data/lib/active_record/connection_adapters/abstract/transaction.rb +283 -0
- data/lib/active_record/connection_adapters/abstract_adapter.rb +450 -118
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +657 -446
- data/lib/active_record/connection_adapters/column.rb +50 -255
- data/lib/active_record/connection_adapters/connection_specification.rb +287 -0
- data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +33 -0
- data/lib/active_record/connection_adapters/mysql/column.rb +27 -0
- data/lib/active_record/connection_adapters/mysql/database_statements.rb +140 -0
- data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +72 -0
- data/lib/active_record/connection_adapters/mysql/quoting.rb +44 -0
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +73 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +87 -0
- data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +80 -0
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +148 -0
- data/lib/active_record/connection_adapters/mysql/type_metadata.rb +35 -0
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +59 -210
- data/lib/active_record/connection_adapters/postgresql/column.rb +44 -0
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +163 -0
- data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +92 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +56 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +17 -0
- data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +21 -0
- data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +71 -0
- data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +45 -0
- data/lib/active_record/connection_adapters/postgresql/oid/money.rb +41 -0
- data/lib/active_record/connection_adapters/postgresql/oid/oid.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +65 -0
- data/lib/active_record/connection_adapters/postgresql/oid/range.rb +97 -0
- data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +18 -0
- data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +111 -0
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +28 -0
- data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +30 -0
- data/lib/active_record/connection_adapters/postgresql/oid.rb +34 -0
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +168 -0
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +43 -0
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +65 -0
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +206 -0
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +774 -0
- data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +39 -0
- data/lib/active_record/connection_adapters/postgresql/utils.rb +81 -0
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +620 -1080
- data/lib/active_record/connection_adapters/schema_cache.rb +85 -36
- data/lib/active_record/connection_adapters/sql_type_metadata.rb +34 -0
- data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +67 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +17 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +19 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +18 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +106 -0
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +545 -27
- data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
- data/lib/active_record/connection_handling.rb +145 -0
- data/lib/active_record/core.rb +559 -0
- data/lib/active_record/counter_cache.rb +200 -105
- data/lib/active_record/define_callbacks.rb +22 -0
- data/lib/active_record/dynamic_matchers.rb +107 -69
- data/lib/active_record/enum.rb +244 -0
- data/lib/active_record/errors.rb +245 -60
- data/lib/active_record/explain.rb +35 -71
- data/lib/active_record/explain_registry.rb +32 -0
- data/lib/active_record/explain_subscriber.rb +18 -9
- data/lib/active_record/fixture_set/file.rb +82 -0
- data/lib/active_record/fixtures.rb +418 -275
- data/lib/active_record/gem_version.rb +17 -0
- data/lib/active_record/inheritance.rb +209 -100
- data/lib/active_record/integration.rb +116 -21
- data/lib/active_record/internal_metadata.rb +45 -0
- data/lib/active_record/legacy_yaml_adapter.rb +48 -0
- data/lib/active_record/locale/en.yml +9 -1
- data/lib/active_record/locking/optimistic.rb +107 -94
- data/lib/active_record/locking/pessimistic.rb +20 -8
- data/lib/active_record/log_subscriber.rb +99 -34
- data/lib/active_record/migration/command_recorder.rb +199 -64
- data/lib/active_record/migration/compatibility.rb +217 -0
- data/lib/active_record/migration/join_table.rb +17 -0
- data/lib/active_record/migration.rb +893 -296
- data/lib/active_record/model_schema.rb +328 -175
- data/lib/active_record/nested_attributes.rb +338 -242
- data/lib/active_record/no_touching.rb +58 -0
- data/lib/active_record/null_relation.rb +68 -0
- data/lib/active_record/persistence.rb +557 -170
- data/lib/active_record/query_cache.rb +14 -43
- data/lib/active_record/querying.rb +36 -24
- data/lib/active_record/railtie.rb +147 -52
- data/lib/active_record/railties/console_sandbox.rb +5 -4
- data/lib/active_record/railties/controller_runtime.rb +13 -6
- data/lib/active_record/railties/databases.rake +206 -488
- data/lib/active_record/readonly_attributes.rb +4 -6
- data/lib/active_record/reflection.rb +734 -228
- data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
- data/lib/active_record/relation/batches.rb +249 -52
- data/lib/active_record/relation/calculations.rb +330 -284
- data/lib/active_record/relation/delegation.rb +135 -37
- data/lib/active_record/relation/finder_methods.rb +450 -287
- data/lib/active_record/relation/from_clause.rb +26 -0
- data/lib/active_record/relation/merger.rb +193 -0
- data/lib/active_record/relation/predicate_builder/array_handler.rb +48 -0
- data/lib/active_record/relation/predicate_builder/association_query_value.rb +46 -0
- data/lib/active_record/relation/predicate_builder/base_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +20 -0
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +56 -0
- data/lib/active_record/relation/predicate_builder/range_handler.rb +42 -0
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder.rb +132 -43
- data/lib/active_record/relation/query_attribute.rb +45 -0
- data/lib/active_record/relation/query_methods.rb +1037 -221
- data/lib/active_record/relation/record_fetch_warning.rb +51 -0
- data/lib/active_record/relation/spawn_methods.rb +48 -151
- data/lib/active_record/relation/where_clause.rb +186 -0
- data/lib/active_record/relation/where_clause_factory.rb +34 -0
- data/lib/active_record/relation.rb +451 -359
- data/lib/active_record/result.rb +129 -20
- data/lib/active_record/runtime_registry.rb +24 -0
- data/lib/active_record/sanitization.rb +164 -136
- data/lib/active_record/schema.rb +31 -19
- data/lib/active_record/schema_dumper.rb +154 -107
- data/lib/active_record/schema_migration.rb +56 -0
- data/lib/active_record/scoping/default.rb +108 -98
- data/lib/active_record/scoping/named.rb +125 -112
- data/lib/active_record/scoping.rb +77 -123
- data/lib/active_record/secure_token.rb +40 -0
- data/lib/active_record/serialization.rb +10 -6
- data/lib/active_record/statement_cache.rb +121 -0
- data/lib/active_record/store.rb +175 -16
- data/lib/active_record/suppressor.rb +61 -0
- data/lib/active_record/table_metadata.rb +82 -0
- data/lib/active_record/tasks/database_tasks.rb +337 -0
- data/lib/active_record/tasks/mysql_database_tasks.rb +115 -0
- data/lib/active_record/tasks/postgresql_database_tasks.rb +143 -0
- data/lib/active_record/tasks/sqlite_database_tasks.rb +83 -0
- data/lib/active_record/timestamp.rb +80 -41
- data/lib/active_record/touch_later.rb +64 -0
- data/lib/active_record/transactions.rb +240 -119
- data/lib/active_record/translation.rb +2 -0
- data/lib/active_record/type/adapter_specific_registry.rb +136 -0
- data/lib/active_record/type/date.rb +9 -0
- data/lib/active_record/type/date_time.rb +9 -0
- data/lib/active_record/type/decimal_without_scale.rb +15 -0
- data/lib/active_record/type/hash_lookup_type_map.rb +25 -0
- data/lib/active_record/type/internal/timezone.rb +17 -0
- data/lib/active_record/type/json.rb +30 -0
- data/lib/active_record/type/serialized.rb +71 -0
- data/lib/active_record/type/text.rb +11 -0
- data/lib/active_record/type/time.rb +21 -0
- data/lib/active_record/type/type_map.rb +62 -0
- data/lib/active_record/type/unsigned_integer.rb +17 -0
- data/lib/active_record/type.rb +79 -0
- data/lib/active_record/type_caster/connection.rb +33 -0
- data/lib/active_record/type_caster/map.rb +23 -0
- data/lib/active_record/type_caster.rb +9 -0
- data/lib/active_record/validations/absence.rb +25 -0
- data/lib/active_record/validations/associated.rb +35 -18
- data/lib/active_record/validations/length.rb +26 -0
- data/lib/active_record/validations/presence.rb +68 -0
- data/lib/active_record/validations/uniqueness.rb +133 -75
- data/lib/active_record/validations.rb +53 -43
- data/lib/active_record/version.rb +7 -7
- data/lib/active_record.rb +89 -57
- data/lib/rails/generators/active_record/application_record/application_record_generator.rb +27 -0
- data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +5 -0
- data/lib/rails/generators/active_record/migration/migration_generator.rb +61 -8
- data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +24 -0
- data/lib/rails/generators/active_record/migration/templates/migration.rb.tt +46 -0
- data/lib/rails/generators/active_record/migration.rb +28 -8
- data/lib/rails/generators/active_record/model/model_generator.rb +23 -22
- data/lib/rails/generators/active_record/model/templates/model.rb.tt +13 -0
- data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +1 -1
- data/lib/rails/generators/active_record.rb +10 -16
- metadata +141 -62
- data/examples/associations.png +0 -0
- data/lib/active_record/associations/has_and_belongs_to_many_association.rb +0 -63
- data/lib/active_record/associations/join_helper.rb +0 -55
- data/lib/active_record/associations/preloader/belongs_to.rb +0 -17
- data/lib/active_record/associations/preloader/collection_association.rb +0 -24
- data/lib/active_record/associations/preloader/has_and_belongs_to_many.rb +0 -60
- data/lib/active_record/associations/preloader/has_many.rb +0 -17
- data/lib/active_record/associations/preloader/has_many_through.rb +0 -15
- data/lib/active_record/associations/preloader/has_one.rb +0 -23
- data/lib/active_record/associations/preloader/has_one_through.rb +0 -9
- data/lib/active_record/associations/preloader/singular_association.rb +0 -21
- data/lib/active_record/attribute_methods/deprecated_underscore_read.rb +0 -32
- data/lib/active_record/connection_adapters/abstract/connection_specification.rb +0 -191
- data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -441
- data/lib/active_record/connection_adapters/sqlite_adapter.rb +0 -583
- data/lib/active_record/dynamic_finder_match.rb +0 -68
- data/lib/active_record/dynamic_scope_match.rb +0 -23
- data/lib/active_record/fixtures/file.rb +0 -65
- data/lib/active_record/identity_map.rb +0 -162
- data/lib/active_record/observer.rb +0 -121
- data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
- data/lib/active_record/serializers/xml_serializer.rb +0 -203
- data/lib/active_record/session_store.rb +0 -360
- data/lib/active_record/test_case.rb +0 -73
- data/lib/rails/generators/active_record/migration/templates/migration.rb +0 -34
- data/lib/rails/generators/active_record/model/templates/migration.rb +0 -15
- data/lib/rails/generators/active_record/model/templates/model.rb +0 -12
- data/lib/rails/generators/active_record/observer/observer_generator.rb +0 -15
- data/lib/rails/generators/active_record/observer/templates/observer.rb +0 -4
- data/lib/rails/generators/active_record/session_migration/session_migration_generator.rb +0 -25
- data/lib/rails/generators/active_record/session_migration/templates/migration.rb +0 -12
| @@ -0,0 +1,136 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "active_model/type/registry"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            module ActiveRecord
         | 
| 6 | 
            +
              # :stopdoc:
         | 
| 7 | 
            +
              module Type
         | 
| 8 | 
            +
                class AdapterSpecificRegistry < ActiveModel::Type::Registry
         | 
| 9 | 
            +
                  def add_modifier(options, klass, **args)
         | 
| 10 | 
            +
                    registrations << DecorationRegistration.new(options, klass, **args)
         | 
| 11 | 
            +
                  end
         | 
| 12 | 
            +
             | 
| 13 | 
            +
                  private
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                    def registration_klass
         | 
| 16 | 
            +
                      Registration
         | 
| 17 | 
            +
                    end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                    def find_registration(symbol, *args)
         | 
| 20 | 
            +
                      registrations
         | 
| 21 | 
            +
                        .select { |registration| registration.matches?(symbol, *args) }
         | 
| 22 | 
            +
                        .max
         | 
| 23 | 
            +
                    end
         | 
| 24 | 
            +
                end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                class Registration
         | 
| 27 | 
            +
                  def initialize(name, block, adapter: nil, override: nil)
         | 
| 28 | 
            +
                    @name = name
         | 
| 29 | 
            +
                    @block = block
         | 
| 30 | 
            +
                    @adapter = adapter
         | 
| 31 | 
            +
                    @override = override
         | 
| 32 | 
            +
                  end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                  def call(_registry, *args, adapter: nil, **kwargs)
         | 
| 35 | 
            +
                    if kwargs.any? # https://bugs.ruby-lang.org/issues/10856
         | 
| 36 | 
            +
                      block.call(*args, **kwargs)
         | 
| 37 | 
            +
                    else
         | 
| 38 | 
            +
                      block.call(*args)
         | 
| 39 | 
            +
                    end
         | 
| 40 | 
            +
                  end
         | 
| 41 | 
            +
             | 
| 42 | 
            +
                  def matches?(type_name, *args, **kwargs)
         | 
| 43 | 
            +
                    type_name == name && matches_adapter?(**kwargs)
         | 
| 44 | 
            +
                  end
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                  def <=>(other)
         | 
| 47 | 
            +
                    if conflicts_with?(other)
         | 
| 48 | 
            +
                      raise TypeConflictError.new("Type #{name} was registered for all
         | 
| 49 | 
            +
                                                  adapters, but shadows a native type with
         | 
| 50 | 
            +
                                                  the same name for #{other.adapter}".squish)
         | 
| 51 | 
            +
                    end
         | 
| 52 | 
            +
                    priority <=> other.priority
         | 
| 53 | 
            +
                  end
         | 
| 54 | 
            +
             | 
| 55 | 
            +
                  # TODO Change this to private once we've dropped Ruby 2.2 support.
         | 
| 56 | 
            +
                  # Workaround for Ruby 2.2 "private attribute?" warning.
         | 
| 57 | 
            +
                  protected
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                    attr_reader :name, :block, :adapter, :override
         | 
| 60 | 
            +
             | 
| 61 | 
            +
                    def priority
         | 
| 62 | 
            +
                      result = 0
         | 
| 63 | 
            +
                      if adapter
         | 
| 64 | 
            +
                        result |= 1
         | 
| 65 | 
            +
                      end
         | 
| 66 | 
            +
                      if override
         | 
| 67 | 
            +
                        result |= 2
         | 
| 68 | 
            +
                      end
         | 
| 69 | 
            +
                      result
         | 
| 70 | 
            +
                    end
         | 
| 71 | 
            +
             | 
| 72 | 
            +
                    def priority_except_adapter
         | 
| 73 | 
            +
                      priority & 0b111111100
         | 
| 74 | 
            +
                    end
         | 
| 75 | 
            +
             | 
| 76 | 
            +
                  private
         | 
| 77 | 
            +
             | 
| 78 | 
            +
                    def matches_adapter?(adapter: nil, **)
         | 
| 79 | 
            +
                      (self.adapter.nil? || adapter == self.adapter)
         | 
| 80 | 
            +
                    end
         | 
| 81 | 
            +
             | 
| 82 | 
            +
                    def conflicts_with?(other)
         | 
| 83 | 
            +
                      same_priority_except_adapter?(other) &&
         | 
| 84 | 
            +
                        has_adapter_conflict?(other)
         | 
| 85 | 
            +
                    end
         | 
| 86 | 
            +
             | 
| 87 | 
            +
                    def same_priority_except_adapter?(other)
         | 
| 88 | 
            +
                      priority_except_adapter == other.priority_except_adapter
         | 
| 89 | 
            +
                    end
         | 
| 90 | 
            +
             | 
| 91 | 
            +
                    def has_adapter_conflict?(other)
         | 
| 92 | 
            +
                      (override.nil? && other.adapter) ||
         | 
| 93 | 
            +
                        (adapter && other.override.nil?)
         | 
| 94 | 
            +
                    end
         | 
| 95 | 
            +
                end
         | 
| 96 | 
            +
             | 
| 97 | 
            +
                class DecorationRegistration < Registration
         | 
| 98 | 
            +
                  def initialize(options, klass, adapter: nil)
         | 
| 99 | 
            +
                    @options = options
         | 
| 100 | 
            +
                    @klass = klass
         | 
| 101 | 
            +
                    @adapter = adapter
         | 
| 102 | 
            +
                  end
         | 
| 103 | 
            +
             | 
| 104 | 
            +
                  def call(registry, *args, **kwargs)
         | 
| 105 | 
            +
                    subtype = registry.lookup(*args, **kwargs.except(*options.keys))
         | 
| 106 | 
            +
                    klass.new(subtype)
         | 
| 107 | 
            +
                  end
         | 
| 108 | 
            +
             | 
| 109 | 
            +
                  def matches?(*args, **kwargs)
         | 
| 110 | 
            +
                    matches_adapter?(**kwargs) && matches_options?(**kwargs)
         | 
| 111 | 
            +
                  end
         | 
| 112 | 
            +
             | 
| 113 | 
            +
                  def priority
         | 
| 114 | 
            +
                    super | 4
         | 
| 115 | 
            +
                  end
         | 
| 116 | 
            +
             | 
| 117 | 
            +
                  # TODO Change this to private once we've dropped Ruby 2.2 support.
         | 
| 118 | 
            +
                  # Workaround for Ruby 2.2 "private attribute?" warning.
         | 
| 119 | 
            +
                  protected
         | 
| 120 | 
            +
             | 
| 121 | 
            +
                    attr_reader :options, :klass
         | 
| 122 | 
            +
             | 
| 123 | 
            +
                  private
         | 
| 124 | 
            +
             | 
| 125 | 
            +
                    def matches_options?(**kwargs)
         | 
| 126 | 
            +
                      options.all? do |key, value|
         | 
| 127 | 
            +
                        kwargs[key] == value
         | 
| 128 | 
            +
                      end
         | 
| 129 | 
            +
                    end
         | 
| 130 | 
            +
                end
         | 
| 131 | 
            +
              end
         | 
| 132 | 
            +
             | 
| 133 | 
            +
              class TypeConflictError < StandardError
         | 
| 134 | 
            +
              end
         | 
| 135 | 
            +
              # :startdoc:
         | 
| 136 | 
            +
            end
         | 
| @@ -0,0 +1,15 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class DecimalWithoutScale < ActiveModel::Type::BigInteger # :nodoc:
         | 
| 6 | 
            +
                  def type
         | 
| 7 | 
            +
                    :decimal
         | 
| 8 | 
            +
                  end
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                  def type_cast_for_schema(value)
         | 
| 11 | 
            +
                    value.to_s.inspect
         | 
| 12 | 
            +
                  end
         | 
| 13 | 
            +
                end
         | 
| 14 | 
            +
              end
         | 
| 15 | 
            +
            end
         | 
| @@ -0,0 +1,25 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class HashLookupTypeMap < TypeMap # :nodoc:
         | 
| 6 | 
            +
                  def alias_type(type, alias_type)
         | 
| 7 | 
            +
                    register_type(type) { |_, *args| lookup(alias_type, *args) }
         | 
| 8 | 
            +
                  end
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                  def key?(key)
         | 
| 11 | 
            +
                    @mapping.key?(key)
         | 
| 12 | 
            +
                  end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                  def keys
         | 
| 15 | 
            +
                    @mapping.keys
         | 
| 16 | 
            +
                  end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  private
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                    def perform_fetch(type, *args, &block)
         | 
| 21 | 
            +
                      @mapping.fetch(type, block).call(type, *args)
         | 
| 22 | 
            +
                    end
         | 
| 23 | 
            +
                end
         | 
| 24 | 
            +
              end
         | 
| 25 | 
            +
            end
         | 
| @@ -0,0 +1,17 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                module Internal
         | 
| 6 | 
            +
                  module Timezone
         | 
| 7 | 
            +
                    def is_utc?
         | 
| 8 | 
            +
                      ActiveRecord::Base.default_timezone == :utc
         | 
| 9 | 
            +
                    end
         | 
| 10 | 
            +
             | 
| 11 | 
            +
                    def default_timezone
         | 
| 12 | 
            +
                      ActiveRecord::Base.default_timezone
         | 
| 13 | 
            +
                    end
         | 
| 14 | 
            +
                  end
         | 
| 15 | 
            +
                end
         | 
| 16 | 
            +
              end
         | 
| 17 | 
            +
            end
         | 
| @@ -0,0 +1,30 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class Json < ActiveModel::Type::Value
         | 
| 6 | 
            +
                  include ActiveModel::Type::Helpers::Mutable
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  def type
         | 
| 9 | 
            +
                    :json
         | 
| 10 | 
            +
                  end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  def deserialize(value)
         | 
| 13 | 
            +
                    return value unless value.is_a?(::String)
         | 
| 14 | 
            +
                    ActiveSupport::JSON.decode(value) rescue nil
         | 
| 15 | 
            +
                  end
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                  def serialize(value)
         | 
| 18 | 
            +
                    ActiveSupport::JSON.encode(value) unless value.nil?
         | 
| 19 | 
            +
                  end
         | 
| 20 | 
            +
             | 
| 21 | 
            +
                  def changed_in_place?(raw_old_value, new_value)
         | 
| 22 | 
            +
                    deserialize(raw_old_value) != new_value
         | 
| 23 | 
            +
                  end
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                  def accessor
         | 
| 26 | 
            +
                    ActiveRecord::Store::StringKeyedHashAccessor
         | 
| 27 | 
            +
                  end
         | 
| 28 | 
            +
                end
         | 
| 29 | 
            +
              end
         | 
| 30 | 
            +
            end
         | 
| @@ -0,0 +1,71 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class Serialized < DelegateClass(ActiveModel::Type::Value) # :nodoc:
         | 
| 6 | 
            +
                  undef to_yaml if method_defined?(:to_yaml)
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  include ActiveModel::Type::Helpers::Mutable
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                  attr_reader :subtype, :coder
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  def initialize(subtype, coder)
         | 
| 13 | 
            +
                    @subtype = subtype
         | 
| 14 | 
            +
                    @coder = coder
         | 
| 15 | 
            +
                    super(subtype)
         | 
| 16 | 
            +
                  end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  def deserialize(value)
         | 
| 19 | 
            +
                    if default_value?(value)
         | 
| 20 | 
            +
                      value
         | 
| 21 | 
            +
                    else
         | 
| 22 | 
            +
                      coder.load(super)
         | 
| 23 | 
            +
                    end
         | 
| 24 | 
            +
                  end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                  def serialize(value)
         | 
| 27 | 
            +
                    return if value.nil?
         | 
| 28 | 
            +
                    unless default_value?(value)
         | 
| 29 | 
            +
                      super coder.dump(value)
         | 
| 30 | 
            +
                    end
         | 
| 31 | 
            +
                  end
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                  def inspect
         | 
| 34 | 
            +
                    Kernel.instance_method(:inspect).bind(self).call
         | 
| 35 | 
            +
                  end
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                  def changed_in_place?(raw_old_value, value)
         | 
| 38 | 
            +
                    return false if value.nil?
         | 
| 39 | 
            +
                    raw_new_value = encoded(value)
         | 
| 40 | 
            +
                    raw_old_value.nil? != raw_new_value.nil? ||
         | 
| 41 | 
            +
                      subtype.changed_in_place?(raw_old_value, raw_new_value)
         | 
| 42 | 
            +
                  end
         | 
| 43 | 
            +
             | 
| 44 | 
            +
                  def accessor
         | 
| 45 | 
            +
                    ActiveRecord::Store::IndifferentHashAccessor
         | 
| 46 | 
            +
                  end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                  def assert_valid_value(value)
         | 
| 49 | 
            +
                    if coder.respond_to?(:assert_valid_value)
         | 
| 50 | 
            +
                      coder.assert_valid_value(value, action: "serialize")
         | 
| 51 | 
            +
                    end
         | 
| 52 | 
            +
                  end
         | 
| 53 | 
            +
             | 
| 54 | 
            +
                  def force_equality?(value)
         | 
| 55 | 
            +
                    coder.respond_to?(:object_class) && value.is_a?(coder.object_class)
         | 
| 56 | 
            +
                  end
         | 
| 57 | 
            +
             | 
| 58 | 
            +
                  private
         | 
| 59 | 
            +
             | 
| 60 | 
            +
                    def default_value?(value)
         | 
| 61 | 
            +
                      value == coder.load(nil)
         | 
| 62 | 
            +
                    end
         | 
| 63 | 
            +
             | 
| 64 | 
            +
                    def encoded(value)
         | 
| 65 | 
            +
                      unless default_value?(value)
         | 
| 66 | 
            +
                        coder.dump(value)
         | 
| 67 | 
            +
                      end
         | 
| 68 | 
            +
                    end
         | 
| 69 | 
            +
                end
         | 
| 70 | 
            +
              end
         | 
| 71 | 
            +
            end
         | 
| @@ -0,0 +1,21 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class Time < ActiveModel::Type::Time
         | 
| 6 | 
            +
                  include Internal::Timezone
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  class Value < DelegateClass(::Time) # :nodoc:
         | 
| 9 | 
            +
                  end
         | 
| 10 | 
            +
             | 
| 11 | 
            +
                  def serialize(value)
         | 
| 12 | 
            +
                    case value = super
         | 
| 13 | 
            +
                    when ::Time
         | 
| 14 | 
            +
                      Value.new(value)
         | 
| 15 | 
            +
                    else
         | 
| 16 | 
            +
                      value
         | 
| 17 | 
            +
                    end
         | 
| 18 | 
            +
                  end
         | 
| 19 | 
            +
                end
         | 
| 20 | 
            +
              end
         | 
| 21 | 
            +
            end
         | 
| @@ -0,0 +1,62 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "concurrent/map"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            module ActiveRecord
         | 
| 6 | 
            +
              module Type
         | 
| 7 | 
            +
                class TypeMap # :nodoc:
         | 
| 8 | 
            +
                  def initialize
         | 
| 9 | 
            +
                    @mapping = {}
         | 
| 10 | 
            +
                    @cache = Concurrent::Map.new do |h, key|
         | 
| 11 | 
            +
                      h.fetch_or_store(key, Concurrent::Map.new)
         | 
| 12 | 
            +
                    end
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                  def lookup(lookup_key, *args)
         | 
| 16 | 
            +
                    fetch(lookup_key, *args) { Type.default_value }
         | 
| 17 | 
            +
                  end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                  def fetch(lookup_key, *args, &block)
         | 
| 20 | 
            +
                    @cache[lookup_key].fetch_or_store(args) do
         | 
| 21 | 
            +
                      perform_fetch(lookup_key, *args, &block)
         | 
| 22 | 
            +
                    end
         | 
| 23 | 
            +
                  end
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                  def register_type(key, value = nil, &block)
         | 
| 26 | 
            +
                    raise ::ArgumentError unless value || block
         | 
| 27 | 
            +
                    @cache.clear
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                    if block
         | 
| 30 | 
            +
                      @mapping[key] = block
         | 
| 31 | 
            +
                    else
         | 
| 32 | 
            +
                      @mapping[key] = proc { value }
         | 
| 33 | 
            +
                    end
         | 
| 34 | 
            +
                  end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
                  def alias_type(key, target_key)
         | 
| 37 | 
            +
                    register_type(key) do |sql_type, *args|
         | 
| 38 | 
            +
                      metadata = sql_type[/\(.*\)/, 0]
         | 
| 39 | 
            +
                      lookup("#{target_key}#{metadata}", *args)
         | 
| 40 | 
            +
                    end
         | 
| 41 | 
            +
                  end
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                  def clear
         | 
| 44 | 
            +
                    @mapping.clear
         | 
| 45 | 
            +
                  end
         | 
| 46 | 
            +
             | 
| 47 | 
            +
                  private
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                    def perform_fetch(lookup_key, *args)
         | 
| 50 | 
            +
                      matching_pair = @mapping.reverse_each.detect do |key, _|
         | 
| 51 | 
            +
                        key === lookup_key
         | 
| 52 | 
            +
                      end
         | 
| 53 | 
            +
             | 
| 54 | 
            +
                      if matching_pair
         | 
| 55 | 
            +
                        matching_pair.last.call(lookup_key, *args)
         | 
| 56 | 
            +
                      else
         | 
| 57 | 
            +
                        yield lookup_key, *args
         | 
| 58 | 
            +
                      end
         | 
| 59 | 
            +
                    end
         | 
| 60 | 
            +
                end
         | 
| 61 | 
            +
              end
         | 
| 62 | 
            +
            end
         | 
| @@ -0,0 +1,79 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "active_model/type"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            require "active_record/type/internal/timezone"
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            require "active_record/type/date"
         | 
| 8 | 
            +
            require "active_record/type/date_time"
         | 
| 9 | 
            +
            require "active_record/type/decimal_without_scale"
         | 
| 10 | 
            +
            require "active_record/type/json"
         | 
| 11 | 
            +
            require "active_record/type/time"
         | 
| 12 | 
            +
            require "active_record/type/text"
         | 
| 13 | 
            +
            require "active_record/type/unsigned_integer"
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            require "active_record/type/serialized"
         | 
| 16 | 
            +
            require "active_record/type/adapter_specific_registry"
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            require "active_record/type/type_map"
         | 
| 19 | 
            +
            require "active_record/type/hash_lookup_type_map"
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            module ActiveRecord
         | 
| 22 | 
            +
              module Type
         | 
| 23 | 
            +
                @registry = AdapterSpecificRegistry.new
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                class << self
         | 
| 26 | 
            +
                  attr_accessor :registry # :nodoc:
         | 
| 27 | 
            +
                  delegate :add_modifier, to: :registry
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                  # Add a new type to the registry, allowing it to be referenced as a
         | 
| 30 | 
            +
                  # symbol by {ActiveRecord::Base.attribute}[rdoc-ref:Attributes::ClassMethods#attribute].
         | 
| 31 | 
            +
                  # If your type is only meant to be used with a specific database adapter, you can
         | 
| 32 | 
            +
                  # do so by passing <tt>adapter: :postgresql</tt>. If your type has the same
         | 
| 33 | 
            +
                  # name as a native type for the current adapter, an exception will be
         | 
| 34 | 
            +
                  # raised unless you specify an +:override+ option. <tt>override: true</tt> will
         | 
| 35 | 
            +
                  # cause your type to be used instead of the native type. <tt>override:
         | 
| 36 | 
            +
                  # false</tt> will cause the native type to be used over yours if one exists.
         | 
| 37 | 
            +
                  def register(type_name, klass = nil, **options, &block)
         | 
| 38 | 
            +
                    registry.register(type_name, klass, **options, &block)
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                  def lookup(*args, adapter: current_adapter_name, **kwargs) # :nodoc:
         | 
| 42 | 
            +
                    registry.lookup(*args, adapter: adapter, **kwargs)
         | 
| 43 | 
            +
                  end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  def default_value # :nodoc:
         | 
| 46 | 
            +
                    @default_value ||= Value.new
         | 
| 47 | 
            +
                  end
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                  private
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                  def current_adapter_name
         | 
| 52 | 
            +
                    ActiveRecord::Base.connection.adapter_name.downcase.to_sym
         | 
| 53 | 
            +
                  end
         | 
| 54 | 
            +
                end
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                Helpers = ActiveModel::Type::Helpers
         | 
| 57 | 
            +
                BigInteger = ActiveModel::Type::BigInteger
         | 
| 58 | 
            +
                Binary = ActiveModel::Type::Binary
         | 
| 59 | 
            +
                Boolean = ActiveModel::Type::Boolean
         | 
| 60 | 
            +
                Decimal = ActiveModel::Type::Decimal
         | 
| 61 | 
            +
                Float = ActiveModel::Type::Float
         | 
| 62 | 
            +
                Integer = ActiveModel::Type::Integer
         | 
| 63 | 
            +
                String = ActiveModel::Type::String
         | 
| 64 | 
            +
                Value = ActiveModel::Type::Value
         | 
| 65 | 
            +
             | 
| 66 | 
            +
                register(:big_integer, Type::BigInteger, override: false)
         | 
| 67 | 
            +
                register(:binary, Type::Binary, override: false)
         | 
| 68 | 
            +
                register(:boolean, Type::Boolean, override: false)
         | 
| 69 | 
            +
                register(:date, Type::Date, override: false)
         | 
| 70 | 
            +
                register(:datetime, Type::DateTime, override: false)
         | 
| 71 | 
            +
                register(:decimal, Type::Decimal, override: false)
         | 
| 72 | 
            +
                register(:float, Type::Float, override: false)
         | 
| 73 | 
            +
                register(:integer, Type::Integer, override: false)
         | 
| 74 | 
            +
                register(:json, Type::Json, override: false)
         | 
| 75 | 
            +
                register(:string, Type::String, override: false)
         | 
| 76 | 
            +
                register(:text, Type::Text, override: false)
         | 
| 77 | 
            +
                register(:time, Type::Time, override: false)
         | 
| 78 | 
            +
              end
         | 
| 79 | 
            +
            end
         | 
| @@ -0,0 +1,33 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module TypeCaster
         | 
| 5 | 
            +
                class Connection # :nodoc:
         | 
| 6 | 
            +
                  def initialize(klass, table_name)
         | 
| 7 | 
            +
                    @klass = klass
         | 
| 8 | 
            +
                    @table_name = table_name
         | 
| 9 | 
            +
                  end
         | 
| 10 | 
            +
             | 
| 11 | 
            +
                  def type_cast_for_database(attribute_name, value)
         | 
| 12 | 
            +
                    return value if value.is_a?(Arel::Nodes::BindParam)
         | 
| 13 | 
            +
                    column = column_for(attribute_name)
         | 
| 14 | 
            +
                    connection.type_cast_from_column(column, value)
         | 
| 15 | 
            +
                  end
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                  # TODO Change this to private once we've dropped Ruby 2.2 support.
         | 
| 18 | 
            +
                  # Workaround for Ruby 2.2 "private attribute?" warning.
         | 
| 19 | 
            +
                  protected
         | 
| 20 | 
            +
             | 
| 21 | 
            +
                    attr_reader :table_name
         | 
| 22 | 
            +
                    delegate :connection, to: :@klass
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                  private
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                    def column_for(attribute_name)
         | 
| 27 | 
            +
                      if connection.schema_cache.data_source_exists?(table_name)
         | 
| 28 | 
            +
                        connection.schema_cache.columns_hash(table_name)[attribute_name.to_s]
         | 
| 29 | 
            +
                      end
         | 
| 30 | 
            +
                    end
         | 
| 31 | 
            +
                end
         | 
| 32 | 
            +
              end
         | 
| 33 | 
            +
            end
         | 
| @@ -0,0 +1,23 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module TypeCaster
         | 
| 5 | 
            +
                class Map # :nodoc:
         | 
| 6 | 
            +
                  def initialize(types)
         | 
| 7 | 
            +
                    @types = types
         | 
| 8 | 
            +
                  end
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                  def type_cast_for_database(attr_name, value)
         | 
| 11 | 
            +
                    return value if value.is_a?(Arel::Nodes::BindParam)
         | 
| 12 | 
            +
                    type = types.type_for_attribute(attr_name)
         | 
| 13 | 
            +
                    type.serialize(value)
         | 
| 14 | 
            +
                  end
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                  # TODO Change this to private once we've dropped Ruby 2.2 support.
         | 
| 17 | 
            +
                  # Workaround for Ruby 2.2 "private attribute?" warning.
         | 
| 18 | 
            +
                  protected
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                    attr_reader :types
         | 
| 21 | 
            +
                end
         | 
| 22 | 
            +
              end
         | 
| 23 | 
            +
            end
         | 
| @@ -0,0 +1,25 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Validations
         | 
| 5 | 
            +
                class AbsenceValidator < ActiveModel::Validations::AbsenceValidator # :nodoc:
         | 
| 6 | 
            +
                  def validate_each(record, attribute, association_or_value)
         | 
| 7 | 
            +
                    if record.class._reflect_on_association(attribute)
         | 
| 8 | 
            +
                      association_or_value = Array.wrap(association_or_value).reject(&:marked_for_destruction?)
         | 
| 9 | 
            +
                    end
         | 
| 10 | 
            +
                    super
         | 
| 11 | 
            +
                  end
         | 
| 12 | 
            +
                end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                module ClassMethods
         | 
| 15 | 
            +
                  # Validates that the specified attributes are not present (as defined by
         | 
| 16 | 
            +
                  # Object#present?). If the attribute is an association, the associated object
         | 
| 17 | 
            +
                  # is considered absent if it was marked for destruction.
         | 
| 18 | 
            +
                  #
         | 
| 19 | 
            +
                  # See ActiveModel::Validations::HelperMethods.validates_absence_of for more information.
         | 
| 20 | 
            +
                  def validates_absence_of(*attr_names)
         | 
| 21 | 
            +
                    validates_with AbsenceValidator, _merge_attributes(attr_names)
         | 
| 22 | 
            +
                  end
         | 
| 23 | 
            +
                end
         | 
| 24 | 
            +
              end
         | 
| 25 | 
            +
            end
         | 
| @@ -1,15 +1,24 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 1 3 | 
             
            module ActiveRecord
         | 
| 2 4 | 
             
              module Validations
         | 
| 3 | 
            -
                class AssociatedValidator < ActiveModel::EachValidator
         | 
| 5 | 
            +
                class AssociatedValidator < ActiveModel::EachValidator #:nodoc:
         | 
| 4 6 | 
             
                  def validate_each(record, attribute, value)
         | 
| 5 | 
            -
                    if Array | 
| 6 | 
            -
                      record.errors.add(attribute, :invalid, options.merge(: | 
| 7 | 
            +
                    if Array(value).reject { |r| valid_object?(r) }.any?
         | 
| 8 | 
            +
                      record.errors.add(attribute, :invalid, options.merge(value: value))
         | 
| 7 9 | 
             
                    end
         | 
| 8 10 | 
             
                  end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  private
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                    def valid_object?(record)
         | 
| 15 | 
            +
                      (record.respond_to?(:marked_for_destruction?) && record.marked_for_destruction?) || record.valid?
         | 
| 16 | 
            +
                    end
         | 
| 9 17 | 
             
                end
         | 
| 10 18 |  | 
| 11 19 | 
             
                module ClassMethods
         | 
| 12 | 
            -
                  # Validates whether the associated object or objects are all valid | 
| 20 | 
            +
                  # Validates whether the associated object or objects are all valid.
         | 
| 21 | 
            +
                  # Works with any kind of association.
         | 
| 13 22 | 
             
                  #
         | 
| 14 23 | 
             
                  #   class Book < ActiveRecord::Base
         | 
| 15 24 | 
             
                  #     has_many :pages
         | 
| @@ -18,23 +27,31 @@ module ActiveRecord | |
| 18 27 | 
             
                  #     validates_associated :pages, :library
         | 
| 19 28 | 
             
                  #   end
         | 
| 20 29 | 
             
                  #
         | 
| 21 | 
            -
                  # WARNING: This validation must not be used on both ends of an association. | 
| 30 | 
            +
                  # WARNING: This validation must not be used on both ends of an association.
         | 
| 31 | 
            +
                  # Doing so will lead to a circular dependency and cause infinite recursion.
         | 
| 22 32 | 
             
                  #
         | 
| 23 | 
            -
                  # NOTE: This validation will not fail if the association hasn't been | 
| 24 | 
            -
                  # ensure that the association is both present and | 
| 25 | 
            -
                  # use | 
| 33 | 
            +
                  # NOTE: This validation will not fail if the association hasn't been
         | 
| 34 | 
            +
                  # assigned. If you want to ensure that the association is both present and
         | 
| 35 | 
            +
                  # guaranteed to be valid, you also need to use
         | 
| 36 | 
            +
                  # {validates_presence_of}[rdoc-ref:Validations::ClassMethods#validates_presence_of].
         | 
| 26 37 | 
             
                  #
         | 
| 27 38 | 
             
                  # Configuration options:
         | 
| 28 | 
            -
                  # | 
| 29 | 
            -
                  # * <tt>: | 
| 30 | 
            -
                  # | 
| 31 | 
            -
                  #    | 
| 32 | 
            -
                  #  | 
| 33 | 
            -
                  #    | 
| 34 | 
            -
                  #    | 
| 35 | 
            -
                  # * <tt>: | 
| 36 | 
            -
                  #    | 
| 37 | 
            -
                  #    | 
| 39 | 
            +
                  #
         | 
| 40 | 
            +
                  # * <tt>:message</tt> - A custom error message (default is: "is invalid").
         | 
| 41 | 
            +
                  # * <tt>:on</tt> - Specifies the contexts where this validation is active.
         | 
| 42 | 
            +
                  #   Runs in all validation contexts by default +nil+. You can pass a symbol
         | 
| 43 | 
            +
                  #   or an array of symbols. (e.g. <tt>on: :create</tt> or
         | 
| 44 | 
            +
                  #   <tt>on: :custom_validation_context</tt> or
         | 
| 45 | 
            +
                  #   <tt>on: [:create, :custom_validation_context]</tt>)
         | 
| 46 | 
            +
                  # * <tt>:if</tt> - Specifies a method, proc or string to call to determine
         | 
| 47 | 
            +
                  #   if the validation should occur (e.g. <tt>if: :allow_validation</tt>,
         | 
| 48 | 
            +
                  #   or <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method,
         | 
| 49 | 
            +
                  #   proc or string should return or evaluate to a +true+ or +false+ value.
         | 
| 50 | 
            +
                  # * <tt>:unless</tt> - Specifies a method, proc or string to call to
         | 
| 51 | 
            +
                  #   determine if the validation should not occur (e.g. <tt>unless: :skip_validation</tt>,
         | 
| 52 | 
            +
                  #   or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The
         | 
| 53 | 
            +
                  #   method, proc or string should return or evaluate to a +true+ or +false+
         | 
| 54 | 
            +
                  #   value.
         | 
| 38 55 | 
             
                  def validates_associated(*attr_names)
         | 
| 39 56 | 
             
                    validates_with AssociatedValidator, _merge_attributes(attr_names)
         | 
| 40 57 | 
             
                  end
         |