activerecord 7.0.0 → 7.1.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +1607 -1040
- data/MIT-LICENSE +1 -1
- data/README.rdoc +17 -18
- data/lib/active_record/aggregations.rb +16 -13
- data/lib/active_record/association_relation.rb +1 -1
- data/lib/active_record/associations/association.rb +18 -3
- data/lib/active_record/associations/association_scope.rb +16 -9
- data/lib/active_record/associations/belongs_to_association.rb +14 -6
- data/lib/active_record/associations/builder/association.rb +3 -3
- data/lib/active_record/associations/builder/belongs_to.rb +21 -8
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +1 -5
- data/lib/active_record/associations/builder/singular_association.rb +4 -0
- data/lib/active_record/associations/collection_association.rb +17 -12
- data/lib/active_record/associations/collection_proxy.rb +22 -12
- data/lib/active_record/associations/foreign_association.rb +10 -3
- data/lib/active_record/associations/has_many_association.rb +27 -17
- data/lib/active_record/associations/has_many_through_association.rb +10 -6
- data/lib/active_record/associations/has_one_association.rb +10 -3
- data/lib/active_record/associations/join_dependency.rb +20 -14
- data/lib/active_record/associations/preloader/association.rb +27 -6
- data/lib/active_record/associations/preloader/through_association.rb +1 -1
- data/lib/active_record/associations/preloader.rb +13 -10
- data/lib/active_record/associations/singular_association.rb +1 -1
- data/lib/active_record/associations/through_association.rb +22 -11
- data/lib/active_record/associations.rb +345 -219
- data/lib/active_record/attribute_assignment.rb +0 -2
- data/lib/active_record/attribute_methods/before_type_cast.rb +17 -0
- data/lib/active_record/attribute_methods/dirty.rb +40 -26
- data/lib/active_record/attribute_methods/primary_key.rb +76 -24
- data/lib/active_record/attribute_methods/query.rb +28 -16
- data/lib/active_record/attribute_methods/read.rb +18 -5
- data/lib/active_record/attribute_methods/serialization.rb +172 -69
- data/lib/active_record/attribute_methods/write.rb +3 -3
- data/lib/active_record/attribute_methods.rb +110 -28
- data/lib/active_record/attributes.rb +3 -3
- data/lib/active_record/autosave_association.rb +56 -10
- data/lib/active_record/base.rb +10 -5
- data/lib/active_record/callbacks.rb +16 -32
- data/lib/active_record/coders/column_serializer.rb +61 -0
- data/lib/active_record/coders/json.rb +1 -1
- data/lib/active_record/coders/yaml_column.rb +70 -34
- data/lib/active_record/connection_adapters/abstract/connection_handler.rb +164 -89
- data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +2 -0
- data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +3 -1
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +63 -43
- data/lib/active_record/connection_adapters/abstract/database_limits.rb +5 -0
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +128 -32
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +60 -22
- data/lib/active_record/connection_adapters/abstract/quoting.rb +52 -8
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +4 -3
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +18 -4
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +163 -29
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +14 -1
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +302 -129
- data/lib/active_record/connection_adapters/abstract/transaction.rb +287 -58
- data/lib/active_record/connection_adapters/abstract_adapter.rb +504 -106
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +217 -104
- data/lib/active_record/connection_adapters/column.rb +9 -0
- data/lib/active_record/connection_adapters/mysql/column.rb +1 -0
- data/lib/active_record/connection_adapters/mysql/database_statements.rb +23 -144
- data/lib/active_record/connection_adapters/mysql/quoting.rb +29 -12
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +9 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +10 -1
- data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +8 -2
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +38 -14
- data/lib/active_record/connection_adapters/mysql2/database_statements.rb +148 -0
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +98 -53
- data/lib/active_record/connection_adapters/pool_config.rb +14 -5
- data/lib/active_record/connection_adapters/pool_manager.rb +19 -9
- data/lib/active_record/connection_adapters/postgresql/column.rb +3 -2
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +72 -45
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +1 -1
- data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +2 -2
- data/lib/active_record/connection_adapters/postgresql/oid/range.rb +11 -2
- data/lib/active_record/connection_adapters/postgresql/oid/timestamp_with_time_zone.rb +3 -1
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +41 -8
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +6 -10
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +76 -6
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +131 -2
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +53 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +358 -57
- data/lib/active_record/connection_adapters/postgresql/utils.rb +9 -10
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +343 -181
- data/lib/active_record/connection_adapters/schema_cache.rb +287 -59
- data/lib/active_record/connection_adapters/sqlite3/column.rb +49 -0
- data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +45 -39
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +22 -5
- data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +7 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +41 -22
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +242 -81
- data/lib/active_record/connection_adapters/statement_pool.rb +7 -0
- data/lib/active_record/connection_adapters/trilogy/database_statements.rb +98 -0
- data/lib/active_record/connection_adapters/trilogy_adapter.rb +254 -0
- data/lib/active_record/connection_adapters.rb +3 -1
- data/lib/active_record/connection_handling.rb +73 -96
- data/lib/active_record/core.rb +136 -148
- data/lib/active_record/counter_cache.rb +46 -25
- data/lib/active_record/database_configurations/connection_url_resolver.rb +1 -0
- data/lib/active_record/database_configurations/database_config.rb +9 -3
- data/lib/active_record/database_configurations/hash_config.rb +22 -12
- data/lib/active_record/database_configurations/url_config.rb +17 -11
- data/lib/active_record/database_configurations.rb +87 -34
- data/lib/active_record/delegated_type.rb +9 -4
- data/lib/active_record/deprecator.rb +7 -0
- data/lib/active_record/destroy_association_async_job.rb +2 -0
- data/lib/active_record/disable_joins_association_relation.rb +1 -1
- data/lib/active_record/encryption/auto_filtered_parameters.rb +66 -0
- data/lib/active_record/encryption/cipher/aes256_gcm.rb +4 -1
- data/lib/active_record/encryption/config.rb +25 -1
- data/lib/active_record/encryption/configurable.rb +13 -14
- data/lib/active_record/encryption/context.rb +10 -3
- data/lib/active_record/encryption/contexts.rb +8 -4
- data/lib/active_record/encryption/derived_secret_key_provider.rb +9 -3
- data/lib/active_record/encryption/deterministic_key_provider.rb +1 -1
- data/lib/active_record/encryption/encryptable_record.rb +38 -22
- data/lib/active_record/encryption/encrypted_attribute_type.rb +19 -8
- data/lib/active_record/encryption/encryptor.rb +7 -7
- data/lib/active_record/encryption/envelope_encryption_key_provider.rb +3 -3
- data/lib/active_record/encryption/extended_deterministic_queries.rb +83 -71
- data/lib/active_record/encryption/extended_deterministic_uniqueness_validator.rb +3 -3
- data/lib/active_record/encryption/key_generator.rb +12 -1
- data/lib/active_record/encryption/message.rb +1 -1
- data/lib/active_record/encryption/message_serializer.rb +2 -0
- data/lib/active_record/encryption/properties.rb +4 -4
- data/lib/active_record/encryption/scheme.rb +20 -23
- data/lib/active_record/encryption.rb +1 -0
- data/lib/active_record/enum.rb +114 -27
- data/lib/active_record/errors.rb +108 -15
- data/lib/active_record/explain.rb +23 -3
- data/lib/active_record/explain_subscriber.rb +1 -1
- data/lib/active_record/fixture_set/model_metadata.rb +14 -4
- data/lib/active_record/fixture_set/render_context.rb +2 -0
- data/lib/active_record/fixture_set/table_row.rb +29 -8
- data/lib/active_record/fixtures.rb +121 -73
- data/lib/active_record/future_result.rb +30 -5
- data/lib/active_record/gem_version.rb +2 -2
- data/lib/active_record/inheritance.rb +30 -16
- data/lib/active_record/insert_all.rb +55 -8
- data/lib/active_record/integration.rb +10 -10
- data/lib/active_record/internal_metadata.rb +118 -30
- data/lib/active_record/locking/optimistic.rb +32 -18
- data/lib/active_record/locking/pessimistic.rb +8 -5
- data/lib/active_record/log_subscriber.rb +39 -17
- data/lib/active_record/marshalling.rb +56 -0
- data/lib/active_record/message_pack.rb +124 -0
- data/lib/active_record/middleware/database_selector/resolver.rb +4 -0
- data/lib/active_record/middleware/database_selector.rb +18 -13
- data/lib/active_record/middleware/shard_selector.rb +7 -5
- data/lib/active_record/migration/command_recorder.rb +104 -9
- data/lib/active_record/migration/compatibility.rb +158 -64
- data/lib/active_record/migration/default_strategy.rb +23 -0
- data/lib/active_record/migration/execution_strategy.rb +19 -0
- data/lib/active_record/migration.rb +271 -117
- data/lib/active_record/model_schema.rb +82 -50
- data/lib/active_record/nested_attributes.rb +23 -3
- data/lib/active_record/normalization.rb +159 -0
- data/lib/active_record/persistence.rb +200 -47
- data/lib/active_record/promise.rb +84 -0
- data/lib/active_record/query_cache.rb +3 -21
- data/lib/active_record/query_logs.rb +87 -51
- data/lib/active_record/query_logs_formatter.rb +41 -0
- data/lib/active_record/querying.rb +16 -3
- data/lib/active_record/railtie.rb +127 -61
- data/lib/active_record/railties/controller_runtime.rb +12 -8
- data/lib/active_record/railties/databases.rake +142 -143
- data/lib/active_record/railties/job_runtime.rb +23 -0
- data/lib/active_record/readonly_attributes.rb +32 -5
- data/lib/active_record/reflection.rb +177 -45
- data/lib/active_record/relation/batches/batch_enumerator.rb +5 -3
- data/lib/active_record/relation/batches.rb +190 -61
- data/lib/active_record/relation/calculations.rb +200 -83
- data/lib/active_record/relation/delegation.rb +23 -9
- data/lib/active_record/relation/finder_methods.rb +77 -16
- data/lib/active_record/relation/merger.rb +2 -0
- data/lib/active_record/relation/predicate_builder/association_query_value.rb +31 -3
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +4 -6
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +5 -1
- data/lib/active_record/relation/predicate_builder.rb +26 -14
- data/lib/active_record/relation/query_attribute.rb +25 -1
- data/lib/active_record/relation/query_methods.rb +429 -76
- data/lib/active_record/relation/spawn_methods.rb +18 -1
- data/lib/active_record/relation.rb +98 -41
- data/lib/active_record/result.rb +25 -9
- data/lib/active_record/runtime_registry.rb +10 -1
- data/lib/active_record/sanitization.rb +57 -16
- data/lib/active_record/schema.rb +36 -22
- data/lib/active_record/schema_dumper.rb +65 -23
- data/lib/active_record/schema_migration.rb +68 -33
- data/lib/active_record/scoping/default.rb +20 -12
- data/lib/active_record/scoping/named.rb +2 -2
- data/lib/active_record/scoping.rb +2 -1
- data/lib/active_record/secure_password.rb +60 -0
- data/lib/active_record/secure_token.rb +21 -3
- data/lib/active_record/serialization.rb +5 -0
- data/lib/active_record/signed_id.rb +9 -7
- data/lib/active_record/store.rb +16 -11
- data/lib/active_record/suppressor.rb +3 -1
- data/lib/active_record/table_metadata.rb +16 -3
- data/lib/active_record/tasks/database_tasks.rb +138 -107
- data/lib/active_record/tasks/mysql_database_tasks.rb +15 -6
- data/lib/active_record/tasks/postgresql_database_tasks.rb +17 -15
- data/lib/active_record/tasks/sqlite_database_tasks.rb +15 -7
- data/lib/active_record/test_fixtures.rb +123 -99
- data/lib/active_record/timestamp.rb +26 -14
- data/lib/active_record/token_for.rb +113 -0
- data/lib/active_record/touch_later.rb +11 -6
- data/lib/active_record/transactions.rb +39 -13
- data/lib/active_record/translation.rb +1 -1
- data/lib/active_record/type/adapter_specific_registry.rb +1 -8
- data/lib/active_record/type/internal/timezone.rb +7 -2
- data/lib/active_record/type/serialized.rb +8 -4
- data/lib/active_record/type/time.rb +4 -0
- data/lib/active_record/validations/absence.rb +1 -1
- data/lib/active_record/validations/associated.rb +3 -3
- data/lib/active_record/validations/numericality.rb +5 -4
- data/lib/active_record/validations/presence.rb +5 -28
- data/lib/active_record/validations/uniqueness.rb +50 -5
- data/lib/active_record/validations.rb +8 -4
- data/lib/active_record/version.rb +1 -1
- data/lib/active_record.rb +143 -16
- data/lib/arel/errors.rb +10 -0
- data/lib/arel/factory_methods.rb +4 -0
- data/lib/arel/filter_predications.rb +1 -1
- data/lib/arel/nodes/and.rb +4 -0
- data/lib/arel/nodes/binary.rb +6 -1
- data/lib/arel/nodes/bound_sql_literal.rb +61 -0
- data/lib/arel/nodes/cte.rb +36 -0
- data/lib/arel/nodes/filter.rb +1 -1
- data/lib/arel/nodes/fragments.rb +35 -0
- data/lib/arel/nodes/homogeneous_in.rb +0 -8
- data/lib/arel/nodes/leading_join.rb +8 -0
- data/lib/arel/nodes/node.rb +111 -2
- data/lib/arel/nodes/sql_literal.rb +6 -0
- data/lib/arel/nodes/table_alias.rb +4 -0
- data/lib/arel/nodes.rb +4 -0
- data/lib/arel/predications.rb +2 -0
- data/lib/arel/table.rb +9 -5
- data/lib/arel/visitors/mysql.rb +8 -1
- data/lib/arel/visitors/to_sql.rb +81 -17
- data/lib/arel/visitors/visitor.rb +2 -2
- data/lib/arel.rb +16 -2
- data/lib/rails/generators/active_record/application_record/USAGE +8 -0
- data/lib/rails/generators/active_record/migration.rb +3 -1
- data/lib/rails/generators/active_record/model/USAGE +113 -0
- data/lib/rails/generators/active_record/model/model_generator.rb +15 -6
- metadata +50 -15
- data/lib/active_record/connection_adapters/legacy_pool_manager.rb +0 -35
- data/lib/active_record/null_relation.rb +0 -63
@@ -8,15 +8,21 @@ module ActiveRecord
|
|
8
8
|
attr_reader :on_duplicate, :update_only, :returning, :unique_by, :update_sql
|
9
9
|
|
10
10
|
def initialize(model, inserts, on_duplicate:, update_only: nil, returning: nil, unique_by: nil, record_timestamps: nil)
|
11
|
-
|
12
|
-
|
13
|
-
@model, @connection, @inserts, @keys = model, model.connection, inserts, inserts.first.keys.map(&:to_s)
|
11
|
+
@model, @connection, @inserts = model, model.connection, inserts.map(&:stringify_keys)
|
14
12
|
@on_duplicate, @update_only, @returning, @unique_by = on_duplicate, update_only, returning, unique_by
|
15
13
|
@record_timestamps = record_timestamps.nil? ? model.record_timestamps : record_timestamps
|
16
14
|
|
17
15
|
disallow_raw_sql!(on_duplicate)
|
18
16
|
disallow_raw_sql!(returning)
|
19
17
|
|
18
|
+
if @inserts.empty?
|
19
|
+
@keys = []
|
20
|
+
else
|
21
|
+
resolve_sti
|
22
|
+
resolve_attribute_aliases
|
23
|
+
@keys = @inserts.first.keys
|
24
|
+
end
|
25
|
+
|
20
26
|
configure_on_duplicate_update_logic
|
21
27
|
|
22
28
|
if model.scope_attributes?
|
@@ -28,13 +34,15 @@ module ActiveRecord
|
|
28
34
|
@returning = (connection.supports_insert_returning? ? primary_keys : false) if @returning.nil?
|
29
35
|
@returning = false if @returning == []
|
30
36
|
|
31
|
-
@unique_by = find_unique_index_for(unique_by)
|
37
|
+
@unique_by = find_unique_index_for(@unique_by)
|
32
38
|
@on_duplicate = :skip if @on_duplicate == :update && updatable_columns.empty?
|
33
39
|
|
34
40
|
ensure_valid_options_for_connection!
|
35
41
|
end
|
36
42
|
|
37
43
|
def execute
|
44
|
+
return ActiveRecord::Result.empty if inserts.empty?
|
45
|
+
|
38
46
|
message = +"#{model} "
|
39
47
|
message << "Bulk " if inserts.many?
|
40
48
|
message << (on_duplicate == :update ? "Upsert" : "Insert")
|
@@ -76,7 +84,7 @@ module ActiveRecord
|
|
76
84
|
@record_timestamps
|
77
85
|
end
|
78
86
|
|
79
|
-
# TODO: Consider
|
87
|
+
# TODO: Consider renaming this method, as it only conditionally extends keys, not always
|
80
88
|
def keys_including_timestamps
|
81
89
|
@keys_including_timestamps ||= if record_timestamps?
|
82
90
|
keys + model.all_timestamp_attributes_in_model
|
@@ -88,6 +96,34 @@ module ActiveRecord
|
|
88
96
|
private
|
89
97
|
attr_reader :scope_attributes
|
90
98
|
|
99
|
+
def has_attribute_aliases?(attributes)
|
100
|
+
attributes.keys.any? { |attribute| model.attribute_alias?(attribute) }
|
101
|
+
end
|
102
|
+
|
103
|
+
def resolve_sti
|
104
|
+
return if model.descends_from_active_record?
|
105
|
+
|
106
|
+
sti_type = model.sti_name
|
107
|
+
@inserts = @inserts.map do |insert|
|
108
|
+
insert.reverse_merge(model.inheritance_column.to_s => sti_type)
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
def resolve_attribute_aliases
|
113
|
+
return unless has_attribute_aliases?(@inserts.first)
|
114
|
+
|
115
|
+
@inserts = @inserts.map do |insert|
|
116
|
+
insert.transform_keys { |attribute| resolve_attribute_alias(attribute) }
|
117
|
+
end
|
118
|
+
|
119
|
+
@update_only = Array(@update_only).map { |attribute| resolve_attribute_alias(attribute) } if @update_only
|
120
|
+
@unique_by = Array(@unique_by).map { |attribute| resolve_attribute_alias(attribute) } if @unique_by
|
121
|
+
end
|
122
|
+
|
123
|
+
def resolve_attribute_alias(attribute)
|
124
|
+
model.attribute_alias(attribute) || attribute
|
125
|
+
end
|
126
|
+
|
91
127
|
def configure_on_duplicate_update_logic
|
92
128
|
if custom_update_sql_provided? && update_only.present?
|
93
129
|
raise ArgumentError, "You can't set :update_only and provide custom update SQL via :on_duplicate at the same time"
|
@@ -115,8 +151,9 @@ module ActiveRecord
|
|
115
151
|
|
116
152
|
name_or_columns = unique_by || model.primary_key
|
117
153
|
match = Array(name_or_columns).map(&:to_s)
|
154
|
+
sorted_match = match.sort
|
118
155
|
|
119
|
-
if index = unique_indexes.find { |i| match.include?(i.name) || i.columns ==
|
156
|
+
if index = unique_indexes.find { |i| match.include?(i.name) || Array(i.columns).sort == sorted_match }
|
120
157
|
index
|
121
158
|
elsif match == primary_keys
|
122
159
|
unique_by.nil? ? nil : ActiveRecord::ConnectionAdapters::IndexDefinition.new(model.table_name, "#{model.table_name}_primary_key", true, match)
|
@@ -212,7 +249,13 @@ module ActiveRecord
|
|
212
249
|
if insert_all.returning.is_a?(String)
|
213
250
|
insert_all.returning
|
214
251
|
else
|
215
|
-
|
252
|
+
Array(insert_all.returning).map do |attribute|
|
253
|
+
if model.attribute_alias?(attribute)
|
254
|
+
"#{quote_column(model.attribute_alias(attribute))} AS #{quote_column(attribute)}"
|
255
|
+
else
|
256
|
+
quote_column(attribute)
|
257
|
+
end
|
258
|
+
end.join(",")
|
216
259
|
end
|
217
260
|
end
|
218
261
|
|
@@ -271,7 +314,11 @@ module ActiveRecord
|
|
271
314
|
end
|
272
315
|
|
273
316
|
def quote_columns(columns)
|
274
|
-
columns.map(
|
317
|
+
columns.map { |column| quote_column(column) }
|
318
|
+
end
|
319
|
+
|
320
|
+
def quote_column(column)
|
321
|
+
connection.quote_column_name(column)
|
275
322
|
end
|
276
323
|
end
|
277
324
|
end
|
@@ -10,7 +10,7 @@ module ActiveRecord
|
|
10
10
|
##
|
11
11
|
# :singleton-method:
|
12
12
|
# Indicates the format used to generate the timestamp in the cache key, if
|
13
|
-
# versioning is off. Accepts any of the symbols in
|
13
|
+
# versioning is off. Accepts any of the symbols in +Time::DATE_FORMATS+.
|
14
14
|
#
|
15
15
|
# This is +:usec+, by default.
|
16
16
|
class_attribute :cache_timestamp_format, instance_writer: false, default: :usec
|
@@ -20,7 +20,7 @@ module ActiveRecord
|
|
20
20
|
# Indicates whether to use a stable #cache_key method that is accompanied
|
21
21
|
# by a changing version in the #cache_version method.
|
22
22
|
#
|
23
|
-
# This is +true+, by default on Rails 5.2 and above.
|
23
|
+
# This is +true+, by default on \Rails 5.2 and above.
|
24
24
|
class_attribute :cache_versioning, instance_writer: false, default: false
|
25
25
|
|
26
26
|
##
|
@@ -28,7 +28,7 @@ module ActiveRecord
|
|
28
28
|
# Indicates whether to use a stable #cache_key method that is accompanied
|
29
29
|
# by a changing version in the #cache_version method on collections.
|
30
30
|
#
|
31
|
-
# This is +false+, by default until Rails 6.1.
|
31
|
+
# This is +false+, by default until \Rails 6.1.
|
32
32
|
class_attribute :collection_cache_versioning, instance_writer: false, default: false
|
33
33
|
end
|
34
34
|
|
@@ -55,8 +55,8 @@ module ActiveRecord
|
|
55
55
|
# user = User.find_by(name: 'Phusion')
|
56
56
|
# user_path(user) # => "/users/Phusion"
|
57
57
|
def to_param
|
58
|
-
|
59
|
-
id
|
58
|
+
return unless id
|
59
|
+
Array(id).join(self.class.param_delimiter)
|
60
60
|
end
|
61
61
|
|
62
62
|
# Returns a stable cache key that can be used to identify this record.
|
@@ -64,7 +64,7 @@ module ActiveRecord
|
|
64
64
|
# Product.new.cache_key # => "products/new"
|
65
65
|
# Product.find(5).cache_key # => "products/5"
|
66
66
|
#
|
67
|
-
# If ActiveRecord::Base.cache_versioning is turned off, as it was in Rails 5.1 and earlier,
|
67
|
+
# If ActiveRecord::Base.cache_versioning is turned off, as it was in \Rails 5.1 and earlier,
|
68
68
|
# the cache key will also include a version.
|
69
69
|
#
|
70
70
|
# Product.cache_versioning = false
|
@@ -79,7 +79,7 @@ module ActiveRecord
|
|
79
79
|
timestamp = max_updated_column_timestamp
|
80
80
|
|
81
81
|
if timestamp
|
82
|
-
timestamp = timestamp.utc.
|
82
|
+
timestamp = timestamp.utc.to_fs(cache_timestamp_format)
|
83
83
|
"#{model_name.cache_key}/#{id}-#{timestamp}"
|
84
84
|
else
|
85
85
|
"#{model_name.cache_key}/#{id}"
|
@@ -103,10 +103,10 @@ module ActiveRecord
|
|
103
103
|
raw_timestamp_to_cache_version(timestamp)
|
104
104
|
|
105
105
|
elsif timestamp = updated_at
|
106
|
-
timestamp.utc.
|
106
|
+
timestamp.utc.to_fs(cache_timestamp_format)
|
107
107
|
end
|
108
108
|
elsif self.class.has_attribute?("updated_at")
|
109
|
-
raise ActiveModel::MissingAttributeError, "missing attribute
|
109
|
+
raise ActiveModel::MissingAttributeError, "missing attribute 'updated_at' for #{self.class}"
|
110
110
|
end
|
111
111
|
end
|
112
112
|
|
@@ -178,7 +178,7 @@ module ActiveRecord
|
|
178
178
|
def can_use_fast_cache_version?(timestamp)
|
179
179
|
timestamp.is_a?(String) &&
|
180
180
|
cache_timestamp_format == :usec &&
|
181
|
-
|
181
|
+
self.class.connection.default_timezone == :utc &&
|
182
182
|
!updated_at_came_from_user?
|
183
183
|
end
|
184
184
|
|
@@ -9,52 +9,140 @@ module ActiveRecord
|
|
9
9
|
#
|
10
10
|
# This is enabled by default. To disable this functionality set
|
11
11
|
# `use_metadata_table` to false in your database configuration.
|
12
|
-
class InternalMetadata
|
13
|
-
|
12
|
+
class InternalMetadata # :nodoc:
|
13
|
+
class NullInternalMetadata
|
14
|
+
end
|
14
15
|
|
15
|
-
|
16
|
-
def enabled?
|
17
|
-
ActiveRecord::Base.connection.use_metadata_table?
|
18
|
-
end
|
16
|
+
attr_reader :connection, :arel_table
|
19
17
|
|
20
|
-
|
21
|
-
|
22
|
-
|
18
|
+
def initialize(connection)
|
19
|
+
@connection = connection
|
20
|
+
@arel_table = Arel::Table.new(table_name)
|
21
|
+
end
|
23
22
|
|
24
|
-
|
25
|
-
|
26
|
-
|
23
|
+
def enabled?
|
24
|
+
connection.use_metadata_table?
|
25
|
+
end
|
27
26
|
|
28
|
-
|
29
|
-
|
27
|
+
def primary_key
|
28
|
+
"key"
|
29
|
+
end
|
30
|
+
|
31
|
+
def value_key
|
32
|
+
"value"
|
33
|
+
end
|
34
|
+
|
35
|
+
def table_name
|
36
|
+
"#{ActiveRecord::Base.table_name_prefix}#{ActiveRecord::Base.internal_metadata_table_name}#{ActiveRecord::Base.table_name_suffix}"
|
37
|
+
end
|
30
38
|
|
31
|
-
|
39
|
+
def []=(key, value)
|
40
|
+
return unless enabled?
|
41
|
+
|
42
|
+
update_or_create_entry(key, value)
|
43
|
+
end
|
44
|
+
|
45
|
+
def [](key)
|
46
|
+
return unless enabled?
|
47
|
+
|
48
|
+
if entry = select_entry(key)
|
49
|
+
entry[value_key]
|
32
50
|
end
|
51
|
+
end
|
33
52
|
|
34
|
-
|
35
|
-
|
53
|
+
def delete_all_entries
|
54
|
+
dm = Arel::DeleteManager.new(arel_table)
|
55
|
+
|
56
|
+
connection.delete(dm, "#{self.class} Destroy")
|
57
|
+
end
|
58
|
+
|
59
|
+
def count
|
60
|
+
sm = Arel::SelectManager.new(arel_table)
|
61
|
+
sm.project(*Arel::Nodes::Count.new([Arel.star]))
|
62
|
+
|
63
|
+
connection.select_values(sm, "#{self.class} Count").first
|
64
|
+
end
|
36
65
|
|
37
|
-
|
66
|
+
def create_table_and_set_flags(environment, schema_sha1 = nil)
|
67
|
+
create_table
|
68
|
+
update_or_create_entry(:environment, environment)
|
69
|
+
update_or_create_entry(:schema_sha1, schema_sha1) if schema_sha1
|
70
|
+
end
|
71
|
+
|
72
|
+
# Creates an internal metadata table with columns +key+ and +value+
|
73
|
+
def create_table
|
74
|
+
return unless enabled?
|
75
|
+
|
76
|
+
unless connection.table_exists?(table_name)
|
77
|
+
connection.create_table(table_name, id: false) do |t|
|
78
|
+
t.string :key, **connection.internal_string_options_for_primary_key
|
79
|
+
t.string :value
|
80
|
+
t.timestamps
|
81
|
+
end
|
38
82
|
end
|
83
|
+
end
|
84
|
+
|
85
|
+
def drop_table
|
86
|
+
return unless enabled?
|
87
|
+
|
88
|
+
connection.drop_table table_name, if_exists: true
|
89
|
+
end
|
90
|
+
|
91
|
+
def table_exists?
|
92
|
+
connection.schema_cache.data_source_exists?(table_name)
|
93
|
+
end
|
39
94
|
|
40
|
-
|
41
|
-
def
|
42
|
-
|
95
|
+
private
|
96
|
+
def update_or_create_entry(key, value)
|
97
|
+
entry = select_entry(key)
|
43
98
|
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
99
|
+
if entry
|
100
|
+
if entry[value_key] != value
|
101
|
+
update_entry(key, value)
|
102
|
+
else
|
103
|
+
entry[value_key]
|
49
104
|
end
|
105
|
+
else
|
106
|
+
create_entry(key, value)
|
50
107
|
end
|
51
108
|
end
|
52
109
|
|
53
|
-
def
|
54
|
-
|
110
|
+
def current_time
|
111
|
+
connection.default_timezone == :utc ? Time.now.utc : Time.now
|
112
|
+
end
|
113
|
+
|
114
|
+
def create_entry(key, value)
|
115
|
+
im = Arel::InsertManager.new(arel_table)
|
116
|
+
im.insert [
|
117
|
+
[arel_table[primary_key], key],
|
118
|
+
[arel_table[value_key], value],
|
119
|
+
[arel_table[:created_at], current_time],
|
120
|
+
[arel_table[:updated_at], current_time]
|
121
|
+
]
|
55
122
|
|
56
|
-
connection.
|
123
|
+
connection.insert(im, "#{self.class} Create", primary_key, key)
|
124
|
+
end
|
125
|
+
|
126
|
+
def update_entry(key, new_value)
|
127
|
+
um = Arel::UpdateManager.new(arel_table)
|
128
|
+
um.set [
|
129
|
+
[arel_table[value_key], new_value],
|
130
|
+
[arel_table[:updated_at], current_time]
|
131
|
+
]
|
132
|
+
|
133
|
+
um.where(arel_table[primary_key].eq(key))
|
134
|
+
|
135
|
+
connection.update(um, "#{self.class} Update")
|
136
|
+
end
|
137
|
+
|
138
|
+
def select_entry(key)
|
139
|
+
sm = Arel::SelectManager.new(arel_table)
|
140
|
+
sm.project(Arel::Nodes::SqlLiteral.new("*"))
|
141
|
+
sm.where(arel_table[primary_key].eq(Arel::Nodes::BindParam.new(key)))
|
142
|
+
sm.order(arel_table[primary_key].asc)
|
143
|
+
sm.limit = 1
|
144
|
+
|
145
|
+
connection.select_all(sm, "#{self.class} Load").first
|
57
146
|
end
|
58
|
-
end
|
59
147
|
end
|
60
148
|
end
|
@@ -2,14 +2,14 @@
|
|
2
2
|
|
3
3
|
module ActiveRecord
|
4
4
|
module Locking
|
5
|
-
# == What is Optimistic Locking
|
5
|
+
# == What is \Optimistic \Locking
|
6
6
|
#
|
7
7
|
# Optimistic locking allows multiple users to access the same record for edits, and assumes a minimum of
|
8
8
|
# conflicts with the data. It does this by checking whether another process has made changes to a record since
|
9
|
-
# it was opened, an
|
9
|
+
# it was opened, an ActiveRecord::StaleObjectError exception is thrown if that has occurred
|
10
10
|
# and the update is ignored.
|
11
11
|
#
|
12
|
-
# Check out
|
12
|
+
# Check out +ActiveRecord::Locking::Pessimistic+ for an alternative.
|
13
13
|
#
|
14
14
|
# == Usage
|
15
15
|
#
|
@@ -69,6 +69,11 @@ module ActiveRecord
|
|
69
69
|
end
|
70
70
|
end
|
71
71
|
|
72
|
+
def initialize_dup(other) # :nodoc:
|
73
|
+
super
|
74
|
+
_clear_locking_column if locking_enabled?
|
75
|
+
end
|
76
|
+
|
72
77
|
private
|
73
78
|
def _create_record(attribute_names = self.attribute_names)
|
74
79
|
if locking_enabled?
|
@@ -91,8 +96,7 @@ module ActiveRecord
|
|
91
96
|
locking_column = self.class.locking_column
|
92
97
|
lock_attribute_was = @attributes[locking_column]
|
93
98
|
|
94
|
-
update_constraints =
|
95
|
-
update_constraints[locking_column] = _lock_value_for_database(locking_column)
|
99
|
+
update_constraints = _query_constraints_hash
|
96
100
|
|
97
101
|
attribute_names = attribute_names.dup if attribute_names.frozen?
|
98
102
|
attribute_names << locking_column
|
@@ -118,16 +122,9 @@ module ActiveRecord
|
|
118
122
|
end
|
119
123
|
|
120
124
|
def destroy_row
|
121
|
-
|
122
|
-
|
123
|
-
locking_column = self.class.locking_column
|
125
|
+
affected_rows = super
|
124
126
|
|
125
|
-
|
126
|
-
delete_constraints[locking_column] = _lock_value_for_database(locking_column)
|
127
|
-
|
128
|
-
affected_rows = self.class._delete_record(delete_constraints)
|
129
|
-
|
130
|
-
if affected_rows != 1
|
127
|
+
if locking_enabled? && affected_rows != 1
|
131
128
|
raise ActiveRecord::StaleObjectError.new(self, "destroy")
|
132
129
|
end
|
133
130
|
|
@@ -142,6 +139,18 @@ module ActiveRecord
|
|
142
139
|
end
|
143
140
|
end
|
144
141
|
|
142
|
+
def _clear_locking_column
|
143
|
+
self[self.class.locking_column] = nil
|
144
|
+
clear_attribute_change(self.class.locking_column)
|
145
|
+
end
|
146
|
+
|
147
|
+
def _query_constraints_hash
|
148
|
+
return super unless locking_enabled?
|
149
|
+
|
150
|
+
locking_column = self.class.locking_column
|
151
|
+
super.merge(locking_column => _lock_value_for_database(locking_column))
|
152
|
+
end
|
153
|
+
|
145
154
|
module ClassMethods
|
146
155
|
DEFAULT_LOCKING_COLUMN = "lock_version"
|
147
156
|
|
@@ -159,10 +168,7 @@ module ActiveRecord
|
|
159
168
|
end
|
160
169
|
|
161
170
|
# The version column used for optimistic locking. Defaults to +lock_version+.
|
162
|
-
|
163
|
-
@locking_column = DEFAULT_LOCKING_COLUMN unless defined?(@locking_column)
|
164
|
-
@locking_column
|
165
|
-
end
|
171
|
+
attr_reader :locking_column
|
166
172
|
|
167
173
|
# Reset the column used for optimistic locking back to the +lock_version+ default.
|
168
174
|
def reset_locking_column
|
@@ -182,6 +188,14 @@ module ActiveRecord
|
|
182
188
|
end
|
183
189
|
super
|
184
190
|
end
|
191
|
+
|
192
|
+
private
|
193
|
+
def inherited(base)
|
194
|
+
super
|
195
|
+
base.class_eval do
|
196
|
+
@locking_column = DEFAULT_LOCKING_COLUMN
|
197
|
+
end
|
198
|
+
end
|
185
199
|
end
|
186
200
|
end
|
187
201
|
|
@@ -2,10 +2,12 @@
|
|
2
2
|
|
3
3
|
module ActiveRecord
|
4
4
|
module Locking
|
5
|
+
# = \Pessimistic \Locking
|
6
|
+
#
|
5
7
|
# Locking::Pessimistic provides support for row-level locking using
|
6
8
|
# SELECT ... FOR UPDATE and other lock types.
|
7
9
|
#
|
8
|
-
# Chain <tt>ActiveRecord::Base#find</tt> to
|
10
|
+
# Chain <tt>ActiveRecord::Base#find</tt> to ActiveRecord::QueryMethods#lock to obtain an exclusive
|
9
11
|
# lock on the selected rows:
|
10
12
|
# # select * from accounts where id=1 for update
|
11
13
|
# Account.lock.find(1)
|
@@ -71,6 +73,7 @@ module ActiveRecord
|
|
71
73
|
Locking a record with unpersisted changes is not supported. Use
|
72
74
|
`save` to persist the changes, or `reload` to discard them
|
73
75
|
explicitly.
|
76
|
+
Changed attributes: #{changed.map(&:inspect).join(', ')}.
|
74
77
|
MSG
|
75
78
|
end
|
76
79
|
|
@@ -79,13 +82,13 @@ module ActiveRecord
|
|
79
82
|
self
|
80
83
|
end
|
81
84
|
|
82
|
-
# Wraps the passed block in a transaction,
|
83
|
-
# before yielding. You can pass the SQL locking clause
|
84
|
-
# as an optional argument (see
|
85
|
+
# Wraps the passed block in a transaction, reloading the object with a
|
86
|
+
# lock before yielding. You can pass the SQL locking clause
|
87
|
+
# as an optional argument (see #lock!).
|
85
88
|
#
|
86
89
|
# You can also pass options like <tt>requires_new:</tt>, <tt>isolation:</tt>,
|
87
90
|
# and <tt>joinable:</tt> to the wrapping transaction (see
|
88
|
-
#
|
91
|
+
# ActiveRecord::ConnectionAdapters::DatabaseStatements#transaction).
|
89
92
|
def with_lock(*args)
|
90
93
|
transaction_opts = args.extract_options!
|
91
94
|
lock = args.present? ? args.first : true
|
@@ -7,32 +7,36 @@ module ActiveRecord
|
|
7
7
|
class_attribute :backtrace_cleaner, default: ActiveSupport::BacktraceCleaner.new
|
8
8
|
|
9
9
|
def self.runtime=(value)
|
10
|
+
ActiveRecord.deprecator.warn(<<-MSG.squish)
|
11
|
+
ActiveRecord::LogSubscriber.runtime= is deprecated and will be removed in Rails 7.2.
|
12
|
+
MSG
|
10
13
|
ActiveRecord::RuntimeRegistry.sql_runtime = value
|
11
14
|
end
|
12
15
|
|
13
16
|
def self.runtime
|
14
|
-
ActiveRecord
|
17
|
+
ActiveRecord.deprecator.warn(<<-MSG.squish)
|
18
|
+
ActiveRecord::LogSubscriber.runtime is deprecated and will be removed in Rails 7.2.
|
19
|
+
MSG
|
20
|
+
ActiveRecord::RuntimeRegistry.sql_runtime
|
15
21
|
end
|
16
22
|
|
17
23
|
def self.reset_runtime
|
18
|
-
|
19
|
-
|
24
|
+
ActiveRecord.deprecator.warn(<<-MSG.squish)
|
25
|
+
ActiveRecord::LogSubscriber.reset_runtime is deprecated and will be removed in Rails 7.2.
|
26
|
+
MSG
|
27
|
+
ActiveRecord::RuntimeRegistry.reset
|
20
28
|
end
|
21
29
|
|
22
30
|
def strict_loading_violation(event)
|
23
31
|
debug do
|
24
32
|
owner = event.payload[:owner]
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
color("Strict loading violation: #{owner} is marked for strict loading. The #{association} association named :#{name} cannot be lazily loaded.", RED)
|
33
|
+
reflection = event.payload[:reflection]
|
34
|
+
color(reflection.strict_loading_violation_message(owner), RED)
|
29
35
|
end
|
30
36
|
end
|
37
|
+
subscribe_log_level :strict_loading_violation, :debug
|
31
38
|
|
32
39
|
def sql(event)
|
33
|
-
self.class.runtime += event.duration
|
34
|
-
return unless logger.debug?
|
35
|
-
|
36
40
|
payload = event.payload
|
37
41
|
|
38
42
|
return if IGNORE_PAYLOAD_NAMES.include?(payload[:name])
|
@@ -51,7 +55,14 @@ module ActiveRecord
|
|
51
55
|
|
52
56
|
binds = []
|
53
57
|
payload[:binds].each_with_index do |attr, i|
|
54
|
-
attribute_name = attr.respond_to?(:name)
|
58
|
+
attribute_name = if attr.respond_to?(:name)
|
59
|
+
attr.name
|
60
|
+
elsif attr.respond_to?(:[]) && attr[i].respond_to?(:name)
|
61
|
+
attr[i].name
|
62
|
+
else
|
63
|
+
nil
|
64
|
+
end
|
65
|
+
|
55
66
|
filtered_params = filter(attribute_name, casted_params[i])
|
56
67
|
|
57
68
|
binds << render_bind(attr, filtered_params)
|
@@ -61,10 +72,11 @@ module ActiveRecord
|
|
61
72
|
end
|
62
73
|
|
63
74
|
name = colorize_payload_name(name, payload[:name])
|
64
|
-
sql = color(sql, sql_color(sql), true) if colorize_logging
|
75
|
+
sql = color(sql, sql_color(sql), bold: true) if colorize_logging
|
65
76
|
|
66
77
|
debug " #{name} #{sql}#{binds}"
|
67
78
|
end
|
79
|
+
subscribe_log_level :sql, :debug
|
68
80
|
|
69
81
|
private
|
70
82
|
def type_casted_binds(casted_binds)
|
@@ -88,9 +100,9 @@ module ActiveRecord
|
|
88
100
|
|
89
101
|
def colorize_payload_name(name, payload_name)
|
90
102
|
if payload_name.blank? || payload_name == "SQL" # SQL vs Model Load/Exists
|
91
|
-
color(name, MAGENTA, true)
|
103
|
+
color(name, MAGENTA, bold: true)
|
92
104
|
else
|
93
|
-
color(name, CYAN, true)
|
105
|
+
color(name, CYAN, bold: true)
|
94
106
|
end
|
95
107
|
end
|
96
108
|
|
@@ -128,15 +140,25 @@ module ActiveRecord
|
|
128
140
|
end
|
129
141
|
|
130
142
|
def log_query_source
|
131
|
-
source =
|
143
|
+
source = query_source_location
|
132
144
|
|
133
145
|
if source
|
134
146
|
logger.debug(" ↳ #{source}")
|
135
147
|
end
|
136
148
|
end
|
137
149
|
|
138
|
-
|
139
|
-
|
150
|
+
if Thread.respond_to?(:each_caller_location)
|
151
|
+
def query_source_location
|
152
|
+
Thread.each_caller_location do |location|
|
153
|
+
frame = backtrace_cleaner.clean_frame(location)
|
154
|
+
return frame if frame
|
155
|
+
end
|
156
|
+
nil
|
157
|
+
end
|
158
|
+
else
|
159
|
+
def query_source_location
|
160
|
+
backtrace_cleaner.clean(caller(1).lazy).first
|
161
|
+
end
|
140
162
|
end
|
141
163
|
|
142
164
|
def filter(name, value)
|
@@ -0,0 +1,56 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveRecord
|
4
|
+
module Marshalling
|
5
|
+
@format_version = 6.1
|
6
|
+
|
7
|
+
class << self
|
8
|
+
attr_reader :format_version
|
9
|
+
|
10
|
+
def format_version=(version)
|
11
|
+
case version
|
12
|
+
when 6.1
|
13
|
+
Methods.remove_method(:marshal_dump) if Methods.method_defined?(:marshal_dump)
|
14
|
+
when 7.1
|
15
|
+
Methods.alias_method(:marshal_dump, :_marshal_dump_7_1)
|
16
|
+
else
|
17
|
+
raise ArgumentError, "Unknown marshalling format: #{version.inspect}"
|
18
|
+
end
|
19
|
+
@format_version = version
|
20
|
+
end
|
21
|
+
end
|
22
|
+
|
23
|
+
module Methods
|
24
|
+
def _marshal_dump_7_1
|
25
|
+
payload = [attributes_for_database, new_record?]
|
26
|
+
|
27
|
+
cached_associations = self.class.reflect_on_all_associations.select do |reflection|
|
28
|
+
association_cached?(reflection.name)
|
29
|
+
end
|
30
|
+
|
31
|
+
unless cached_associations.empty?
|
32
|
+
payload << cached_associations.map do |reflection|
|
33
|
+
[reflection.name, association(reflection.name).target]
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
payload
|
38
|
+
end
|
39
|
+
|
40
|
+
def marshal_load(state)
|
41
|
+
attributes_from_database, new_record, associations = state
|
42
|
+
|
43
|
+
attributes = self.class.attributes_builder.build_from_database(attributes_from_database)
|
44
|
+
init_with_attributes(attributes, new_record)
|
45
|
+
|
46
|
+
if associations
|
47
|
+
associations.each do |name, target|
|
48
|
+
association(name).target = target
|
49
|
+
rescue ActiveRecord::AssociationNotFoundError
|
50
|
+
# the association no longer exist, we can just skip it.
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
56
|
+
end
|