activerecord 4.1.16 → 4.2.11.3
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 +1162 -1801
- data/README.rdoc +15 -10
- data/lib/active_record/aggregations.rb +15 -8
- data/lib/active_record/association_relation.rb +13 -0
- data/lib/active_record/associations/alias_tracker.rb +3 -12
- data/lib/active_record/associations/association.rb +16 -4
- data/lib/active_record/associations/association_scope.rb +83 -38
- data/lib/active_record/associations/belongs_to_association.rb +28 -10
- data/lib/active_record/associations/builder/association.rb +15 -4
- data/lib/active_record/associations/builder/belongs_to.rb +7 -29
- data/lib/active_record/associations/builder/collection_association.rb +5 -1
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +8 -13
- data/lib/active_record/associations/builder/has_many.rb +1 -1
- data/lib/active_record/associations/builder/has_one.rb +2 -2
- data/lib/active_record/associations/builder/singular_association.rb +8 -1
- data/lib/active_record/associations/collection_association.rb +63 -27
- data/lib/active_record/associations/collection_proxy.rb +29 -35
- data/lib/active_record/associations/foreign_association.rb +11 -0
- data/lib/active_record/associations/has_many_association.rb +83 -22
- data/lib/active_record/associations/has_many_through_association.rb +49 -26
- data/lib/active_record/associations/has_one_association.rb +1 -1
- data/lib/active_record/associations/join_dependency/join_association.rb +25 -15
- data/lib/active_record/associations/join_dependency/join_part.rb +0 -1
- data/lib/active_record/associations/join_dependency.rb +26 -13
- data/lib/active_record/associations/preloader/association.rb +14 -11
- data/lib/active_record/associations/preloader/through_association.rb +4 -3
- data/lib/active_record/associations/preloader.rb +36 -26
- data/lib/active_record/associations/singular_association.rb +17 -2
- data/lib/active_record/associations/through_association.rb +5 -12
- data/lib/active_record/associations.rb +158 -49
- data/lib/active_record/attribute.rb +163 -0
- data/lib/active_record/attribute_assignment.rb +19 -11
- data/lib/active_record/attribute_decorators.rb +66 -0
- data/lib/active_record/attribute_methods/before_type_cast.rb +7 -2
- data/lib/active_record/attribute_methods/dirty.rb +107 -43
- data/lib/active_record/attribute_methods/primary_key.rb +7 -8
- data/lib/active_record/attribute_methods/query.rb +1 -1
- data/lib/active_record/attribute_methods/read.rb +22 -59
- data/lib/active_record/attribute_methods/serialization.rb +16 -150
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +38 -40
- data/lib/active_record/attribute_methods/write.rb +9 -24
- data/lib/active_record/attribute_methods.rb +56 -94
- data/lib/active_record/attribute_set/builder.rb +106 -0
- data/lib/active_record/attribute_set.rb +81 -0
- data/lib/active_record/attributes.rb +147 -0
- data/lib/active_record/autosave_association.rb +19 -12
- data/lib/active_record/base.rb +13 -24
- data/lib/active_record/callbacks.rb +6 -6
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +84 -52
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +52 -50
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +1 -1
- data/lib/active_record/connection_adapters/abstract/quoting.rb +60 -60
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +1 -1
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +39 -4
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +138 -56
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +14 -34
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +268 -71
- data/lib/active_record/connection_adapters/abstract/transaction.rb +125 -118
- data/lib/active_record/connection_adapters/abstract_adapter.rb +171 -59
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +293 -139
- data/lib/active_record/connection_adapters/column.rb +29 -240
- data/lib/active_record/connection_adapters/connection_specification.rb +15 -24
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +16 -32
- data/lib/active_record/connection_adapters/mysql_adapter.rb +67 -144
- data/lib/active_record/connection_adapters/postgresql/array_parser.rb +15 -27
- data/lib/active_record/connection_adapters/postgresql/column.rb +20 -0
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +40 -25
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +100 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +52 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +13 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +46 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date.rb +11 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +36 -0
- data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +13 -0
- data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +19 -0
- data/lib/active_record/connection_adapters/postgresql/oid/float.rb +21 -0
- data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +59 -0
- data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +13 -0
- data/lib/active_record/connection_adapters/postgresql/oid/infinity.rb +13 -0
- data/lib/active_record/connection_adapters/postgresql/oid/integer.rb +11 -0
- data/lib/active_record/connection_adapters/postgresql/oid/json.rb +35 -0
- data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/money.rb +43 -0
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +43 -0
- data/lib/active_record/connection_adapters/postgresql/oid/range.rb +79 -0
- data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +19 -0
- data/lib/active_record/connection_adapters/postgresql/oid/time.rb +11 -0
- data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +109 -0
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +21 -0
- data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +26 -0
- data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +28 -0
- data/lib/active_record/connection_adapters/postgresql/oid.rb +29 -388
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +46 -136
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +4 -4
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +152 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +131 -43
- data/lib/active_record/connection_adapters/postgresql/utils.rb +77 -0
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +224 -477
- data/lib/active_record/connection_adapters/schema_cache.rb +14 -28
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +61 -75
- data/lib/active_record/connection_handling.rb +1 -1
- data/lib/active_record/core.rb +163 -39
- data/lib/active_record/counter_cache.rb +60 -6
- data/lib/active_record/enum.rb +9 -11
- data/lib/active_record/errors.rb +53 -30
- data/lib/active_record/explain.rb +1 -1
- data/lib/active_record/explain_subscriber.rb +1 -1
- data/lib/active_record/fixtures.rb +55 -69
- data/lib/active_record/gem_version.rb +4 -4
- data/lib/active_record/inheritance.rb +35 -10
- data/lib/active_record/integration.rb +4 -4
- data/lib/active_record/legacy_yaml_adapter.rb +30 -0
- data/lib/active_record/locking/optimistic.rb +46 -26
- data/lib/active_record/migration/command_recorder.rb +19 -2
- data/lib/active_record/migration/join_table.rb +1 -1
- data/lib/active_record/migration.rb +71 -46
- data/lib/active_record/model_schema.rb +52 -58
- data/lib/active_record/nested_attributes.rb +5 -5
- data/lib/active_record/no_touching.rb +1 -1
- data/lib/active_record/persistence.rb +46 -26
- data/lib/active_record/query_cache.rb +3 -3
- data/lib/active_record/querying.rb +10 -7
- data/lib/active_record/railtie.rb +18 -11
- data/lib/active_record/railties/databases.rake +50 -51
- data/lib/active_record/readonly_attributes.rb +0 -1
- data/lib/active_record/reflection.rb +273 -114
- data/lib/active_record/relation/batches.rb +0 -2
- data/lib/active_record/relation/calculations.rb +41 -37
- data/lib/active_record/relation/finder_methods.rb +70 -47
- data/lib/active_record/relation/merger.rb +39 -29
- data/lib/active_record/relation/predicate_builder/array_handler.rb +32 -13
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +1 -5
- data/lib/active_record/relation/predicate_builder.rb +16 -8
- data/lib/active_record/relation/query_methods.rb +114 -65
- data/lib/active_record/relation/spawn_methods.rb +3 -0
- data/lib/active_record/relation.rb +57 -25
- data/lib/active_record/result.rb +18 -7
- data/lib/active_record/sanitization.rb +12 -2
- data/lib/active_record/schema.rb +0 -1
- data/lib/active_record/schema_dumper.rb +59 -28
- data/lib/active_record/schema_migration.rb +5 -4
- data/lib/active_record/scoping/default.rb +6 -4
- data/lib/active_record/scoping/named.rb +4 -0
- data/lib/active_record/serializers/xml_serializer.rb +3 -7
- data/lib/active_record/statement_cache.rb +95 -10
- data/lib/active_record/store.rb +5 -5
- data/lib/active_record/tasks/database_tasks.rb +61 -6
- data/lib/active_record/tasks/mysql_database_tasks.rb +20 -11
- data/lib/active_record/tasks/postgresql_database_tasks.rb +20 -9
- data/lib/active_record/timestamp.rb +9 -7
- data/lib/active_record/transactions.rb +53 -27
- data/lib/active_record/type/big_integer.rb +13 -0
- data/lib/active_record/type/binary.rb +50 -0
- data/lib/active_record/type/boolean.rb +31 -0
- data/lib/active_record/type/date.rb +50 -0
- data/lib/active_record/type/date_time.rb +54 -0
- data/lib/active_record/type/decimal.rb +64 -0
- data/lib/active_record/type/decimal_without_scale.rb +11 -0
- data/lib/active_record/type/decorator.rb +14 -0
- data/lib/active_record/type/float.rb +19 -0
- data/lib/active_record/type/hash_lookup_type_map.rb +23 -0
- data/lib/active_record/type/integer.rb +59 -0
- data/lib/active_record/type/mutable.rb +16 -0
- data/lib/active_record/type/numeric.rb +36 -0
- data/lib/active_record/type/serialized.rb +62 -0
- data/lib/active_record/type/string.rb +40 -0
- data/lib/active_record/type/text.rb +11 -0
- data/lib/active_record/type/time.rb +26 -0
- data/lib/active_record/type/time_value.rb +38 -0
- data/lib/active_record/type/type_map.rb +64 -0
- data/lib/active_record/type/unsigned_integer.rb +15 -0
- data/lib/active_record/type/value.rb +110 -0
- data/lib/active_record/type.rb +23 -0
- data/lib/active_record/validations/associated.rb +5 -3
- data/lib/active_record/validations/presence.rb +5 -3
- data/lib/active_record/validations/uniqueness.rb +25 -29
- data/lib/active_record/validations.rb +25 -19
- data/lib/active_record.rb +4 -0
- data/lib/rails/generators/active_record/migration/migration_generator.rb +8 -4
- data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb +1 -1
- data/lib/rails/generators/active_record/model/templates/model.rb +1 -1
- metadata +66 -11
- data/lib/active_record/connection_adapters/postgresql/cast.rb +0 -168
| @@ -0,0 +1,59 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                class Integer < Value # :nodoc:
         | 
| 4 | 
            +
                  include Numeric
         | 
| 5 | 
            +
             | 
| 6 | 
            +
                  def initialize(*)
         | 
| 7 | 
            +
                    super
         | 
| 8 | 
            +
                    @range = min_value...max_value
         | 
| 9 | 
            +
                  end
         | 
| 10 | 
            +
             | 
| 11 | 
            +
                  def type
         | 
| 12 | 
            +
                    :integer
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                  def type_cast_from_database(value)
         | 
| 16 | 
            +
                    return if value.nil?
         | 
| 17 | 
            +
                    value.to_i
         | 
| 18 | 
            +
                  end
         | 
| 19 | 
            +
             | 
| 20 | 
            +
                  def type_cast_for_database(value)
         | 
| 21 | 
            +
                    result = type_cast(value)
         | 
| 22 | 
            +
                    if result
         | 
| 23 | 
            +
                      ensure_in_range(result)
         | 
| 24 | 
            +
                    end
         | 
| 25 | 
            +
                    result
         | 
| 26 | 
            +
                  end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                  protected
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                  attr_reader :range
         | 
| 31 | 
            +
             | 
| 32 | 
            +
                  private
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                  def cast_value(value)
         | 
| 35 | 
            +
                    case value
         | 
| 36 | 
            +
                    when true then 1
         | 
| 37 | 
            +
                    when false then 0
         | 
| 38 | 
            +
                    else
         | 
| 39 | 
            +
                      value.to_i rescue nil
         | 
| 40 | 
            +
                    end
         | 
| 41 | 
            +
                  end
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                  def ensure_in_range(value)
         | 
| 44 | 
            +
                    unless range.cover?(value)
         | 
| 45 | 
            +
                      raise RangeError, "#{value} is out of range for #{self.class} with limit #{limit || 4}"
         | 
| 46 | 
            +
                    end
         | 
| 47 | 
            +
                  end
         | 
| 48 | 
            +
             | 
| 49 | 
            +
                  def max_value
         | 
| 50 | 
            +
                    limit = self.limit || 4
         | 
| 51 | 
            +
                    1 << (limit * 8 - 1) # 8 bits per byte with one bit for sign
         | 
| 52 | 
            +
                  end
         | 
| 53 | 
            +
             | 
| 54 | 
            +
                  def min_value
         | 
| 55 | 
            +
                    -max_value
         | 
| 56 | 
            +
                  end
         | 
| 57 | 
            +
                end
         | 
| 58 | 
            +
              end
         | 
| 59 | 
            +
            end
         | 
| @@ -0,0 +1,16 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                module Mutable # :nodoc:
         | 
| 4 | 
            +
                  def type_cast_from_user(value)
         | 
| 5 | 
            +
                    type_cast_from_database(type_cast_for_database(value))
         | 
| 6 | 
            +
                  end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  # +raw_old_value+ will be the `_before_type_cast` version of the
         | 
| 9 | 
            +
                  # value (likely a string). +new_value+ will be the current, type
         | 
| 10 | 
            +
                  # cast value.
         | 
| 11 | 
            +
                  def changed_in_place?(raw_old_value, new_value)
         | 
| 12 | 
            +
                    raw_old_value != type_cast_for_database(new_value)
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
                end
         | 
| 15 | 
            +
              end
         | 
| 16 | 
            +
            end
         | 
| @@ -0,0 +1,36 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                module Numeric # :nodoc:
         | 
| 4 | 
            +
                  def number?
         | 
| 5 | 
            +
                    true
         | 
| 6 | 
            +
                  end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  def type_cast(value)
         | 
| 9 | 
            +
                    value = case value
         | 
| 10 | 
            +
                            when true then 1
         | 
| 11 | 
            +
                            when false then 0
         | 
| 12 | 
            +
                            when ::String then value.presence
         | 
| 13 | 
            +
                            else value
         | 
| 14 | 
            +
                            end
         | 
| 15 | 
            +
                    super(value)
         | 
| 16 | 
            +
                  end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  def changed?(old_value, _new_value, new_value_before_type_cast) # :nodoc:
         | 
| 19 | 
            +
                    super || number_to_non_number?(old_value, new_value_before_type_cast)
         | 
| 20 | 
            +
                  end
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                  private
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                  def number_to_non_number?(old_value, new_value_before_type_cast)
         | 
| 25 | 
            +
                    old_value != nil && non_numeric_string?(new_value_before_type_cast)
         | 
| 26 | 
            +
                  end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                  def non_numeric_string?(value)
         | 
| 29 | 
            +
                    # 'wibble'.to_i will give zero, we want to make sure
         | 
| 30 | 
            +
                    # that we aren't marking int zero to string zero as
         | 
| 31 | 
            +
                    # changed.
         | 
| 32 | 
            +
                    value.to_s !~ /\A-?\d+\.?\d*\z/
         | 
| 33 | 
            +
                  end
         | 
| 34 | 
            +
                end
         | 
| 35 | 
            +
              end
         | 
| 36 | 
            +
            end
         | 
| @@ -0,0 +1,62 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                class Serialized < DelegateClass(Type::Value) # :nodoc:
         | 
| 4 | 
            +
                  include Mutable
         | 
| 5 | 
            +
                  include Decorator
         | 
| 6 | 
            +
             | 
| 7 | 
            +
                  attr_reader :subtype, :coder
         | 
| 8 | 
            +
             | 
| 9 | 
            +
                  def initialize(subtype, coder)
         | 
| 10 | 
            +
                    @subtype = subtype
         | 
| 11 | 
            +
                    @coder = coder
         | 
| 12 | 
            +
                    super(subtype)
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                  def type_cast_from_database(value)
         | 
| 16 | 
            +
                    if default_value?(value)
         | 
| 17 | 
            +
                      value
         | 
| 18 | 
            +
                    else
         | 
| 19 | 
            +
                      coder.load(super)
         | 
| 20 | 
            +
                    end
         | 
| 21 | 
            +
                  end
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                  def type_cast_for_database(value)
         | 
| 24 | 
            +
                    return if value.nil?
         | 
| 25 | 
            +
                    unless default_value?(value)
         | 
| 26 | 
            +
                      super coder.dump(value)
         | 
| 27 | 
            +
                    end
         | 
| 28 | 
            +
                  end
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                  def inspect
         | 
| 31 | 
            +
                    Kernel.instance_method(:inspect).bind(self).call
         | 
| 32 | 
            +
                  end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                  def changed_in_place?(raw_old_value, value)
         | 
| 35 | 
            +
                    return false if value.nil?
         | 
| 36 | 
            +
                    raw_new_value = type_cast_for_database(value)
         | 
| 37 | 
            +
                    raw_old_value.nil? != raw_new_value.nil? ||
         | 
| 38 | 
            +
                      subtype.changed_in_place?(raw_old_value, raw_new_value)
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                  def accessor
         | 
| 42 | 
            +
                    ActiveRecord::Store::IndifferentHashAccessor
         | 
| 43 | 
            +
                  end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  def init_with(coder)
         | 
| 46 | 
            +
                    @coder = coder['coder']
         | 
| 47 | 
            +
                    super
         | 
| 48 | 
            +
                  end
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                  def encode_with(coder)
         | 
| 51 | 
            +
                    coder['coder'] = @coder
         | 
| 52 | 
            +
                    super
         | 
| 53 | 
            +
                  end
         | 
| 54 | 
            +
             | 
| 55 | 
            +
                  private
         | 
| 56 | 
            +
             | 
| 57 | 
            +
                  def default_value?(value)
         | 
| 58 | 
            +
                    value == coder.load(nil)
         | 
| 59 | 
            +
                  end
         | 
| 60 | 
            +
                end
         | 
| 61 | 
            +
              end
         | 
| 62 | 
            +
            end
         | 
| @@ -0,0 +1,40 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                class String < Value # :nodoc:
         | 
| 4 | 
            +
                  def type
         | 
| 5 | 
            +
                    :string
         | 
| 6 | 
            +
                  end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  def changed_in_place?(raw_old_value, new_value)
         | 
| 9 | 
            +
                    if new_value.is_a?(::String)
         | 
| 10 | 
            +
                      raw_old_value != new_value
         | 
| 11 | 
            +
                    end
         | 
| 12 | 
            +
                  end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                  def type_cast_for_database(value)
         | 
| 15 | 
            +
                    case value
         | 
| 16 | 
            +
                    when ::Numeric, ActiveSupport::Duration then value.to_s
         | 
| 17 | 
            +
                    when ::String then ::String.new(value)
         | 
| 18 | 
            +
                    when true then "t"
         | 
| 19 | 
            +
                    when false then "f"
         | 
| 20 | 
            +
                    else super
         | 
| 21 | 
            +
                    end
         | 
| 22 | 
            +
                  end
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                  def text?
         | 
| 25 | 
            +
                    true
         | 
| 26 | 
            +
                  end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                  private
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                  def cast_value(value)
         | 
| 31 | 
            +
                    case value
         | 
| 32 | 
            +
                    when true then "t"
         | 
| 33 | 
            +
                    when false then "f"
         | 
| 34 | 
            +
                    # String.new is slightly faster than dup
         | 
| 35 | 
            +
                    else ::String.new(value.to_s)
         | 
| 36 | 
            +
                    end
         | 
| 37 | 
            +
                  end
         | 
| 38 | 
            +
                end
         | 
| 39 | 
            +
              end
         | 
| 40 | 
            +
            end
         | 
| @@ -0,0 +1,26 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                class Time < Value # :nodoc:
         | 
| 4 | 
            +
                  include TimeValue
         | 
| 5 | 
            +
             | 
| 6 | 
            +
                  def type
         | 
| 7 | 
            +
                    :time
         | 
| 8 | 
            +
                  end
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                  private
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  def cast_value(value)
         | 
| 13 | 
            +
                    return value unless value.is_a?(::String)
         | 
| 14 | 
            +
                    return if value.empty?
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                    dummy_time_value = "2000-01-01 #{value}"
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                    fast_string_to_time(dummy_time_value) || begin
         | 
| 19 | 
            +
                      time_hash = ::Date._parse(dummy_time_value)
         | 
| 20 | 
            +
                      return if time_hash[:hour].nil?
         | 
| 21 | 
            +
                      new_time(*time_hash.values_at(:year, :mon, :mday, :hour, :min, :sec, :sec_fraction))
         | 
| 22 | 
            +
                    end
         | 
| 23 | 
            +
                  end
         | 
| 24 | 
            +
                end
         | 
| 25 | 
            +
              end
         | 
| 26 | 
            +
            end
         | 
| @@ -0,0 +1,38 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                module TimeValue # :nodoc:
         | 
| 4 | 
            +
                  def klass
         | 
| 5 | 
            +
                    ::Time
         | 
| 6 | 
            +
                  end
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                  def type_cast_for_schema(value)
         | 
| 9 | 
            +
                    "'#{value.to_s(:db)}'"
         | 
| 10 | 
            +
                  end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                  private
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                  def new_time(year, mon, mday, hour, min, sec, microsec, offset = nil)
         | 
| 15 | 
            +
                    # Treat 0000-00-00 00:00:00 as nil.
         | 
| 16 | 
            +
                    return if year.nil? || (year == 0 && mon == 0 && mday == 0)
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                    if offset
         | 
| 19 | 
            +
                      time = ::Time.utc(year, mon, mday, hour, min, sec, microsec) rescue nil
         | 
| 20 | 
            +
                      return unless time
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                      time -= offset
         | 
| 23 | 
            +
                      Base.default_timezone == :utc ? time : time.getlocal
         | 
| 24 | 
            +
                    else
         | 
| 25 | 
            +
                      ::Time.public_send(Base.default_timezone, year, mon, mday, hour, min, sec, microsec) rescue nil
         | 
| 26 | 
            +
                    end
         | 
| 27 | 
            +
                  end
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                  # Doesn't handle time zones.
         | 
| 30 | 
            +
                  def fast_string_to_time(string)
         | 
| 31 | 
            +
                    if string =~ ConnectionAdapters::Column::Format::ISO_DATETIME
         | 
| 32 | 
            +
                      microsec = ($7.to_r * 1_000_000).to_i
         | 
| 33 | 
            +
                      new_time $1.to_i, $2.to_i, $3.to_i, $4.to_i, $5.to_i, $6.to_i, microsec
         | 
| 34 | 
            +
                    end
         | 
| 35 | 
            +
                  end
         | 
| 36 | 
            +
                end
         | 
| 37 | 
            +
              end
         | 
| 38 | 
            +
            end
         | 
| @@ -0,0 +1,64 @@ | |
| 1 | 
            +
            require 'thread_safe'
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module Type
         | 
| 5 | 
            +
                class TypeMap # :nodoc:
         | 
| 6 | 
            +
                  def initialize
         | 
| 7 | 
            +
                    @mapping = {}
         | 
| 8 | 
            +
                    @cache = ThreadSafe::Cache.new do |h, key|
         | 
| 9 | 
            +
                      h.fetch_or_store(key, ThreadSafe::Cache.new)
         | 
| 10 | 
            +
                    end
         | 
| 11 | 
            +
                  end
         | 
| 12 | 
            +
             | 
| 13 | 
            +
                  def lookup(lookup_key, *args)
         | 
| 14 | 
            +
                    fetch(lookup_key, *args) { default_value }
         | 
| 15 | 
            +
                  end
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                  def fetch(lookup_key, *args, &block)
         | 
| 18 | 
            +
                    @cache[lookup_key].fetch_or_store(args) do
         | 
| 19 | 
            +
                      perform_fetch(lookup_key, *args, &block)
         | 
| 20 | 
            +
                    end
         | 
| 21 | 
            +
                  end
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                  def register_type(key, value = nil, &block)
         | 
| 24 | 
            +
                    raise ::ArgumentError unless value || block
         | 
| 25 | 
            +
                    @cache.clear
         | 
| 26 | 
            +
             | 
| 27 | 
            +
                    if block
         | 
| 28 | 
            +
                      @mapping[key] = block
         | 
| 29 | 
            +
                    else
         | 
| 30 | 
            +
                      @mapping[key] = proc { value }
         | 
| 31 | 
            +
                    end
         | 
| 32 | 
            +
                  end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                  def alias_type(key, target_key)
         | 
| 35 | 
            +
                    register_type(key) do |sql_type, *args|
         | 
| 36 | 
            +
                      metadata = sql_type[/\(.*\)/, 0]
         | 
| 37 | 
            +
                      lookup("#{target_key}#{metadata}", *args)
         | 
| 38 | 
            +
                    end
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                  def clear
         | 
| 42 | 
            +
                    @mapping.clear
         | 
| 43 | 
            +
                  end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  private
         | 
| 46 | 
            +
             | 
| 47 | 
            +
                  def perform_fetch(lookup_key, *args)
         | 
| 48 | 
            +
                    matching_pair = @mapping.reverse_each.detect do |key, _|
         | 
| 49 | 
            +
                      key === lookup_key
         | 
| 50 | 
            +
                    end
         | 
| 51 | 
            +
             | 
| 52 | 
            +
                    if matching_pair
         | 
| 53 | 
            +
                      matching_pair.last.call(lookup_key, *args)
         | 
| 54 | 
            +
                    else
         | 
| 55 | 
            +
                      yield lookup_key, *args
         | 
| 56 | 
            +
                    end
         | 
| 57 | 
            +
                  end
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                  def default_value
         | 
| 60 | 
            +
                    @default_value ||= Value.new
         | 
| 61 | 
            +
                  end
         | 
| 62 | 
            +
                end
         | 
| 63 | 
            +
              end
         | 
| 64 | 
            +
            end
         | 
| @@ -0,0 +1,110 @@ | |
| 1 | 
            +
            module ActiveRecord
         | 
| 2 | 
            +
              module Type
         | 
| 3 | 
            +
                class Value # :nodoc:
         | 
| 4 | 
            +
                  attr_reader :precision, :scale, :limit
         | 
| 5 | 
            +
             | 
| 6 | 
            +
                  # Valid options are +precision+, +scale+, and +limit+. They are only
         | 
| 7 | 
            +
                  # used when dumping schema.
         | 
| 8 | 
            +
                  def initialize(options = {})
         | 
| 9 | 
            +
                    options.assert_valid_keys(:precision, :scale, :limit)
         | 
| 10 | 
            +
                    @precision = options[:precision]
         | 
| 11 | 
            +
                    @scale = options[:scale]
         | 
| 12 | 
            +
                    @limit = options[:limit]
         | 
| 13 | 
            +
                  end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                  # The simplified type that this object represents. Returns a symbol such
         | 
| 16 | 
            +
                  # as +:string+ or +:integer+
         | 
| 17 | 
            +
                  def type; end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                  # Type casts a string from the database into the appropriate ruby type.
         | 
| 20 | 
            +
                  # Classes which do not need separate type casting behavior for database
         | 
| 21 | 
            +
                  # and user provided values should override +cast_value+ instead.
         | 
| 22 | 
            +
                  def type_cast_from_database(value)
         | 
| 23 | 
            +
                    type_cast(value)
         | 
| 24 | 
            +
                  end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                  # Type casts a value from user input (e.g. from a setter). This value may
         | 
| 27 | 
            +
                  # be a string from the form builder, or an already type cast value
         | 
| 28 | 
            +
                  # provided manually to a setter.
         | 
| 29 | 
            +
                  #
         | 
| 30 | 
            +
                  # Classes which do not need separate type casting behavior for database
         | 
| 31 | 
            +
                  # and user provided values should override +type_cast+ or +cast_value+
         | 
| 32 | 
            +
                  # instead.
         | 
| 33 | 
            +
                  def type_cast_from_user(value)
         | 
| 34 | 
            +
                    type_cast(value)
         | 
| 35 | 
            +
                  end
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                  # Cast a value from the ruby type to a type that the database knows how
         | 
| 38 | 
            +
                  # to understand. The returned value from this method should be a
         | 
| 39 | 
            +
                  # +String+, +Numeric+, +Date+, +Time+, +Symbol+, +true+, +false+, or
         | 
| 40 | 
            +
                  # +nil+
         | 
| 41 | 
            +
                  def type_cast_for_database(value)
         | 
| 42 | 
            +
                    value
         | 
| 43 | 
            +
                  end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  # Type cast a value for schema dumping. This method is private, as we are
         | 
| 46 | 
            +
                  # hoping to remove it entirely.
         | 
| 47 | 
            +
                  def type_cast_for_schema(value) # :nodoc:
         | 
| 48 | 
            +
                    value.inspect
         | 
| 49 | 
            +
                  end
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                  # These predicates are not documented, as I need to look further into
         | 
| 52 | 
            +
                  # their use, and see if they can be removed entirely.
         | 
| 53 | 
            +
                  def text? # :nodoc:
         | 
| 54 | 
            +
                    false
         | 
| 55 | 
            +
                  end
         | 
| 56 | 
            +
             | 
| 57 | 
            +
                  def number? # :nodoc:
         | 
| 58 | 
            +
                    false
         | 
| 59 | 
            +
                  end
         | 
| 60 | 
            +
             | 
| 61 | 
            +
                  def binary? # :nodoc:
         | 
| 62 | 
            +
                    false
         | 
| 63 | 
            +
                  end
         | 
| 64 | 
            +
             | 
| 65 | 
            +
                  def klass # :nodoc:
         | 
| 66 | 
            +
                  end
         | 
| 67 | 
            +
             | 
| 68 | 
            +
                  # Determines whether a value has changed for dirty checking. +old_value+
         | 
| 69 | 
            +
                  # and +new_value+ will always be type-cast. Types should not need to
         | 
| 70 | 
            +
                  # override this method.
         | 
| 71 | 
            +
                  def changed?(old_value, new_value, _new_value_before_type_cast)
         | 
| 72 | 
            +
                    old_value != new_value
         | 
| 73 | 
            +
                  end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                  # Determines whether the mutable value has been modified since it was
         | 
| 76 | 
            +
                  # read. Returns +false+ by default. This method should not be overridden
         | 
| 77 | 
            +
                  # directly. Types which return a mutable value should include
         | 
| 78 | 
            +
                  # +Type::Mutable+, which will define this method.
         | 
| 79 | 
            +
                  def changed_in_place?(*)
         | 
| 80 | 
            +
                    false
         | 
| 81 | 
            +
                  end
         | 
| 82 | 
            +
             | 
| 83 | 
            +
                  def ==(other)
         | 
| 84 | 
            +
                    self.class == other.class &&
         | 
| 85 | 
            +
                      precision == other.precision &&
         | 
| 86 | 
            +
                      scale == other.scale &&
         | 
| 87 | 
            +
                      limit == other.limit
         | 
| 88 | 
            +
                  end
         | 
| 89 | 
            +
                  alias eql? ==
         | 
| 90 | 
            +
             | 
| 91 | 
            +
                  def hash
         | 
| 92 | 
            +
                    [self.class, precision, scale, limit].hash
         | 
| 93 | 
            +
                  end
         | 
| 94 | 
            +
             | 
| 95 | 
            +
                  private
         | 
| 96 | 
            +
             | 
| 97 | 
            +
                  def type_cast(value)
         | 
| 98 | 
            +
                    cast_value(value) unless value.nil?
         | 
| 99 | 
            +
                  end
         | 
| 100 | 
            +
             | 
| 101 | 
            +
                  # Convenience method for types which do not need separate type casting
         | 
| 102 | 
            +
                  # behavior for user and database inputs. Called by
         | 
| 103 | 
            +
                  # `type_cast_from_database` and `type_cast_from_user` for all values
         | 
| 104 | 
            +
                  # except `nil`.
         | 
| 105 | 
            +
                  def cast_value(value) # :doc:
         | 
| 106 | 
            +
                    value
         | 
| 107 | 
            +
                  end
         | 
| 108 | 
            +
                end
         | 
| 109 | 
            +
              end
         | 
| 110 | 
            +
            end
         | 
| @@ -0,0 +1,23 @@ | |
| 1 | 
            +
            require 'active_record/type/decorator'
         | 
| 2 | 
            +
            require 'active_record/type/mutable'
         | 
| 3 | 
            +
            require 'active_record/type/numeric'
         | 
| 4 | 
            +
            require 'active_record/type/time_value'
         | 
| 5 | 
            +
            require 'active_record/type/value'
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            require 'active_record/type/big_integer'
         | 
| 8 | 
            +
            require 'active_record/type/binary'
         | 
| 9 | 
            +
            require 'active_record/type/boolean'
         | 
| 10 | 
            +
            require 'active_record/type/date'
         | 
| 11 | 
            +
            require 'active_record/type/date_time'
         | 
| 12 | 
            +
            require 'active_record/type/decimal'
         | 
| 13 | 
            +
            require 'active_record/type/decimal_without_scale'
         | 
| 14 | 
            +
            require 'active_record/type/float'
         | 
| 15 | 
            +
            require 'active_record/type/integer'
         | 
| 16 | 
            +
            require 'active_record/type/serialized'
         | 
| 17 | 
            +
            require 'active_record/type/string'
         | 
| 18 | 
            +
            require 'active_record/type/text'
         | 
| 19 | 
            +
            require 'active_record/type/time'
         | 
| 20 | 
            +
            require 'active_record/type/unsigned_integer'
         | 
| 21 | 
            +
             | 
| 22 | 
            +
            require 'active_record/type/type_map'
         | 
| 23 | 
            +
            require 'active_record/type/hash_lookup_type_map'
         | 
| @@ -29,9 +29,11 @@ module ActiveRecord | |
| 29 29 | 
             
                  # Configuration options:
         | 
| 30 30 | 
             
                  #
         | 
| 31 31 | 
             
                  # * <tt>:message</tt> - A custom error message (default is: "is invalid").
         | 
| 32 | 
            -
                  # * <tt>:on</tt> - Specifies  | 
| 33 | 
            -
                  #   validation contexts by default ( | 
| 34 | 
            -
                  #    | 
| 32 | 
            +
                  # * <tt>:on</tt> - Specifies the contexts where this validation is active.
         | 
| 33 | 
            +
                  #   Runs in all validation contexts by default (nil). You can pass a symbol
         | 
| 34 | 
            +
                  #   or an array of symbols. (e.g. <tt>on: :create</tt> or
         | 
| 35 | 
            +
                  #   <tt>on: :custom_validation_context</tt> or
         | 
| 36 | 
            +
                  #   <tt>on: [:create, :custom_validation_context]</tt>)
         | 
| 35 37 | 
             
                  # * <tt>:if</tt> - Specifies a method, proc or string to call to determine
         | 
| 36 38 | 
             
                  #   if the validation should occur (e.g. <tt>if: :allow_validation</tt>,
         | 
| 37 39 | 
             
                  #   or <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method,
         | 
| @@ -44,9 +44,11 @@ module ActiveRecord | |
| 44 44 | 
             
                  #
         | 
| 45 45 | 
             
                  # Configuration options:
         | 
| 46 46 | 
             
                  # * <tt>:message</tt> - A custom error message (default is: "can't be blank").
         | 
| 47 | 
            -
                  # * <tt>:on</tt> - Specifies  | 
| 48 | 
            -
                  #   validation contexts by default ( | 
| 49 | 
            -
                  #    | 
| 47 | 
            +
                  # * <tt>:on</tt> - Specifies the contexts where this validation is active.
         | 
| 48 | 
            +
                  #   Runs in all validation contexts by default (nil). You can pass a symbol
         | 
| 49 | 
            +
                  #   or an array of symbols. (e.g. <tt>on: :create</tt> or
         | 
| 50 | 
            +
                  #   <tt>on: :custom_validation_context</tt> or
         | 
| 51 | 
            +
                  #   <tt>on: [:create, :custom_validation_context]</tt>)
         | 
| 50 52 | 
             
                  # * <tt>:if</tt> - Specifies a method, proc or string to call to determine if
         | 
| 51 53 | 
             
                  #   the validation should occur (e.g. <tt>if: :allow_validation</tt>, or
         | 
| 52 54 | 
             
                  #   <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method, proc
         | 
| @@ -13,14 +13,23 @@ module ActiveRecord | |
| 13 13 | 
             
                  def validate_each(record, attribute, value)
         | 
| 14 14 | 
             
                    finder_class = find_finder_class_for(record)
         | 
| 15 15 | 
             
                    table = finder_class.arel_table
         | 
| 16 | 
            -
                    value = map_enum_attribute(finder_class,attribute,value)
         | 
| 17 | 
            -
             | 
| 18 | 
            -
             | 
| 19 | 
            -
             | 
| 20 | 
            -
             | 
| 21 | 
            -
             | 
| 22 | 
            -
             | 
| 23 | 
            -
             | 
| 16 | 
            +
                    value = map_enum_attribute(finder_class, attribute, value)
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                    begin
         | 
| 19 | 
            +
                      relation = build_relation(finder_class, table, attribute, value)
         | 
| 20 | 
            +
                      if record.persisted?
         | 
| 21 | 
            +
                        if finder_class.primary_key
         | 
| 22 | 
            +
                          relation = relation.and(table[finder_class.primary_key.to_sym].not_eq(record.id))
         | 
| 23 | 
            +
                        else
         | 
| 24 | 
            +
                          raise UnknownPrimaryKey.new(finder_class, "Can not validate uniqueness for persisted record without primary key.")
         | 
| 25 | 
            +
                        end
         | 
| 26 | 
            +
                      end
         | 
| 27 | 
            +
                      relation = scope_relation(record, table, relation)
         | 
| 28 | 
            +
                      relation = finder_class.unscoped.where(relation)
         | 
| 29 | 
            +
                      relation = relation.merge(options[:conditions]) if options[:conditions]
         | 
| 30 | 
            +
                    rescue RangeError
         | 
| 31 | 
            +
                      relation = finder_class.none
         | 
| 32 | 
            +
                    end
         | 
| 24 33 |  | 
| 25 34 | 
             
                    if relation.exists?
         | 
| 26 35 | 
             
                      error_options = options.except(:case_sensitive, :scope, :conditions)
         | 
| @@ -49,7 +58,7 @@ module ActiveRecord | |
| 49 58 | 
             
                  def build_relation(klass, table, attribute, value) #:nodoc:
         | 
| 50 59 | 
             
                    if reflection = klass._reflect_on_association(attribute)
         | 
| 51 60 | 
             
                      attribute = reflection.foreign_key
         | 
| 52 | 
            -
                      value = value.attributes[reflection. | 
| 61 | 
            +
                      value = value.attributes[reflection.klass.primary_key] unless value.nil?
         | 
| 53 62 | 
             
                    end
         | 
| 54 63 |  | 
| 55 64 | 
             
                    attribute_name = attribute.to_s
         | 
| @@ -62,14 +71,15 @@ module ActiveRecord | |
| 62 71 |  | 
| 63 72 | 
             
                    column = klass.columns_hash[attribute_name]
         | 
| 64 73 | 
             
                    value  = klass.connection.type_cast(value, column)
         | 
| 65 | 
            -
                     | 
| 74 | 
            +
                    if value.is_a?(String) && column.limit
         | 
| 75 | 
            +
                      value = value.to_s[0, column.limit]
         | 
| 76 | 
            +
                    end
         | 
| 66 77 |  | 
| 67 78 | 
             
                    if !options[:case_sensitive] && value && column.text?
         | 
| 68 79 | 
             
                      # will use SQL LOWER function before comparison, unless it detects a case insensitive collation
         | 
| 69 80 | 
             
                      klass.connection.case_insensitive_comparison(table, attribute, column, value)
         | 
| 70 81 | 
             
                    else
         | 
| 71 | 
            -
                       | 
| 72 | 
            -
                      table[attribute].eq(value)
         | 
| 82 | 
            +
                      klass.connection.case_sensitive_comparison(table, attribute, column, value)
         | 
| 73 83 | 
             
                    end
         | 
| 74 84 | 
             
                  end
         | 
| 75 85 |  | 
| @@ -79,28 +89,14 @@ module ActiveRecord | |
| 79 89 | 
             
                        scope_value = record.send(reflection.foreign_key)
         | 
| 80 90 | 
             
                        scope_item  = reflection.foreign_key
         | 
| 81 91 | 
             
                      else
         | 
| 82 | 
            -
                        scope_value = record. | 
| 83 | 
            -
                      end
         | 
| 84 | 
            -
             | 
| 85 | 
            -
                      # This is basically emulating an Arel::Nodes::Casted
         | 
| 86 | 
            -
                      column = record.class.columns_hash[scope_item.to_s]
         | 
| 87 | 
            -
                      quoted_value = record.class.connection.quote(scope_value, column)
         | 
| 88 | 
            -
                      unless scope_value.nil?
         | 
| 89 | 
            -
                        node = Arel::Nodes::SqlLiteral.new(quoted_value)
         | 
| 92 | 
            +
                        scope_value = record._read_attribute(scope_item)
         | 
| 90 93 | 
             
                      end
         | 
| 91 | 
            -
             | 
| 92 | 
            -
                      relation = relation.and(table[scope_item].eq(node))
         | 
| 94 | 
            +
                      relation = relation.and(table[scope_item].eq(scope_value))
         | 
| 93 95 | 
             
                    end
         | 
| 94 96 |  | 
| 95 97 | 
             
                    relation
         | 
| 96 98 | 
             
                  end
         | 
| 97 99 |  | 
| 98 | 
            -
                  def deserialize_attribute(record, attribute, value)
         | 
| 99 | 
            -
                    coder = record.class.serialized_attributes[attribute.to_s]
         | 
| 100 | 
            -
                    value = coder.dump value if value && coder
         | 
| 101 | 
            -
                    value
         | 
| 102 | 
            -
                  end
         | 
| 103 | 
            -
             | 
| 104 100 | 
             
                  def map_enum_attribute(klass, attribute, value)
         | 
| 105 101 | 
             
                    mapping = klass.defined_enums[attribute.to_s]
         | 
| 106 102 | 
             
                    value = mapping[value] if value && mapping
         | 
| @@ -166,7 +162,7 @@ module ActiveRecord | |
| 166 162 | 
             
                  #   or <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method,
         | 
| 167 163 | 
             
                  #   proc or string should return or evaluate to a +true+ or +false+ value.
         | 
| 168 164 | 
             
                  # * <tt>:unless</tt> - Specifies a method, proc or string to call to
         | 
| 169 | 
            -
                  #   determine if the validation should  | 
| 165 | 
            +
                  #   determine if the validation should not occur (e.g. <tt>unless: :skip_validation</tt>,
         | 
| 170 166 | 
             
                  #   or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The
         | 
| 171 167 | 
             
                  #   method, proc or string should return or evaluate to a +true+ or +false+
         | 
| 172 168 | 
             
                  #   value.
         |