activerecord 4.1.15 → 4.2.0.beta1
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of activerecord might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/CHANGELOG.md +634 -2176
- data/README.rdoc +15 -10
- data/lib/active_record/aggregations.rb +12 -8
- data/lib/active_record/associations/association.rb +1 -1
- data/lib/active_record/associations/association_scope.rb +53 -21
- data/lib/active_record/associations/belongs_to_association.rb +15 -5
- data/lib/active_record/associations/builder/association.rb +16 -5
- data/lib/active_record/associations/builder/belongs_to.rb +7 -29
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +2 -11
- 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 +32 -44
- data/lib/active_record/associations/collection_proxy.rb +1 -10
- data/lib/active_record/associations/has_many_association.rb +60 -14
- data/lib/active_record/associations/has_many_through_association.rb +34 -23
- data/lib/active_record/associations/has_one_association.rb +0 -1
- data/lib/active_record/associations/join_dependency/join_association.rb +18 -14
- data/lib/active_record/associations/join_dependency.rb +7 -9
- data/lib/active_record/associations/preloader/association.rb +9 -5
- data/lib/active_record/associations/preloader/through_association.rb +3 -3
- data/lib/active_record/associations/preloader.rb +2 -2
- data/lib/active_record/associations/singular_association.rb +16 -1
- data/lib/active_record/associations/through_association.rb +6 -22
- data/lib/active_record/associations.rb +58 -33
- data/lib/active_record/attribute.rb +131 -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 +2 -2
- data/lib/active_record/attribute_methods/dirty.rb +85 -42
- data/lib/active_record/attribute_methods/primary_key.rb +6 -8
- data/lib/active_record/attribute_methods/read.rb +14 -57
- data/lib/active_record/attribute_methods/serialization.rb +12 -146
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +32 -40
- data/lib/active_record/attribute_methods/write.rb +8 -23
- data/lib/active_record/attribute_methods.rb +53 -90
- data/lib/active_record/attribute_set/builder.rb +32 -0
- data/lib/active_record/attribute_set.rb +77 -0
- data/lib/active_record/attributes.rb +122 -0
- data/lib/active_record/autosave_association.rb +11 -21
- data/lib/active_record/base.rb +9 -19
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +69 -45
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +22 -42
- data/lib/active_record/connection_adapters/abstract/quoting.rb +59 -60
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +37 -2
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +102 -21
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +9 -33
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +178 -55
- data/lib/active_record/connection_adapters/abstract/transaction.rb +120 -115
- data/lib/active_record/connection_adapters/abstract_adapter.rb +143 -57
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +156 -107
- data/lib/active_record/connection_adapters/column.rb +13 -244
- data/lib/active_record/connection_adapters/connection_specification.rb +6 -20
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +12 -15
- data/lib/active_record/connection_adapters/mysql_adapter.rb +55 -143
- 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 +39 -20
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +96 -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 +14 -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 +27 -0
- data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +13 -0
- data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +17 -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 +76 -0
- data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +15 -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 +85 -0
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +26 -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 +42 -122
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +4 -4
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +154 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +86 -34
- data/lib/active_record/connection_adapters/postgresql/utils.rb +66 -0
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +188 -452
- data/lib/active_record/connection_adapters/schema_cache.rb +14 -28
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +54 -47
- data/lib/active_record/connection_handling.rb +1 -1
- data/lib/active_record/core.rb +119 -22
- data/lib/active_record/counter_cache.rb +60 -6
- data/lib/active_record/enum.rb +9 -10
- data/lib/active_record/errors.rb +27 -26
- data/lib/active_record/explain.rb +1 -1
- data/lib/active_record/fixtures.rb +52 -45
- data/lib/active_record/gem_version.rb +3 -3
- data/lib/active_record/inheritance.rb +33 -8
- data/lib/active_record/integration.rb +4 -4
- data/lib/active_record/locking/optimistic.rb +34 -16
- 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 +22 -32
- data/lib/active_record/model_schema.rb +39 -48
- data/lib/active_record/nested_attributes.rb +8 -18
- data/lib/active_record/persistence.rb +39 -22
- data/lib/active_record/query_cache.rb +3 -3
- data/lib/active_record/querying.rb +1 -8
- data/lib/active_record/railtie.rb +17 -10
- data/lib/active_record/railties/databases.rake +47 -42
- data/lib/active_record/readonly_attributes.rb +0 -1
- data/lib/active_record/reflection.rb +225 -92
- data/lib/active_record/relation/batches.rb +0 -2
- data/lib/active_record/relation/calculations.rb +28 -32
- data/lib/active_record/relation/delegation.rb +1 -1
- data/lib/active_record/relation/finder_methods.rb +42 -20
- data/lib/active_record/relation/merger.rb +0 -1
- data/lib/active_record/relation/predicate_builder/array_handler.rb +16 -11
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +0 -4
- data/lib/active_record/relation/predicate_builder.rb +1 -22
- data/lib/active_record/relation/query_methods.rb +98 -62
- data/lib/active_record/relation/spawn_methods.rb +6 -7
- data/lib/active_record/relation.rb +35 -11
- data/lib/active_record/result.rb +16 -9
- data/lib/active_record/sanitization.rb +8 -1
- data/lib/active_record/schema.rb +0 -1
- data/lib/active_record/schema_dumper.rb +51 -9
- data/lib/active_record/schema_migration.rb +4 -0
- data/lib/active_record/scoping/default.rb +5 -4
- data/lib/active_record/serializers/xml_serializer.rb +3 -7
- data/lib/active_record/statement_cache.rb +79 -5
- data/lib/active_record/store.rb +5 -5
- data/lib/active_record/tasks/database_tasks.rb +37 -5
- data/lib/active_record/tasks/mysql_database_tasks.rb +1 -1
- data/lib/active_record/tasks/postgresql_database_tasks.rb +2 -2
- data/lib/active_record/timestamp.rb +9 -7
- data/lib/active_record/transactions.rb +35 -21
- data/lib/active_record/type/binary.rb +40 -0
- data/lib/active_record/type/boolean.rb +19 -0
- data/lib/active_record/type/date.rb +46 -0
- data/lib/active_record/type/date_time.rb +43 -0
- data/lib/active_record/type/decimal.rb +40 -0
- data/lib/active_record/type/decimal_without_scale.rb +11 -0
- data/lib/active_record/type/float.rb +19 -0
- data/lib/active_record/type/hash_lookup_type_map.rb +19 -0
- data/lib/active_record/type/integer.rb +23 -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 +51 -0
- data/lib/active_record/type/string.rb +36 -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 +48 -0
- data/lib/active_record/type/value.rb +101 -0
- data/lib/active_record/type.rb +20 -0
- data/lib/active_record/validations/uniqueness.rb +9 -23
- data/lib/active_record/validations.rb +21 -16
- data/lib/active_record.rb +2 -1
- 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 +71 -14
- data/lib/active_record/connection_adapters/postgresql/cast.rb +0 -168
@@ -0,0 +1,19 @@
|
|
1
|
+
module ActiveRecord
|
2
|
+
module Type
|
3
|
+
class HashLookupTypeMap < TypeMap # :nodoc:
|
4
|
+
delegate :key?, to: :@mapping
|
5
|
+
|
6
|
+
def lookup(type, *args)
|
7
|
+
@mapping.fetch(type, proc { default_value }).call(type, *args)
|
8
|
+
end
|
9
|
+
|
10
|
+
def fetch(type, *args, &block)
|
11
|
+
@mapping.fetch(type, block).call(type, *args)
|
12
|
+
end
|
13
|
+
|
14
|
+
def alias_type(type, alias_type)
|
15
|
+
register_type(type) { |_, *args| lookup(alias_type, *args) }
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module ActiveRecord
|
2
|
+
module Type
|
3
|
+
class Integer < Value # :nodoc:
|
4
|
+
include Numeric
|
5
|
+
|
6
|
+
def type
|
7
|
+
:integer
|
8
|
+
end
|
9
|
+
|
10
|
+
alias type_cast_for_database type_cast
|
11
|
+
|
12
|
+
private
|
13
|
+
|
14
|
+
def cast_value(value)
|
15
|
+
case value
|
16
|
+
when true then 1
|
17
|
+
when false then 0
|
18
|
+
else value.to_i rescue nil
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
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,51 @@
|
|
1
|
+
module ActiveRecord
|
2
|
+
module Type
|
3
|
+
class Serialized < SimpleDelegator # :nodoc:
|
4
|
+
include Mutable
|
5
|
+
|
6
|
+
attr_reader :subtype, :coder
|
7
|
+
|
8
|
+
def initialize(subtype, coder)
|
9
|
+
@subtype = subtype
|
10
|
+
@coder = coder
|
11
|
+
super(subtype)
|
12
|
+
end
|
13
|
+
|
14
|
+
def type_cast_from_database(value)
|
15
|
+
if default_value?(value)
|
16
|
+
value
|
17
|
+
else
|
18
|
+
coder.load(super)
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
def type_cast_for_database(value)
|
23
|
+
return if value.nil?
|
24
|
+
unless default_value?(value)
|
25
|
+
super coder.dump(value)
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
def accessor
|
30
|
+
ActiveRecord::Store::IndifferentHashAccessor
|
31
|
+
end
|
32
|
+
|
33
|
+
def init_with(coder)
|
34
|
+
@subtype = coder['subtype']
|
35
|
+
@coder = coder['coder']
|
36
|
+
__setobj__(@subtype)
|
37
|
+
end
|
38
|
+
|
39
|
+
def encode_with(coder)
|
40
|
+
coder['subtype'] = @subtype
|
41
|
+
coder['coder'] = @coder
|
42
|
+
end
|
43
|
+
|
44
|
+
private
|
45
|
+
|
46
|
+
def default_value?(value)
|
47
|
+
value == coder.load(nil)
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,36 @@
|
|
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 "1"
|
19
|
+
when false then "0"
|
20
|
+
else super
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
private
|
25
|
+
|
26
|
+
def cast_value(value)
|
27
|
+
case value
|
28
|
+
when true then "1"
|
29
|
+
when false then "0"
|
30
|
+
# String.new is slightly faster than dup
|
31
|
+
else ::String.new(value.to_s)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
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,48 @@
|
|
1
|
+
module ActiveRecord
|
2
|
+
module Type
|
3
|
+
class TypeMap # :nodoc:
|
4
|
+
def initialize
|
5
|
+
@mapping = {}
|
6
|
+
end
|
7
|
+
|
8
|
+
def lookup(lookup_key, *args)
|
9
|
+
matching_pair = @mapping.reverse_each.detect do |key, _|
|
10
|
+
key === lookup_key
|
11
|
+
end
|
12
|
+
|
13
|
+
if matching_pair
|
14
|
+
matching_pair.last.call(lookup_key, *args)
|
15
|
+
else
|
16
|
+
default_value
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
def register_type(key, value = nil, &block)
|
21
|
+
raise ::ArgumentError unless value || block
|
22
|
+
|
23
|
+
if block
|
24
|
+
@mapping[key] = block
|
25
|
+
else
|
26
|
+
@mapping[key] = proc { value }
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
def alias_type(key, target_key)
|
31
|
+
register_type(key) do |sql_type, *args|
|
32
|
+
metadata = sql_type[/\(.*\)/, 0]
|
33
|
+
lookup("#{target_key}#{metadata}", *args)
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
def clear
|
38
|
+
@mapping.clear
|
39
|
+
end
|
40
|
+
|
41
|
+
private
|
42
|
+
|
43
|
+
def default_value
|
44
|
+
@default_value ||= Value.new
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
48
|
+
end
|
@@ -0,0 +1,101 @@
|
|
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 number? # :nodoc:
|
54
|
+
false
|
55
|
+
end
|
56
|
+
|
57
|
+
def binary? # :nodoc:
|
58
|
+
false
|
59
|
+
end
|
60
|
+
|
61
|
+
def klass # :nodoc:
|
62
|
+
end
|
63
|
+
|
64
|
+
# Determines whether a value has changed for dirty checking. +old_value+
|
65
|
+
# and +new_value+ will always be type-cast. Types should not need to
|
66
|
+
# override this method.
|
67
|
+
def changed?(old_value, new_value, _new_value_before_type_cast)
|
68
|
+
old_value != new_value
|
69
|
+
end
|
70
|
+
|
71
|
+
# Determines whether the mutable value has been modified since it was
|
72
|
+
# read. Returns +false+ by default. This method should not be overridden
|
73
|
+
# directly. Types which return a mutable value should include
|
74
|
+
# +Type::Mutable+, which will define this method.
|
75
|
+
def changed_in_place?(*)
|
76
|
+
false
|
77
|
+
end
|
78
|
+
|
79
|
+
def ==(other)
|
80
|
+
self.class == other.class &&
|
81
|
+
precision == other.precision &&
|
82
|
+
scale == other.scale &&
|
83
|
+
limit == other.limit
|
84
|
+
end
|
85
|
+
|
86
|
+
private
|
87
|
+
|
88
|
+
def type_cast(value)
|
89
|
+
cast_value(value) unless value.nil?
|
90
|
+
end
|
91
|
+
|
92
|
+
# Convenience method for types which do not need separate type casting
|
93
|
+
# behavior for user and database inputs. Called by
|
94
|
+
# `type_cast_from_database` and `type_cast_from_user` for all values
|
95
|
+
# except `nil`.
|
96
|
+
def cast_value(value) # :doc:
|
97
|
+
value
|
98
|
+
end
|
99
|
+
end
|
100
|
+
end
|
101
|
+
end
|
@@ -0,0 +1,20 @@
|
|
1
|
+
require 'active_record/type/mutable'
|
2
|
+
require 'active_record/type/numeric'
|
3
|
+
require 'active_record/type/time_value'
|
4
|
+
require 'active_record/type/value'
|
5
|
+
|
6
|
+
require 'active_record/type/binary'
|
7
|
+
require 'active_record/type/boolean'
|
8
|
+
require 'active_record/type/date'
|
9
|
+
require 'active_record/type/date_time'
|
10
|
+
require 'active_record/type/decimal'
|
11
|
+
require 'active_record/type/decimal_without_scale'
|
12
|
+
require 'active_record/type/float'
|
13
|
+
require 'active_record/type/integer'
|
14
|
+
require 'active_record/type/serialized'
|
15
|
+
require 'active_record/type/string'
|
16
|
+
require 'active_record/type/text'
|
17
|
+
require 'active_record/type/time'
|
18
|
+
|
19
|
+
require 'active_record/type/type_map'
|
20
|
+
require 'active_record/type/hash_lookup_type_map'
|
@@ -13,8 +13,7 @@ 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
|
-
value = deserialize_attribute(record, attribute, value)
|
16
|
+
value = map_enum_attribute(finder_class, attribute, value)
|
18
17
|
|
19
18
|
relation = build_relation(finder_class, table, attribute, value)
|
20
19
|
relation = relation.and(table[finder_class.primary_key.to_sym].not_eq(record.id)) if record.persisted?
|
@@ -49,7 +48,7 @@ module ActiveRecord
|
|
49
48
|
def build_relation(klass, table, attribute, value) #:nodoc:
|
50
49
|
if reflection = klass._reflect_on_association(attribute)
|
51
50
|
attribute = reflection.foreign_key
|
52
|
-
value = value.attributes[reflection.
|
51
|
+
value = value.attributes[reflection.klass.primary_key] unless value.nil?
|
53
52
|
end
|
54
53
|
|
55
54
|
attribute_name = attribute.to_s
|
@@ -62,14 +61,15 @@ module ActiveRecord
|
|
62
61
|
|
63
62
|
column = klass.columns_hash[attribute_name]
|
64
63
|
value = klass.connection.type_cast(value, column)
|
65
|
-
|
64
|
+
if value.is_a?(String) && column.limit
|
65
|
+
value = value.to_s[0, column.limit]
|
66
|
+
end
|
66
67
|
|
67
|
-
if !options[:case_sensitive] && value
|
68
|
+
if !options[:case_sensitive] && value.is_a?(String)
|
68
69
|
# will use SQL LOWER function before comparison, unless it detects a case insensitive collation
|
69
70
|
klass.connection.case_insensitive_comparison(table, attribute, column, value)
|
70
71
|
else
|
71
|
-
|
72
|
-
table[attribute].eq(value)
|
72
|
+
klass.connection.case_sensitive_comparison(table, attribute, column, value)
|
73
73
|
end
|
74
74
|
end
|
75
75
|
|
@@ -81,26 +81,12 @@ module ActiveRecord
|
|
81
81
|
else
|
82
82
|
scope_value = record.read_attribute(scope_item)
|
83
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)
|
90
|
-
end
|
91
|
-
|
92
|
-
relation = relation.and(table[scope_item].eq(node))
|
84
|
+
relation = relation.and(table[scope_item].eq(scope_value))
|
93
85
|
end
|
94
86
|
|
95
87
|
relation
|
96
88
|
end
|
97
89
|
|
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
90
|
def map_enum_attribute(klass, attribute, value)
|
105
91
|
mapping = klass.defined_enums[attribute.to_s]
|
106
92
|
value = mapping[value] if value && mapping
|
@@ -166,7 +152,7 @@ module ActiveRecord
|
|
166
152
|
# or <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method,
|
167
153
|
# proc or string should return or evaluate to a +true+ or +false+ value.
|
168
154
|
# * <tt>:unless</tt> - Specifies a method, proc or string to call to
|
169
|
-
# determine if the validation should
|
155
|
+
# determine if the validation should not occur (e.g. <tt>unless: :skip_validation</tt>,
|
170
156
|
# or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The
|
171
157
|
# method, proc or string should return or evaluate to a +true+ or +false+
|
172
158
|
# value.
|
@@ -29,21 +29,6 @@ module ActiveRecord
|
|
29
29
|
extend ActiveSupport::Concern
|
30
30
|
include ActiveModel::Validations
|
31
31
|
|
32
|
-
module ClassMethods
|
33
|
-
# Creates an object just like Base.create but calls <tt>save!</tt> instead of +save+
|
34
|
-
# so an exception is raised if the record is invalid.
|
35
|
-
def create!(attributes = nil, &block)
|
36
|
-
if attributes.is_a?(Array)
|
37
|
-
attributes.collect { |attr| create!(attr, &block) }
|
38
|
-
else
|
39
|
-
object = new(attributes)
|
40
|
-
yield(object) if block_given?
|
41
|
-
object.save!
|
42
|
-
object
|
43
|
-
end
|
44
|
-
end
|
45
|
-
end
|
46
|
-
|
47
32
|
# The validation process on save can be skipped by passing <tt>validate: false</tt>.
|
48
33
|
# The regular Base#save method is replaced with this when the validations
|
49
34
|
# module is mixed in, which it is by default.
|
@@ -54,12 +39,14 @@ module ActiveRecord
|
|
54
39
|
# Attempts to save the record just like Base#save but will raise a +RecordInvalid+
|
55
40
|
# exception instead of returning +false+ if the record is not valid.
|
56
41
|
def save!(options={})
|
57
|
-
perform_validations(options) ? super :
|
42
|
+
perform_validations(options) ? super : raise_record_invalid
|
58
43
|
end
|
59
44
|
|
60
45
|
# Runs all the validations within the specified context. Returns +true+ if
|
61
46
|
# no errors are found, +false+ otherwise.
|
62
47
|
#
|
48
|
+
# Aliased as validate.
|
49
|
+
#
|
63
50
|
# If the argument is +false+ (default is +nil+), the context is set to <tt>:create</tt> if
|
64
51
|
# <tt>new_record?</tt> is +true+, and to <tt>:update</tt> if it is not.
|
65
52
|
#
|
@@ -71,8 +58,26 @@ module ActiveRecord
|
|
71
58
|
errors.empty? && output
|
72
59
|
end
|
73
60
|
|
61
|
+
alias_method :validate, :valid?
|
62
|
+
|
63
|
+
# Runs all the validations within the specified context. Returns +true+ if
|
64
|
+
# no errors are found, raises +RecordInvalid+ otherwise.
|
65
|
+
#
|
66
|
+
# If the argument is +false+ (default is +nil+), the context is set to <tt>:create</tt> if
|
67
|
+
# <tt>new_record?</tt> is +true+, and to <tt>:update</tt> if it is not.
|
68
|
+
#
|
69
|
+
# Validations with no <tt>:on</tt> option will run no matter the context. Validations with
|
70
|
+
# some <tt>:on</tt> option will only run in the specified context.
|
71
|
+
def validate!(context = nil)
|
72
|
+
valid?(context) || raise_record_invalid
|
73
|
+
end
|
74
|
+
|
74
75
|
protected
|
75
76
|
|
77
|
+
def raise_record_invalid
|
78
|
+
raise(RecordInvalid.new(self))
|
79
|
+
end
|
80
|
+
|
76
81
|
def perform_validations(options={}) # :nodoc:
|
77
82
|
options[:validate] == false || valid?(options[:context])
|
78
83
|
end
|
data/lib/active_record.rb
CHANGED
@@ -27,10 +27,12 @@ require 'active_model'
|
|
27
27
|
require 'arel'
|
28
28
|
|
29
29
|
require 'active_record/version'
|
30
|
+
require 'active_record/attribute_set'
|
30
31
|
|
31
32
|
module ActiveRecord
|
32
33
|
extend ActiveSupport::Autoload
|
33
34
|
|
35
|
+
autoload :Attribute
|
34
36
|
autoload :Base
|
35
37
|
autoload :Callbacks
|
36
38
|
autoload :Core
|
@@ -50,7 +52,6 @@ module ActiveRecord
|
|
50
52
|
autoload :QueryCache
|
51
53
|
autoload :Querying
|
52
54
|
autoload :ReadonlyAttributes
|
53
|
-
autoload :RecordInvalid, 'active_record/validations'
|
54
55
|
autoload :Reflection
|
55
56
|
autoload :RuntimeRegistry
|
56
57
|
autoload :Sanitization
|
@@ -23,16 +23,16 @@ module ActiveRecord
|
|
23
23
|
case file_name
|
24
24
|
when /^(add|remove)_.*_(?:to|from)_(.*)/
|
25
25
|
@migration_action = $1
|
26
|
-
@table_name = $2
|
26
|
+
@table_name = normalize_table_name($2)
|
27
27
|
when /join_table/
|
28
28
|
if attributes.length == 2
|
29
29
|
@migration_action = 'join'
|
30
|
-
@join_tables = attributes.map(&:plural_name)
|
30
|
+
@join_tables = pluralize_table_names? ? attributes.map(&:plural_name) : attributes.map(&:singular_name)
|
31
31
|
|
32
32
|
set_index_names
|
33
33
|
end
|
34
34
|
when /^create_(.+)/
|
35
|
-
@table_name = $1
|
35
|
+
@table_name = normalize_table_name($1)
|
36
36
|
@migration_template = "create_table_migration.rb"
|
37
37
|
end
|
38
38
|
end
|
@@ -55,12 +55,16 @@ module ActiveRecord
|
|
55
55
|
def attributes_with_index
|
56
56
|
attributes.select { |a| !a.reference? && a.has_index? }
|
57
57
|
end
|
58
|
-
|
58
|
+
|
59
59
|
def validate_file_name!
|
60
60
|
unless file_name =~ /^[_a-z0-9]+$/
|
61
61
|
raise IllegalMigrationNameError.new(file_name)
|
62
62
|
end
|
63
63
|
end
|
64
|
+
|
65
|
+
def normalize_table_name(_table_name)
|
66
|
+
pluralize_table_names? ? _table_name.pluralize : _table_name.singularize
|
67
|
+
end
|
64
68
|
end
|
65
69
|
end
|
66
70
|
end
|
@@ -1,7 +1,7 @@
|
|
1
1
|
<% module_namespacing do -%>
|
2
2
|
class <%= class_name %> < <%= parent_class_name.classify %>
|
3
3
|
<% attributes.select(&:reference?).each do |attribute| -%>
|
4
|
-
belongs_to :<%= attribute.name %><%= ', polymorphic: true' if attribute.polymorphic? %>
|
4
|
+
belongs_to :<%= attribute.name %><%= ', polymorphic: true' if attribute.polymorphic? %><%= ', required: true' if attribute.required? %>
|
5
5
|
<% end -%>
|
6
6
|
<% if attributes.any?(&:password_digest?) -%>
|
7
7
|
has_secure_password
|