activerecord 4.2.0 → 5.2.8.1
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 +5 -5
- data/CHANGELOG.md +640 -928
- data/MIT-LICENSE +2 -2
- data/README.rdoc +10 -11
- data/examples/performance.rb +32 -31
- data/examples/simple.rb +5 -4
- data/lib/active_record/aggregations.rb +264 -247
- data/lib/active_record/association_relation.rb +24 -6
- data/lib/active_record/associations/alias_tracker.rb +29 -35
- data/lib/active_record/associations/association.rb +87 -41
- data/lib/active_record/associations/association_scope.rb +106 -132
- data/lib/active_record/associations/belongs_to_association.rb +55 -36
- data/lib/active_record/associations/belongs_to_polymorphic_association.rb +8 -8
- data/lib/active_record/associations/builder/association.rb +29 -38
- data/lib/active_record/associations/builder/belongs_to.rb +77 -30
- data/lib/active_record/associations/builder/collection_association.rb +14 -23
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +50 -39
- data/lib/active_record/associations/builder/has_many.rb +6 -4
- data/lib/active_record/associations/builder/has_one.rb +13 -6
- data/lib/active_record/associations/builder/singular_association.rb +15 -11
- data/lib/active_record/associations/collection_association.rb +145 -266
- data/lib/active_record/associations/collection_proxy.rb +242 -138
- data/lib/active_record/associations/foreign_association.rb +13 -0
- data/lib/active_record/associations/has_many_association.rb +35 -75
- data/lib/active_record/associations/has_many_through_association.rb +51 -69
- data/lib/active_record/associations/has_one_association.rb +39 -24
- data/lib/active_record/associations/has_one_through_association.rb +18 -9
- data/lib/active_record/associations/join_dependency/join_association.rb +40 -81
- data/lib/active_record/associations/join_dependency/join_base.rb +10 -9
- data/lib/active_record/associations/join_dependency/join_part.rb +12 -12
- data/lib/active_record/associations/join_dependency.rb +134 -154
- data/lib/active_record/associations/preloader/association.rb +85 -116
- data/lib/active_record/associations/preloader/through_association.rb +85 -74
- data/lib/active_record/associations/preloader.rb +83 -93
- data/lib/active_record/associations/singular_association.rb +27 -40
- data/lib/active_record/associations/through_association.rb +48 -23
- data/lib/active_record/associations.rb +1732 -1596
- data/lib/active_record/attribute_assignment.rb +58 -182
- data/lib/active_record/attribute_decorators.rb +39 -15
- data/lib/active_record/attribute_methods/before_type_cast.rb +12 -5
- data/lib/active_record/attribute_methods/dirty.rb +94 -125
- data/lib/active_record/attribute_methods/primary_key.rb +86 -71
- data/lib/active_record/attribute_methods/query.rb +4 -2
- data/lib/active_record/attribute_methods/read.rb +45 -63
- data/lib/active_record/attribute_methods/serialization.rb +40 -20
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +62 -36
- data/lib/active_record/attribute_methods/write.rb +31 -46
- data/lib/active_record/attribute_methods.rb +170 -117
- data/lib/active_record/attributes.rb +201 -74
- data/lib/active_record/autosave_association.rb +118 -45
- data/lib/active_record/base.rb +60 -48
- data/lib/active_record/callbacks.rb +97 -57
- data/lib/active_record/coders/json.rb +3 -1
- data/lib/active_record/coders/yaml_column.rb +37 -13
- data/lib/active_record/collection_cache_key.rb +53 -0
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +712 -284
- data/lib/active_record/connection_adapters/abstract/database_limits.rb +10 -5
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +254 -87
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +72 -22
- data/lib/active_record/connection_adapters/abstract/quoting.rb +119 -52
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +6 -4
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +67 -46
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +328 -217
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +81 -36
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +617 -212
- data/lib/active_record/connection_adapters/abstract/transaction.rb +139 -75
- data/lib/active_record/connection_adapters/abstract_adapter.rb +332 -191
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +567 -563
- data/lib/active_record/connection_adapters/column.rb +50 -41
- data/lib/active_record/connection_adapters/connection_specification.rb +147 -135
- 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 +42 -195
- data/lib/active_record/connection_adapters/postgresql/column.rb +35 -11
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +46 -115
- data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +50 -57
- data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +10 -6
- data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +2 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +5 -2
- data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +5 -1
- data/lib/active_record/connection_adapters/postgresql/oid/date.rb +13 -1
- data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +9 -13
- data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +3 -1
- data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +7 -3
- data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +31 -19
- data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +2 -0
- data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +3 -11
- data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +45 -0
- data/lib/active_record/connection_adapters/postgresql/oid/money.rb +7 -9
- data/lib/active_record/connection_adapters/postgresql/oid/{integer.rb → oid.rb} +6 -2
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +33 -11
- data/lib/active_record/connection_adapters/postgresql/oid/range.rb +52 -34
- data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +4 -1
- data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +65 -51
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +5 -3
- data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +3 -1
- data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +3 -1
- data/lib/active_record/connection_adapters/postgresql/oid.rb +23 -25
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +107 -47
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +27 -14
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +65 -0
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +144 -90
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +466 -280
- data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +39 -0
- data/lib/active_record/connection_adapters/postgresql/utils.rb +12 -8
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +439 -330
- data/lib/active_record/connection_adapters/schema_cache.rb +48 -24
- 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 +269 -324
- data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
- data/lib/active_record/connection_handling.rb +40 -27
- data/lib/active_record/core.rb +205 -202
- data/lib/active_record/counter_cache.rb +80 -37
- data/lib/active_record/define_callbacks.rb +22 -0
- data/lib/active_record/dynamic_matchers.rb +87 -105
- data/lib/active_record/enum.rb +136 -90
- data/lib/active_record/errors.rb +180 -52
- data/lib/active_record/explain.rb +23 -11
- data/lib/active_record/explain_registry.rb +4 -2
- data/lib/active_record/explain_subscriber.rb +11 -6
- data/lib/active_record/fixture_set/file.rb +35 -9
- data/lib/active_record/fixtures.rb +193 -135
- data/lib/active_record/gem_version.rb +5 -3
- data/lib/active_record/inheritance.rb +148 -112
- data/lib/active_record/integration.rb +70 -28
- 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 +3 -2
- data/lib/active_record/locking/optimistic.rb +92 -98
- data/lib/active_record/locking/pessimistic.rb +15 -3
- data/lib/active_record/log_subscriber.rb +95 -33
- data/lib/active_record/migration/command_recorder.rb +133 -90
- data/lib/active_record/migration/compatibility.rb +217 -0
- data/lib/active_record/migration/join_table.rb +8 -6
- data/lib/active_record/migration.rb +594 -267
- data/lib/active_record/model_schema.rb +292 -111
- data/lib/active_record/nested_attributes.rb +266 -214
- data/lib/active_record/no_touching.rb +8 -2
- data/lib/active_record/null_relation.rb +24 -37
- data/lib/active_record/persistence.rb +350 -119
- data/lib/active_record/query_cache.rb +13 -24
- data/lib/active_record/querying.rb +19 -17
- data/lib/active_record/railtie.rb +117 -35
- data/lib/active_record/railties/console_sandbox.rb +2 -0
- data/lib/active_record/railties/controller_runtime.rb +9 -3
- data/lib/active_record/railties/databases.rake +160 -174
- data/lib/active_record/readonly_attributes.rb +5 -4
- data/lib/active_record/reflection.rb +447 -288
- data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
- data/lib/active_record/relation/batches.rb +204 -55
- data/lib/active_record/relation/calculations.rb +259 -244
- data/lib/active_record/relation/delegation.rb +67 -60
- data/lib/active_record/relation/finder_methods.rb +290 -253
- data/lib/active_record/relation/from_clause.rb +26 -0
- data/lib/active_record/relation/merger.rb +91 -68
- data/lib/active_record/relation/predicate_builder/array_handler.rb +24 -23
- 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 +7 -1
- data/lib/active_record/relation/predicate_builder.rb +118 -92
- data/lib/active_record/relation/query_attribute.rb +45 -0
- data/lib/active_record/relation/query_methods.rb +446 -389
- data/lib/active_record/relation/record_fetch_warning.rb +51 -0
- data/lib/active_record/relation/spawn_methods.rb +18 -16
- 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 +287 -339
- data/lib/active_record/result.rb +54 -36
- data/lib/active_record/runtime_registry.rb +6 -4
- data/lib/active_record/sanitization.rb +155 -124
- data/lib/active_record/schema.rb +30 -24
- data/lib/active_record/schema_dumper.rb +91 -87
- data/lib/active_record/schema_migration.rb +19 -19
- data/lib/active_record/scoping/default.rb +102 -84
- data/lib/active_record/scoping/named.rb +81 -32
- data/lib/active_record/scoping.rb +45 -26
- data/lib/active_record/secure_token.rb +40 -0
- data/lib/active_record/serialization.rb +5 -5
- data/lib/active_record/statement_cache.rb +45 -35
- data/lib/active_record/store.rb +42 -36
- 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 +136 -95
- data/lib/active_record/tasks/mysql_database_tasks.rb +59 -89
- data/lib/active_record/tasks/postgresql_database_tasks.rb +84 -31
- data/lib/active_record/tasks/sqlite_database_tasks.rb +44 -16
- data/lib/active_record/timestamp.rb +70 -38
- data/lib/active_record/touch_later.rb +64 -0
- data/lib/active_record/transactions.rb +208 -123
- 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 +4 -41
- data/lib/active_record/type/date_time.rb +4 -38
- data/lib/active_record/type/decimal_without_scale.rb +6 -2
- data/lib/active_record/type/hash_lookup_type_map.rb +13 -5
- 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 +30 -15
- data/lib/active_record/type/text.rb +2 -2
- data/lib/active_record/type/time.rb +11 -16
- data/lib/active_record/type/type_map.rb +15 -17
- data/lib/active_record/type/unsigned_integer.rb +9 -7
- data/lib/active_record/type.rb +79 -23
- 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 +13 -4
- data/lib/active_record/validations/length.rb +26 -0
- data/lib/active_record/validations/presence.rb +14 -13
- data/lib/active_record/validations/uniqueness.rb +41 -32
- data/lib/active_record/validations.rb +38 -35
- data/lib/active_record/version.rb +3 -1
- data/lib/active_record.rb +36 -21
- 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 +43 -35
- data/lib/rails/generators/active_record/migration/templates/{create_table_migration.rb → create_table_migration.rb.tt} +8 -6
- data/lib/rails/generators/active_record/migration/templates/{migration.rb → migration.rb.tt} +8 -7
- data/lib/rails/generators/active_record/migration.rb +18 -1
- data/lib/rails/generators/active_record/model/model_generator.rb +18 -22
- data/lib/rails/generators/active_record/model/templates/{model.rb → model.rb.tt} +3 -0
- data/lib/rails/generators/active_record.rb +7 -5
- metadata +77 -53
- 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_many.rb +0 -17
- data/lib/active_record/associations/preloader/has_many_through.rb +0 -19
- 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.rb +0 -149
- data/lib/active_record/attribute_set/builder.rb +0 -86
- data/lib/active_record/attribute_set.rb +0 -77
- data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -491
- data/lib/active_record/connection_adapters/postgresql/array_parser.rb +0 -93
- data/lib/active_record/connection_adapters/postgresql/oid/float.rb +0 -21
- data/lib/active_record/connection_adapters/postgresql/oid/infinity.rb +0 -13
- data/lib/active_record/connection_adapters/postgresql/oid/json.rb +0 -35
- data/lib/active_record/connection_adapters/postgresql/oid/time.rb +0 -11
- data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
- data/lib/active_record/serializers/xml_serializer.rb +0 -193
- data/lib/active_record/type/big_integer.rb +0 -13
- data/lib/active_record/type/binary.rb +0 -50
- data/lib/active_record/type/boolean.rb +0 -30
- data/lib/active_record/type/decimal.rb +0 -40
- data/lib/active_record/type/decorator.rb +0 -14
- data/lib/active_record/type/float.rb +0 -19
- data/lib/active_record/type/integer.rb +0 -55
- data/lib/active_record/type/mutable.rb +0 -16
- data/lib/active_record/type/numeric.rb +0 -36
- data/lib/active_record/type/string.rb +0 -36
- data/lib/active_record/type/time_value.rb +0 -38
- data/lib/active_record/type/value.rb +0 -101
- /data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +0 -0
@@ -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,11 +1,19 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveRecord
|
2
4
|
module Validations
|
3
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
|
@@ -24,13 +32,14 @@ module ActiveRecord
|
|
24
32
|
#
|
25
33
|
# NOTE: This validation will not fail if the association hasn't been
|
26
34
|
# assigned. If you want to ensure that the association is both present and
|
27
|
-
# guaranteed to be valid, you also need to use
|
35
|
+
# guaranteed to be valid, you also need to use
|
36
|
+
# {validates_presence_of}[rdoc-ref:Validations::ClassMethods#validates_presence_of].
|
28
37
|
#
|
29
38
|
# Configuration options:
|
30
39
|
#
|
31
40
|
# * <tt>:message</tt> - A custom error message (default is: "is invalid").
|
32
41
|
# * <tt>:on</tt> - Specifies the contexts where this validation is active.
|
33
|
-
# Runs in all validation contexts by default
|
42
|
+
# Runs in all validation contexts by default +nil+. You can pass a symbol
|
34
43
|
# or an array of symbols. (e.g. <tt>on: :create</tt> or
|
35
44
|
# <tt>on: :custom_validation_context</tt> or
|
36
45
|
# <tt>on: [:create, :custom_validation_context]</tt>)
|
@@ -0,0 +1,26 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveRecord
|
4
|
+
module Validations
|
5
|
+
class LengthValidator < ActiveModel::Validations::LengthValidator # :nodoc:
|
6
|
+
def validate_each(record, attribute, association_or_value)
|
7
|
+
if association_or_value.respond_to?(:loaded?) && association_or_value.loaded?
|
8
|
+
association_or_value = association_or_value.target.reject(&:marked_for_destruction?)
|
9
|
+
end
|
10
|
+
super
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
module ClassMethods
|
15
|
+
# Validates that the specified attributes match the length restrictions supplied.
|
16
|
+
# If the attribute is an association, records that are marked for destruction are not counted.
|
17
|
+
#
|
18
|
+
# See ActiveModel::Validations::HelperMethods.validates_length_of for more information.
|
19
|
+
def validates_length_of(*attr_names)
|
20
|
+
validates_with LengthValidator, _merge_attributes(attr_names)
|
21
|
+
end
|
22
|
+
|
23
|
+
alias_method :validates_size_of, :validates_length_of
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
@@ -1,17 +1,13 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveRecord
|
2
4
|
module Validations
|
3
5
|
class PresenceValidator < ActiveModel::Validations::PresenceValidator # :nodoc:
|
4
|
-
def
|
5
|
-
|
6
|
-
|
7
|
-
next unless record.class._reflect_on_association(attribute)
|
8
|
-
associated_records = Array.wrap(record.send(attribute))
|
9
|
-
|
10
|
-
# Superclass validates presence. Ensure present records aren't about to be destroyed.
|
11
|
-
if associated_records.present? && associated_records.all? { |r| r.marked_for_destruction? }
|
12
|
-
record.errors.add(attribute, :blank, options)
|
13
|
-
end
|
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?)
|
14
9
|
end
|
10
|
+
super
|
15
11
|
end
|
16
12
|
end
|
17
13
|
|
@@ -36,16 +32,21 @@ module ActiveRecord
|
|
36
32
|
# This is due to the way Object#blank? handles boolean values:
|
37
33
|
# <tt>false.blank? # => true</tt>.
|
38
34
|
#
|
39
|
-
# This validator defers to the
|
35
|
+
# This validator defers to the Active Model validation for presence, adding the
|
40
36
|
# check to see that an associated object is not marked for destruction. This
|
41
37
|
# prevents the parent object from validating successfully and saving, which then
|
42
38
|
# deletes the associated object, thus putting the parent object into an invalid
|
43
39
|
# state.
|
44
40
|
#
|
41
|
+
# NOTE: This validation will not fail while using it with an association
|
42
|
+
# if the latter was assigned but not valid. If you want to ensure that
|
43
|
+
# it is both present and valid, you also need to use
|
44
|
+
# {validates_associated}[rdoc-ref:Validations::ClassMethods#validates_associated].
|
45
|
+
#
|
45
46
|
# Configuration options:
|
46
47
|
# * <tt>:message</tt> - A custom error message (default is: "can't be blank").
|
47
48
|
# * <tt>:on</tt> - Specifies the contexts where this validation is active.
|
48
|
-
# Runs in all validation contexts by default
|
49
|
+
# Runs in all validation contexts by default +nil+. You can pass a symbol
|
49
50
|
# or an array of symbols. (e.g. <tt>on: :create</tt> or
|
50
51
|
# <tt>on: :custom_validation_context</tt> or
|
51
52
|
# <tt>on: [:create, :custom_validation_context]</tt>)
|
@@ -58,7 +59,7 @@ module ActiveRecord
|
|
58
59
|
# or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The method,
|
59
60
|
# proc or string should return or evaluate to a +true+ or +false+ value.
|
60
61
|
# * <tt>:strict</tt> - Specifies whether validation should be strict.
|
61
|
-
# See
|
62
|
+
# See ActiveModel::Validations#validates! for more information.
|
62
63
|
def validates_presence_of(*attr_names)
|
63
64
|
validates_with PresenceValidator, _merge_attributes(attr_names)
|
64
65
|
end
|
@@ -1,3 +1,5 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveRecord
|
2
4
|
module Validations
|
3
5
|
class UniquenessValidator < ActiveModel::EachValidator # :nodoc:
|
@@ -6,19 +8,27 @@ module ActiveRecord
|
|
6
8
|
raise ArgumentError, "#{options[:conditions]} was passed as :conditions but is not callable. " \
|
7
9
|
"Pass a callable instead: `conditions: -> { where(approved: true) }`"
|
8
10
|
end
|
11
|
+
unless Array(options[:scope]).all? { |scope| scope.respond_to?(:to_sym) }
|
12
|
+
raise ArgumentError, "#{options[:scope]} is not supported format for :scope option. " \
|
13
|
+
"Pass a symbol or an array of symbols instead: `scope: :user_id`"
|
14
|
+
end
|
9
15
|
super({ case_sensitive: true }.merge!(options))
|
10
16
|
@klass = options[:class]
|
11
17
|
end
|
12
18
|
|
13
19
|
def validate_each(record, attribute, value)
|
14
20
|
finder_class = find_finder_class_for(record)
|
15
|
-
table = finder_class.arel_table
|
16
21
|
value = map_enum_attribute(finder_class, attribute, value)
|
17
22
|
|
18
|
-
relation = build_relation(finder_class,
|
19
|
-
|
20
|
-
|
21
|
-
|
23
|
+
relation = build_relation(finder_class, attribute, value)
|
24
|
+
if record.persisted?
|
25
|
+
if finder_class.primary_key
|
26
|
+
relation = relation.where.not(finder_class.primary_key => record.id_in_database)
|
27
|
+
else
|
28
|
+
raise UnknownPrimaryKey.new(finder_class, "Can not validate uniqueness for persisted record without primary key.")
|
29
|
+
end
|
30
|
+
end
|
31
|
+
relation = scope_relation(record, relation)
|
22
32
|
relation = relation.merge(options[:conditions]) if options[:conditions]
|
23
33
|
|
24
34
|
if relation.exists?
|
@@ -29,13 +39,13 @@ module ActiveRecord
|
|
29
39
|
end
|
30
40
|
end
|
31
41
|
|
32
|
-
|
42
|
+
private
|
33
43
|
# The check for an existing value should be run from a class that
|
34
44
|
# isn't abstract. This means working down from the current class
|
35
45
|
# (self), to the first non-abstract class. Since classes don't know
|
36
46
|
# their subclasses, we have to build the hierarchy between self and
|
37
47
|
# the record's class.
|
38
|
-
def find_finder_class_for(record)
|
48
|
+
def find_finder_class_for(record)
|
39
49
|
class_hierarchy = [record.class]
|
40
50
|
|
41
51
|
while class_hierarchy.first != @klass
|
@@ -45,43 +55,44 @@ module ActiveRecord
|
|
45
55
|
class_hierarchy.detect { |klass| !klass.abstract_class? }
|
46
56
|
end
|
47
57
|
|
48
|
-
def build_relation(klass,
|
58
|
+
def build_relation(klass, attribute, value)
|
49
59
|
if reflection = klass._reflect_on_association(attribute)
|
50
60
|
attribute = reflection.foreign_key
|
51
61
|
value = value.attributes[reflection.klass.primary_key] unless value.nil?
|
52
62
|
end
|
53
63
|
|
54
|
-
|
64
|
+
if value.nil?
|
65
|
+
return klass.unscoped.where!(attribute => value)
|
66
|
+
end
|
55
67
|
|
56
68
|
# the attribute may be an aliased attribute
|
57
|
-
if klass.
|
58
|
-
attribute = klass.
|
59
|
-
attribute_name = attribute.to_s
|
69
|
+
if klass.attribute_alias?(attribute)
|
70
|
+
attribute = klass.attribute_alias(attribute)
|
60
71
|
end
|
61
72
|
|
73
|
+
attribute_name = attribute.to_s
|
74
|
+
value = klass.predicate_builder.build_bind_attribute(attribute_name, value)
|
75
|
+
|
76
|
+
table = klass.arel_table
|
62
77
|
column = klass.columns_hash[attribute_name]
|
63
|
-
value = klass.connection.type_cast(value, column)
|
64
|
-
if value.is_a?(String) && column.limit
|
65
|
-
value = value.to_s[0, column.limit]
|
66
|
-
end
|
67
78
|
|
68
|
-
if !options[:case_sensitive]
|
79
|
+
comparison = if !options[:case_sensitive]
|
69
80
|
# will use SQL LOWER function before comparison, unless it detects a case insensitive collation
|
70
81
|
klass.connection.case_insensitive_comparison(table, attribute, column, value)
|
71
82
|
else
|
72
83
|
klass.connection.case_sensitive_comparison(table, attribute, column, value)
|
73
84
|
end
|
85
|
+
klass.unscoped.where!(comparison)
|
74
86
|
end
|
75
87
|
|
76
|
-
def scope_relation(record,
|
88
|
+
def scope_relation(record, relation)
|
77
89
|
Array(options[:scope]).each do |scope_item|
|
78
|
-
|
79
|
-
|
80
|
-
scope_item = reflection.foreign_key
|
90
|
+
scope_value = if record.class._reflect_on_association(scope_item)
|
91
|
+
record.association(scope_item).reader
|
81
92
|
else
|
82
|
-
|
93
|
+
record._read_attribute(scope_item)
|
83
94
|
end
|
84
|
-
relation = relation.
|
95
|
+
relation = relation.where(scope_item => scope_value)
|
85
96
|
end
|
86
97
|
|
87
98
|
relation
|
@@ -159,7 +170,8 @@ module ActiveRecord
|
|
159
170
|
#
|
160
171
|
# === Concurrency and integrity
|
161
172
|
#
|
162
|
-
# Using this validation method in conjunction with
|
173
|
+
# Using this validation method in conjunction with
|
174
|
+
# {ActiveRecord::Base#save}[rdoc-ref:Persistence#save]
|
163
175
|
# does not guarantee the absence of duplicate record insertions, because
|
164
176
|
# uniqueness checks on the application level are inherently prone to race
|
165
177
|
# conditions. For example, suppose that two users try to post a Comment at
|
@@ -193,21 +205,19 @@ module ActiveRecord
|
|
193
205
|
# | # Boom! We now have a duplicate
|
194
206
|
# | # title!
|
195
207
|
#
|
196
|
-
#
|
197
|
-
#
|
198
|
-
#
|
199
|
-
# ActiveRecord::ConnectionAdapters::SchemaStatements#add_index. In the
|
200
|
-
# rare case that a race condition occurs, the database will guarantee
|
208
|
+
# The best way to work around this problem is to add a unique index to the database table using
|
209
|
+
# {connection.add_index}[rdoc-ref:ConnectionAdapters::SchemaStatements#add_index].
|
210
|
+
# In the rare case that a race condition occurs, the database will guarantee
|
201
211
|
# the field's uniqueness.
|
202
212
|
#
|
203
213
|
# When the database catches such a duplicate insertion,
|
204
|
-
# ActiveRecord::Base#save will raise an ActiveRecord::StatementInvalid
|
214
|
+
# {ActiveRecord::Base#save}[rdoc-ref:Persistence#save] will raise an ActiveRecord::StatementInvalid
|
205
215
|
# exception. You can either choose to let this error propagate (which
|
206
216
|
# will result in the default Rails exception page being shown), or you
|
207
217
|
# can catch it and restart the transaction (e.g. by telling the user
|
208
218
|
# that the title already exists, and asking them to re-enter the title).
|
209
219
|
# This technique is also known as
|
210
|
-
# {optimistic concurrency control}[
|
220
|
+
# {optimistic concurrency control}[https://en.wikipedia.org/wiki/Optimistic_concurrency_control].
|
211
221
|
#
|
212
222
|
# The bundled ActiveRecord::ConnectionAdapters distinguish unique index
|
213
223
|
# constraint errors from other types of database errors by throwing an
|
@@ -217,7 +227,6 @@ module ActiveRecord
|
|
217
227
|
#
|
218
228
|
# The following bundled adapters throw the ActiveRecord::RecordNotUnique exception:
|
219
229
|
#
|
220
|
-
# * ActiveRecord::ConnectionAdapters::MysqlAdapter.
|
221
230
|
# * ActiveRecord::ConnectionAdapters::Mysql2Adapter.
|
222
231
|
# * ActiveRecord::ConnectionAdapters::SQLite3Adapter.
|
223
232
|
# * ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.
|
@@ -1,8 +1,11 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveRecord
|
2
|
-
# = Active Record RecordInvalid
|
4
|
+
# = Active Record \RecordInvalid
|
3
5
|
#
|
4
|
-
# Raised by
|
5
|
-
#
|
6
|
+
# Raised by {ActiveRecord::Base#save!}[rdoc-ref:Persistence#save!] and
|
7
|
+
# {ActiveRecord::Base#create!}[rdoc-ref:Persistence::ClassMethods#create!] when the record is invalid.
|
8
|
+
# Use the #record method to retrieve the record which did not validate.
|
6
9
|
#
|
7
10
|
# begin
|
8
11
|
# complex_operation_that_internally_calls_save!
|
@@ -12,74 +15,72 @@ module ActiveRecord
|
|
12
15
|
class RecordInvalid < ActiveRecordError
|
13
16
|
attr_reader :record
|
14
17
|
|
15
|
-
def initialize(record)
|
16
|
-
|
17
|
-
|
18
|
-
|
18
|
+
def initialize(record = nil)
|
19
|
+
if record
|
20
|
+
@record = record
|
21
|
+
errors = @record.errors.full_messages.join(", ")
|
22
|
+
message = I18n.t(:"#{@record.class.i18n_scope}.errors.messages.record_invalid", errors: errors, default: :"errors.messages.record_invalid")
|
23
|
+
else
|
24
|
+
message = "Record invalid"
|
25
|
+
end
|
26
|
+
|
27
|
+
super(message)
|
19
28
|
end
|
20
29
|
end
|
21
30
|
|
22
|
-
# = Active Record Validations
|
31
|
+
# = Active Record \Validations
|
23
32
|
#
|
24
|
-
# Active Record includes the majority of its validations from
|
33
|
+
# Active Record includes the majority of its validations from ActiveModel::Validations
|
25
34
|
# all of which accept the <tt>:on</tt> argument to define the context where the
|
26
35
|
# validations are active. Active Record will always supply either the context of
|
27
36
|
# <tt>:create</tt> or <tt>:update</tt> dependent on whether the model is a
|
28
|
-
#
|
37
|
+
# {new_record?}[rdoc-ref:Persistence#new_record?].
|
29
38
|
module Validations
|
30
39
|
extend ActiveSupport::Concern
|
31
40
|
include ActiveModel::Validations
|
32
41
|
|
33
42
|
# The validation process on save can be skipped by passing <tt>validate: false</tt>.
|
34
|
-
# The regular Base#save method is replaced
|
35
|
-
# module is mixed in, which it is by default.
|
36
|
-
def save(options={})
|
43
|
+
# The regular {ActiveRecord::Base#save}[rdoc-ref:Persistence#save] method is replaced
|
44
|
+
# with this when the validations module is mixed in, which it is by default.
|
45
|
+
def save(options = {})
|
37
46
|
perform_validations(options) ? super : false
|
38
47
|
end
|
39
48
|
|
40
|
-
# Attempts to save the record just like Base#save but
|
41
|
-
# exception instead of returning +false+ if the record is not valid.
|
42
|
-
def save!(options={})
|
43
|
-
perform_validations(options) ? super :
|
49
|
+
# Attempts to save the record just like {ActiveRecord::Base#save}[rdoc-ref:Base#save] but
|
50
|
+
# will raise an ActiveRecord::RecordInvalid exception instead of returning +false+ if the record is not valid.
|
51
|
+
def save!(options = {})
|
52
|
+
perform_validations(options) ? super : raise_validation_error
|
44
53
|
end
|
45
54
|
|
46
55
|
# Runs all the validations within the specified context. Returns +true+ if
|
47
56
|
# no errors are found, +false+ otherwise.
|
48
57
|
#
|
49
|
-
# Aliased as validate.
|
58
|
+
# Aliased as #validate.
|
50
59
|
#
|
51
60
|
# If the argument is +false+ (default is +nil+), the context is set to <tt>:create</tt> if
|
52
|
-
#
|
61
|
+
# {new_record?}[rdoc-ref:Persistence#new_record?] is +true+, and to <tt>:update</tt> if it is not.
|
53
62
|
#
|
54
|
-
# Validations with no <tt>:on</tt> option will run no matter the context. Validations with
|
63
|
+
# \Validations with no <tt>:on</tt> option will run no matter the context. \Validations with
|
55
64
|
# some <tt>:on</tt> option will only run in the specified context.
|
56
65
|
def valid?(context = nil)
|
57
|
-
context ||=
|
66
|
+
context ||= default_validation_context
|
58
67
|
output = super(context)
|
59
68
|
errors.empty? && output
|
60
69
|
end
|
61
70
|
|
62
71
|
alias_method :validate, :valid?
|
63
72
|
|
64
|
-
|
65
|
-
# no errors are found, raises +RecordInvalid+ otherwise.
|
66
|
-
#
|
67
|
-
# If the argument is +false+ (default is +nil+), the context is set to <tt>:create</tt> if
|
68
|
-
# <tt>new_record?</tt> is +true+, and to <tt>:update</tt> if it is not.
|
69
|
-
#
|
70
|
-
# Validations with no <tt>:on</tt> option will run no matter the context. Validations with
|
71
|
-
# some <tt>:on</tt> option will only run in the specified context.
|
72
|
-
def validate!(context = nil)
|
73
|
-
valid?(context) || raise_record_invalid
|
74
|
-
end
|
73
|
+
private
|
75
74
|
|
76
|
-
|
75
|
+
def default_validation_context
|
76
|
+
new_record? ? :create : :update
|
77
|
+
end
|
77
78
|
|
78
|
-
def
|
79
|
+
def raise_validation_error
|
79
80
|
raise(RecordInvalid.new(self))
|
80
81
|
end
|
81
82
|
|
82
|
-
def perform_validations(options={})
|
83
|
+
def perform_validations(options = {})
|
83
84
|
options[:validate] == false || valid?(options[:context])
|
84
85
|
end
|
85
86
|
end
|
@@ -88,3 +89,5 @@ end
|
|
88
89
|
require "active_record/validations/associated"
|
89
90
|
require "active_record/validations/uniqueness"
|
90
91
|
require "active_record/validations/presence"
|
92
|
+
require "active_record/validations/absence"
|
93
|
+
require "active_record/validations/length"
|
data/lib/active_record.rb
CHANGED
@@ -1,5 +1,7 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
#--
|
2
|
-
# Copyright (c) 2004-
|
4
|
+
# Copyright (c) 2004-2018 David Heinemeier Hansson
|
3
5
|
#
|
4
6
|
# Permission is hereby granted, free of charge, to any person obtaining
|
5
7
|
# a copy of this software and associated documentation files (the
|
@@ -21,18 +23,18 @@
|
|
21
23
|
# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
22
24
|
#++
|
23
25
|
|
24
|
-
require
|
25
|
-
require
|
26
|
-
require
|
27
|
-
require
|
26
|
+
require "active_support"
|
27
|
+
require "active_support/rails"
|
28
|
+
require "active_model"
|
29
|
+
require "arel"
|
30
|
+
require "yaml"
|
28
31
|
|
29
|
-
require
|
30
|
-
require
|
32
|
+
require "active_record/version"
|
33
|
+
require "active_model/attribute_set"
|
31
34
|
|
32
35
|
module ActiveRecord
|
33
36
|
extend ActiveSupport::Autoload
|
34
37
|
|
35
|
-
autoload :Attribute
|
36
38
|
autoload :Base
|
37
39
|
autoload :Callbacks
|
38
40
|
autoload :Core
|
@@ -40,18 +42,22 @@ module ActiveRecord
|
|
40
42
|
autoload :CounterCache
|
41
43
|
autoload :DynamicMatchers
|
42
44
|
autoload :Enum
|
45
|
+
autoload :InternalMetadata
|
43
46
|
autoload :Explain
|
44
47
|
autoload :Inheritance
|
45
48
|
autoload :Integration
|
46
49
|
autoload :Migration
|
47
|
-
autoload :Migrator,
|
50
|
+
autoload :Migrator, "active_record/migration"
|
48
51
|
autoload :ModelSchema
|
49
52
|
autoload :NestedAttributes
|
50
53
|
autoload :NoTouching
|
54
|
+
autoload :TouchLater
|
51
55
|
autoload :Persistence
|
52
56
|
autoload :QueryCache
|
53
57
|
autoload :Querying
|
58
|
+
autoload :CollectionCacheKey
|
54
59
|
autoload :ReadonlyAttributes
|
60
|
+
autoload :RecordInvalid, "active_record/validations"
|
55
61
|
autoload :Reflection
|
56
62
|
autoload :RuntimeRegistry
|
57
63
|
autoload :Sanitization
|
@@ -62,15 +68,17 @@ module ActiveRecord
|
|
62
68
|
autoload :Serialization
|
63
69
|
autoload :StatementCache
|
64
70
|
autoload :Store
|
71
|
+
autoload :Suppressor
|
65
72
|
autoload :Timestamp
|
66
73
|
autoload :Transactions
|
67
74
|
autoload :Translation
|
68
75
|
autoload :Validations
|
76
|
+
autoload :SecureToken
|
69
77
|
|
70
78
|
eager_autoload do
|
71
|
-
autoload :ActiveRecordError,
|
72
|
-
autoload :ConnectionNotEstablished,
|
73
|
-
autoload :ConnectionAdapters,
|
79
|
+
autoload :ActiveRecordError, "active_record/errors"
|
80
|
+
autoload :ConnectionNotEstablished, "active_record/errors"
|
81
|
+
autoload :ConnectionAdapters, "active_record/connection_adapters/abstract_adapter"
|
74
82
|
|
75
83
|
autoload :Aggregations
|
76
84
|
autoload :Associations
|
@@ -78,11 +86,13 @@ module ActiveRecord
|
|
78
86
|
autoload :AttributeMethods
|
79
87
|
autoload :AutosaveAssociation
|
80
88
|
|
89
|
+
autoload :LegacyYamlAdapter
|
90
|
+
|
81
91
|
autoload :Relation
|
82
92
|
autoload :AssociationRelation
|
83
93
|
autoload :NullRelation
|
84
94
|
|
85
|
-
autoload_under
|
95
|
+
autoload_under "relation" do
|
86
96
|
autoload :QueryMethods
|
87
97
|
autoload :FinderMethods
|
88
98
|
autoload :Calculations
|
@@ -93,11 +103,13 @@ module ActiveRecord
|
|
93
103
|
end
|
94
104
|
|
95
105
|
autoload :Result
|
106
|
+
autoload :TableMetadata
|
107
|
+
autoload :Type
|
96
108
|
end
|
97
109
|
|
98
110
|
module Coders
|
99
|
-
autoload :YAMLColumn,
|
100
|
-
autoload :JSON,
|
111
|
+
autoload :YAMLColumn, "active_record/coders/yaml_column"
|
112
|
+
autoload :JSON, "active_record/coders/json"
|
101
113
|
end
|
102
114
|
|
103
115
|
module AttributeMethods
|
@@ -129,7 +141,6 @@ module ActiveRecord
|
|
129
141
|
|
130
142
|
eager_autoload do
|
131
143
|
autoload :AbstractAdapter
|
132
|
-
autoload :ConnectionManagement, "active_record/connection_adapters/abstract/connection_pool"
|
133
144
|
end
|
134
145
|
end
|
135
146
|
|
@@ -146,13 +157,13 @@ module ActiveRecord
|
|
146
157
|
extend ActiveSupport::Autoload
|
147
158
|
|
148
159
|
autoload :DatabaseTasks
|
149
|
-
autoload :SQLiteDatabaseTasks,
|
150
|
-
autoload :MySQLDatabaseTasks,
|
160
|
+
autoload :SQLiteDatabaseTasks, "active_record/tasks/sqlite_database_tasks"
|
161
|
+
autoload :MySQLDatabaseTasks, "active_record/tasks/mysql_database_tasks"
|
151
162
|
autoload :PostgreSQLDatabaseTasks,
|
152
|
-
|
163
|
+
"active_record/tasks/postgresql_database_tasks"
|
153
164
|
end
|
154
165
|
|
155
|
-
autoload :TestFixtures,
|
166
|
+
autoload :TestFixtures, "active_record/fixtures"
|
156
167
|
|
157
168
|
def self.eager_load!
|
158
169
|
super
|
@@ -169,5 +180,9 @@ ActiveSupport.on_load(:active_record) do
|
|
169
180
|
end
|
170
181
|
|
171
182
|
ActiveSupport.on_load(:i18n) do
|
172
|
-
I18n.load_path << File.
|
183
|
+
I18n.load_path << File.expand_path("active_record/locale/en.yml", __dir__)
|
173
184
|
end
|
185
|
+
|
186
|
+
YAML.load_tags["!ruby/object:ActiveRecord::AttributeSet"] = "ActiveModel::AttributeSet"
|
187
|
+
YAML.load_tags["!ruby/object:ActiveRecord::Attribute::FromDatabase"] = "ActiveModel::Attribute::FromDatabase"
|
188
|
+
YAML.load_tags["!ruby/object:ActiveRecord::LazyAttributeHash"] = "ActiveModel::LazyAttributeHash"
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "rails/generators/active_record"
|
4
|
+
|
5
|
+
module ActiveRecord
|
6
|
+
module Generators # :nodoc:
|
7
|
+
class ApplicationRecordGenerator < ::Rails::Generators::Base # :nodoc:
|
8
|
+
source_root File.expand_path("templates", __dir__)
|
9
|
+
|
10
|
+
# FIXME: Change this file to a symlink once RubyGems 2.5.0 is required.
|
11
|
+
def create_application_record
|
12
|
+
template "application_record.rb", application_record_file_name
|
13
|
+
end
|
14
|
+
|
15
|
+
private
|
16
|
+
|
17
|
+
def application_record_file_name
|
18
|
+
@application_record_file_name ||=
|
19
|
+
if namespaced?
|
20
|
+
"app/models/#{namespaced_path}/application_record.rb"
|
21
|
+
else
|
22
|
+
"app/models/application_record.rb"
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|