activerecord 8.0.2 → 8.1.0.beta1
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 +459 -413
- data/README.rdoc +2 -2
- data/lib/active_record/association_relation.rb +1 -1
- data/lib/active_record/associations/association.rb +1 -1
- data/lib/active_record/associations/belongs_to_association.rb +9 -1
- data/lib/active_record/associations/builder/association.rb +16 -5
- data/lib/active_record/associations/builder/belongs_to.rb +17 -4
- data/lib/active_record/associations/builder/collection_association.rb +7 -3
- data/lib/active_record/associations/builder/has_one.rb +1 -1
- data/lib/active_record/associations/builder/singular_association.rb +33 -5
- data/lib/active_record/associations/collection_association.rb +3 -3
- data/lib/active_record/associations/collection_proxy.rb +22 -4
- data/lib/active_record/associations/deprecation.rb +88 -0
- data/lib/active_record/associations/errors.rb +3 -0
- data/lib/active_record/associations/join_dependency.rb +2 -0
- data/lib/active_record/associations/preloader/branch.rb +1 -0
- data/lib/active_record/associations.rb +159 -21
- data/lib/active_record/attribute_methods/query.rb +34 -0
- data/lib/active_record/attribute_methods/serialization.rb +17 -4
- data/lib/active_record/attributes.rb +38 -24
- data/lib/active_record/base.rb +0 -1
- data/lib/active_record/coders/json.rb +14 -5
- data/lib/active_record/connection_adapters/abstract/connection_handler.rb +2 -4
- data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +15 -0
- data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +51 -12
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +384 -49
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +26 -30
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +19 -1
- data/lib/active_record/connection_adapters/abstract/quoting.rb +15 -24
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +7 -2
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +26 -34
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +2 -1
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +89 -23
- data/lib/active_record/connection_adapters/abstract/transaction.rb +16 -3
- data/lib/active_record/connection_adapters/abstract_adapter.rb +67 -13
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +43 -11
- data/lib/active_record/connection_adapters/column.rb +17 -4
- data/lib/active_record/connection_adapters/mysql/database_statements.rb +4 -4
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +2 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +42 -5
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +26 -4
- data/lib/active_record/connection_adapters/mysql2/database_statements.rb +27 -22
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +1 -0
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +18 -16
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +2 -2
- data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +1 -1
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +21 -10
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +1 -1
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +8 -21
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +65 -30
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +74 -38
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +12 -7
- data/lib/active_record/connection_adapters/schema_cache.rb +2 -2
- data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +39 -27
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +0 -8
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +4 -13
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +56 -32
- data/lib/active_record/connection_adapters/trilogy/database_statements.rb +4 -3
- data/lib/active_record/connection_adapters/trilogy_adapter.rb +1 -1
- data/lib/active_record/connection_adapters.rb +1 -0
- data/lib/active_record/connection_handling.rb +1 -1
- data/lib/active_record/core.rb +13 -10
- data/lib/active_record/counter_cache.rb +33 -8
- data/lib/active_record/database_configurations/database_config.rb +5 -1
- data/lib/active_record/database_configurations/hash_config.rb +56 -9
- data/lib/active_record/database_configurations/url_config.rb +13 -3
- data/lib/active_record/database_configurations.rb +7 -3
- data/lib/active_record/delegated_type.rb +2 -2
- data/lib/active_record/dynamic_matchers.rb +54 -69
- data/lib/active_record/encryption/encryptable_record.rb +5 -5
- data/lib/active_record/encryption/encrypted_attribute_type.rb +2 -2
- data/lib/active_record/encryption/encryptor.rb +27 -25
- data/lib/active_record/encryption/scheme.rb +1 -1
- data/lib/active_record/enum.rb +37 -20
- data/lib/active_record/errors.rb +20 -4
- data/lib/active_record/explain_registry.rb +0 -1
- data/lib/active_record/filter_attribute_handler.rb +73 -0
- data/lib/active_record/fixture_set/table_row.rb +19 -2
- data/lib/active_record/fixtures.rb +2 -2
- data/lib/active_record/gem_version.rb +3 -3
- data/lib/active_record/inheritance.rb +1 -1
- data/lib/active_record/insert_all.rb +12 -7
- data/lib/active_record/locking/optimistic.rb +7 -0
- data/lib/active_record/locking/pessimistic.rb +5 -0
- data/lib/active_record/log_subscriber.rb +1 -5
- data/lib/active_record/middleware/shard_selector.rb +34 -17
- data/lib/active_record/migration/command_recorder.rb +14 -1
- data/lib/active_record/migration/compatibility.rb +34 -24
- data/lib/active_record/migration/default_schema_versions_formatter.rb +30 -0
- data/lib/active_record/migration.rb +31 -21
- data/lib/active_record/model_schema.rb +10 -7
- data/lib/active_record/nested_attributes.rb +2 -0
- data/lib/active_record/persistence.rb +34 -3
- data/lib/active_record/query_cache.rb +22 -15
- data/lib/active_record/query_logs.rb +7 -7
- data/lib/active_record/querying.rb +4 -4
- data/lib/active_record/railtie.rb +34 -5
- data/lib/active_record/railties/databases.rake +23 -19
- data/lib/active_record/railties/job_checkpoints.rb +15 -0
- data/lib/active_record/railties/job_runtime.rb +10 -11
- data/lib/active_record/reflection.rb +42 -3
- data/lib/active_record/relation/batches.rb +26 -12
- data/lib/active_record/relation/calculations.rb +35 -25
- data/lib/active_record/relation/delegation.rb +0 -1
- data/lib/active_record/relation/finder_methods.rb +41 -24
- data/lib/active_record/relation/merger.rb +2 -2
- data/lib/active_record/relation/predicate_builder.rb +2 -2
- data/lib/active_record/relation/query_attribute.rb +3 -1
- data/lib/active_record/relation/query_methods.rb +43 -33
- data/lib/active_record/relation/spawn_methods.rb +6 -6
- data/lib/active_record/relation/where_clause.rb +7 -10
- data/lib/active_record/relation.rb +37 -15
- data/lib/active_record/result.rb +44 -21
- data/lib/active_record/sanitization.rb +2 -0
- data/lib/active_record/schema_dumper.rb +12 -10
- data/lib/active_record/scoping.rb +0 -1
- data/lib/active_record/secure_token.rb +3 -3
- data/lib/active_record/signed_id.rb +46 -18
- data/lib/active_record/statement_cache.rb +13 -9
- data/lib/active_record/store.rb +44 -19
- data/lib/active_record/tasks/abstract_tasks.rb +76 -0
- data/lib/active_record/tasks/database_tasks.rb +24 -35
- data/lib/active_record/tasks/mysql_database_tasks.rb +3 -40
- data/lib/active_record/tasks/postgresql_database_tasks.rb +14 -40
- data/lib/active_record/tasks/sqlite_database_tasks.rb +14 -26
- data/lib/active_record/test_databases.rb +11 -3
- data/lib/active_record/test_fixtures.rb +27 -2
- data/lib/active_record/testing/query_assertions.rb +8 -2
- data/lib/active_record/timestamp.rb +4 -2
- data/lib/active_record/transaction.rb +2 -5
- data/lib/active_record/transactions.rb +34 -10
- data/lib/active_record/type/hash_lookup_type_map.rb +2 -1
- data/lib/active_record/type/internal/timezone.rb +7 -0
- data/lib/active_record/type/json.rb +15 -2
- data/lib/active_record/type/serialized.rb +11 -4
- data/lib/active_record/type/type_map.rb +1 -1
- data/lib/active_record/type_caster/connection.rb +2 -1
- data/lib/active_record/validations/associated.rb +1 -1
- data/lib/active_record.rb +68 -5
- data/lib/arel/alias_predication.rb +2 -0
- data/lib/arel/crud.rb +8 -11
- data/lib/arel/delete_manager.rb +5 -0
- data/lib/arel/nodes/count.rb +2 -2
- data/lib/arel/nodes/delete_statement.rb +4 -2
- data/lib/arel/nodes/function.rb +4 -10
- data/lib/arel/nodes/named_function.rb +2 -2
- data/lib/arel/nodes/node.rb +1 -1
- data/lib/arel/nodes/update_statement.rb +4 -2
- data/lib/arel/nodes.rb +0 -2
- data/lib/arel/select_manager.rb +13 -4
- data/lib/arel/update_manager.rb +5 -0
- data/lib/arel/visitors/dot.rb +2 -3
- data/lib/arel/visitors/postgresql.rb +55 -0
- data/lib/arel/visitors/sqlite.rb +55 -8
- data/lib/arel/visitors/to_sql.rb +5 -21
- data/lib/arel.rb +3 -1
- metadata +15 -11
- data/lib/active_record/normalization.rb +0 -163
@@ -1,5 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require "active_support/json"
|
4
|
+
|
3
5
|
module ActiveRecord
|
4
6
|
module Type
|
5
7
|
class Json < ActiveModel::Type::Value
|
@@ -11,11 +13,22 @@ module ActiveRecord
|
|
11
13
|
|
12
14
|
def deserialize(value)
|
13
15
|
return value unless value.is_a?(::String)
|
14
|
-
|
16
|
+
begin
|
17
|
+
ActiveSupport::JSON.decode(value)
|
18
|
+
rescue JSON::ParserError => e
|
19
|
+
# NOTE: This may hide json with duplicate keys. We don't really want to just ignore it
|
20
|
+
# but it's the best we can do in order to still allow updating columns that somehow already
|
21
|
+
# contain invalid json from some other source.
|
22
|
+
# See https://github.com/rails/rails/pull/55536
|
23
|
+
ActiveSupport.error_reporter.report(e, source: "application.active_record")
|
24
|
+
nil
|
25
|
+
end
|
15
26
|
end
|
16
27
|
|
28
|
+
JSON_ENCODER = ActiveSupport::JSON::Encoding.json_encoder.new(escape: false)
|
29
|
+
|
17
30
|
def serialize(value)
|
18
|
-
|
31
|
+
JSON_ENCODER.encode(value) unless value.nil?
|
19
32
|
end
|
20
33
|
|
21
34
|
def changed_in_place?(raw_old_value, new_value)
|
@@ -9,9 +9,10 @@ module ActiveRecord
|
|
9
9
|
|
10
10
|
attr_reader :subtype, :coder
|
11
11
|
|
12
|
-
def initialize(subtype, coder)
|
12
|
+
def initialize(subtype, coder, comparable: false)
|
13
13
|
@subtype = subtype
|
14
14
|
@coder = coder
|
15
|
+
@comparable = comparable
|
15
16
|
super(subtype)
|
16
17
|
end
|
17
18
|
|
@@ -34,9 +35,15 @@ module ActiveRecord
|
|
34
35
|
|
35
36
|
def changed_in_place?(raw_old_value, value)
|
36
37
|
return false if value.nil?
|
37
|
-
|
38
|
-
|
39
|
-
|
38
|
+
|
39
|
+
if @comparable
|
40
|
+
old_value = deserialize(raw_old_value)
|
41
|
+
old_value != value
|
42
|
+
else
|
43
|
+
raw_new_value = encoded(value)
|
44
|
+
raw_old_value.nil? != raw_new_value.nil? ||
|
45
|
+
subtype.changed_in_place?(raw_old_value, raw_new_value)
|
46
|
+
end
|
40
47
|
end
|
41
48
|
|
42
49
|
def accessor
|
@@ -19,7 +19,8 @@ module ActiveRecord
|
|
19
19
|
if schema_cache.data_source_exists?(table_name)
|
20
20
|
column = schema_cache.columns_hash(table_name)[attr_name.to_s]
|
21
21
|
if column
|
22
|
-
|
22
|
+
# TODO: Remove fetch_cast_type and the need for connection after we release 8.1.
|
23
|
+
type = column.fetch_cast_type(@klass.lease_connection)
|
23
24
|
end
|
24
25
|
end
|
25
26
|
|
@@ -7,7 +7,7 @@ module ActiveRecord
|
|
7
7
|
context = record_validation_context_for_association(record)
|
8
8
|
|
9
9
|
if Array(value).reject { |association| valid_object?(association, context) }.any?
|
10
|
-
record.errors.add(attribute, :invalid, **options
|
10
|
+
record.errors.add(attribute, :invalid, **options, value: value)
|
11
11
|
end
|
12
12
|
end
|
13
13
|
|
data/lib/active_record.rb
CHANGED
@@ -26,6 +26,7 @@
|
|
26
26
|
require "active_support"
|
27
27
|
require "active_support/rails"
|
28
28
|
require "active_support/ordered_options"
|
29
|
+
require "active_support/core_ext/array/conversions"
|
29
30
|
require "active_model"
|
30
31
|
require "arel"
|
31
32
|
require "yaml"
|
@@ -52,6 +53,7 @@ module ActiveRecord
|
|
52
53
|
autoload :Enum
|
53
54
|
autoload :Explain
|
54
55
|
autoload :FixtureSet, "active_record/fixtures"
|
56
|
+
autoload :FilterAttributeHandler
|
55
57
|
autoload :Inheritance
|
56
58
|
autoload :Integration
|
57
59
|
autoload :InternalMetadata
|
@@ -62,7 +64,6 @@ module ActiveRecord
|
|
62
64
|
autoload :ModelSchema
|
63
65
|
autoload :NestedAttributes
|
64
66
|
autoload :NoTouching
|
65
|
-
autoload :Normalization
|
66
67
|
autoload :Persistence
|
67
68
|
autoload :QueryCache
|
68
69
|
autoload :QueryLogs
|
@@ -87,7 +88,6 @@ module ActiveRecord
|
|
87
88
|
autoload :Timestamp
|
88
89
|
autoload :TokenFor
|
89
90
|
autoload :TouchLater
|
90
|
-
autoload :Transaction
|
91
91
|
autoload :Transactions
|
92
92
|
autoload :Translation
|
93
93
|
autoload :Validations
|
@@ -109,6 +109,7 @@ module ActiveRecord
|
|
109
109
|
autoload :Result
|
110
110
|
autoload :StatementCache
|
111
111
|
autoload :TableMetadata
|
112
|
+
autoload :Transaction
|
112
113
|
autoload :Type
|
113
114
|
|
114
115
|
autoload_under "relation" do
|
@@ -174,7 +175,8 @@ module ActiveRecord
|
|
174
175
|
extend ActiveSupport::Autoload
|
175
176
|
|
176
177
|
autoload :DatabaseTasks
|
177
|
-
autoload :
|
178
|
+
autoload :AbstractTasks, "active_record/tasks/abstract_tasks"
|
179
|
+
autoload :MySQLDatabaseTasks, "active_record/tasks/mysql_database_tasks"
|
178
180
|
autoload :PostgreSQLDatabaseTasks, "active_record/tasks/postgresql_database_tasks"
|
179
181
|
autoload :SQLiteDatabaseTasks, "active_record/tasks/sqlite_database_tasks"
|
180
182
|
end
|
@@ -259,6 +261,9 @@ module ActiveRecord
|
|
259
261
|
##
|
260
262
|
# :singleton-method: db_warnings_ignore
|
261
263
|
# Specify allowlist of database warnings.
|
264
|
+
# Can be a string, regular expression, or an error code from the database.
|
265
|
+
#
|
266
|
+
# ActiveRecord::Base.db_warnings_ignore = [/`SHOW WARNINGS` did not return the warnings/, "01000"]
|
262
267
|
singleton_class.attr_accessor :db_warnings_ignore
|
263
268
|
self.db_warnings_ignore = []
|
264
269
|
|
@@ -286,6 +291,7 @@ module ActiveRecord
|
|
286
291
|
def self.global_thread_pool_async_query_executor # :nodoc:
|
287
292
|
concurrency = global_executor_concurrency || 4
|
288
293
|
@global_thread_pool_async_query_executor ||= Concurrent::ThreadPoolExecutor.new(
|
294
|
+
name: "ActiveRecord-global-async-query-executor",
|
289
295
|
min_threads: 0,
|
290
296
|
max_threads: concurrency,
|
291
297
|
max_queue: concurrency * 4,
|
@@ -351,6 +357,9 @@ module ActiveRecord
|
|
351
357
|
singleton_class.attr_accessor :run_after_transaction_callbacks_in_order_defined
|
352
358
|
self.run_after_transaction_callbacks_in_order_defined = false
|
353
359
|
|
360
|
+
singleton_class.attr_accessor :raise_on_missing_required_finder_order_columns
|
361
|
+
self.run_after_transaction_callbacks_in_order_defined = false
|
362
|
+
|
354
363
|
singleton_class.attr_accessor :application_record_class
|
355
364
|
self.application_record_class = nil
|
356
365
|
|
@@ -368,7 +377,8 @@ module ActiveRecord
|
|
368
377
|
# specific) SQL statements. If :ruby, the schema is dumped as an
|
369
378
|
# ActiveRecord::Schema file which can be loaded into any database that
|
370
379
|
# supports migrations. Use :ruby if you want to have different database
|
371
|
-
# adapters for, e.g., your development and test environments.
|
380
|
+
# adapters for, e.g., your development and test environments. This can be
|
381
|
+
# overridden per-database in the database configuration.
|
372
382
|
singleton_class.attr_accessor :schema_format
|
373
383
|
self.schema_format = :ruby
|
374
384
|
|
@@ -400,6 +410,12 @@ module ActiveRecord
|
|
400
410
|
singleton_class.attr_accessor :migration_strategy
|
401
411
|
self.migration_strategy = Migration::DefaultStrategy
|
402
412
|
|
413
|
+
##
|
414
|
+
# :singleton-method: schema_versions_formatter
|
415
|
+
# Specify the formatter used by schema dumper to format versions information.
|
416
|
+
singleton_class.attr_accessor :schema_versions_formatter
|
417
|
+
self.schema_versions_formatter = Migration::DefaultSchemaVersionsFormatter
|
418
|
+
|
403
419
|
##
|
404
420
|
# :singleton-method: dump_schema_after_migration
|
405
421
|
# Specify whether schema dump should happen at the end of the
|
@@ -460,6 +476,29 @@ module ActiveRecord
|
|
460
476
|
singleton_class.attr_accessor :generate_secure_token_on
|
461
477
|
self.generate_secure_token_on = :create
|
462
478
|
|
479
|
+
def self.deprecated_associations_options=(options)
|
480
|
+
raise ArgumentError, "deprecated_associations_options must be a hash" unless options.is_a?(Hash)
|
481
|
+
|
482
|
+
valid_keys = [:mode, :backtrace]
|
483
|
+
|
484
|
+
invalid_keys = options.keys - valid_keys
|
485
|
+
unless invalid_keys.empty?
|
486
|
+
inflected_key = invalid_keys.size == 1 ? "key" : "keys"
|
487
|
+
raise ArgumentError, "invalid deprecated_associations_options #{inflected_key} #{invalid_keys.map(&:inspect).to_sentence} (valid keys are #{valid_keys.map(&:inspect).to_sentence})"
|
488
|
+
end
|
489
|
+
|
490
|
+
options.each do |key, value|
|
491
|
+
ActiveRecord::Associations::Deprecation.send("#{key}=", value)
|
492
|
+
end
|
493
|
+
end
|
494
|
+
|
495
|
+
def self.deprecated_associations_options
|
496
|
+
{
|
497
|
+
mode: ActiveRecord::Associations::Deprecation.mode,
|
498
|
+
backtrace: ActiveRecord::Associations::Deprecation.backtrace
|
499
|
+
}
|
500
|
+
end
|
501
|
+
|
463
502
|
def self.marshalling_format_version
|
464
503
|
Marshalling.format_version
|
465
504
|
end
|
@@ -496,6 +535,13 @@ module ActiveRecord
|
|
496
535
|
}
|
497
536
|
)
|
498
537
|
|
538
|
+
##
|
539
|
+
# :singleton-method: message_verifiers
|
540
|
+
#
|
541
|
+
# ActiveSupport::MessageVerifiers instance for Active Record. If you are using
|
542
|
+
# Rails, this will be set to +Rails.application.message_verifiers+.
|
543
|
+
singleton_class.attr_accessor :message_verifiers
|
544
|
+
|
499
545
|
def self.eager_load!
|
500
546
|
super
|
501
547
|
ActiveRecord::Locking.eager_load!
|
@@ -550,13 +596,30 @@ module ActiveRecord
|
|
550
596
|
if active_connection = pool.active_connection
|
551
597
|
current_transaction = active_connection.current_transaction
|
552
598
|
|
553
|
-
if current_transaction.open? && current_transaction.joinable?
|
599
|
+
if current_transaction.open? && current_transaction.joinable?
|
554
600
|
open_transactions << current_transaction
|
555
601
|
end
|
556
602
|
end
|
557
603
|
end
|
558
604
|
open_transactions
|
559
605
|
end
|
606
|
+
|
607
|
+
def self.default_transaction_isolation_level=(isolation_level) # :nodoc:
|
608
|
+
ActiveSupport::IsolatedExecutionState[:active_record_transaction_isolation] = isolation_level
|
609
|
+
end
|
610
|
+
|
611
|
+
def self.default_transaction_isolation_level # :nodoc:
|
612
|
+
ActiveSupport::IsolatedExecutionState[:active_record_transaction_isolation]
|
613
|
+
end
|
614
|
+
|
615
|
+
# Sets a transaction isolation level for all connection pools within the block.
|
616
|
+
def self.with_transaction_isolation_level(isolation_level, &block)
|
617
|
+
original_level = self.default_transaction_isolation_level
|
618
|
+
self.default_transaction_isolation_level = isolation_level
|
619
|
+
yield
|
620
|
+
ensure
|
621
|
+
self.default_transaction_isolation_level = original_level
|
622
|
+
end
|
560
623
|
end
|
561
624
|
|
562
625
|
ActiveSupport.on_load(:active_record) do
|
data/lib/arel/crud.rb
CHANGED
@@ -14,34 +14,31 @@ module Arel # :nodoc: all
|
|
14
14
|
InsertManager.new
|
15
15
|
end
|
16
16
|
|
17
|
-
def compile_update(
|
18
|
-
values,
|
19
|
-
key = nil,
|
20
|
-
having_clause = nil,
|
21
|
-
group_values_columns = []
|
22
|
-
)
|
17
|
+
def compile_update(values, key = nil)
|
23
18
|
um = UpdateManager.new(source)
|
24
19
|
um.set(values)
|
25
20
|
um.take(limit)
|
26
21
|
um.offset(offset)
|
27
22
|
um.order(*orders)
|
28
23
|
um.wheres = constraints
|
24
|
+
um.comment(comment)
|
29
25
|
um.key = key
|
30
26
|
|
31
|
-
um.
|
32
|
-
um.having(
|
27
|
+
um.ast.groups = @ctx.groups
|
28
|
+
@ctx.havings.each { |h| um.having(h) }
|
33
29
|
um
|
34
30
|
end
|
35
31
|
|
36
|
-
def compile_delete(key = nil
|
32
|
+
def compile_delete(key = nil)
|
37
33
|
dm = DeleteManager.new(source)
|
38
34
|
dm.take(limit)
|
39
35
|
dm.offset(offset)
|
40
36
|
dm.order(*orders)
|
41
37
|
dm.wheres = constraints
|
38
|
+
dm.comment(comment)
|
42
39
|
dm.key = key
|
43
|
-
dm.
|
44
|
-
dm.having(
|
40
|
+
dm.ast.groups = @ctx.groups
|
41
|
+
@ctx.havings.each { |h| dm.having(h) }
|
45
42
|
dm
|
46
43
|
end
|
47
44
|
end
|
data/lib/arel/delete_manager.rb
CHANGED
data/lib/arel/nodes/count.rb
CHANGED
@@ -3,7 +3,7 @@
|
|
3
3
|
module Arel # :nodoc: all
|
4
4
|
module Nodes
|
5
5
|
class DeleteStatement < Arel::Nodes::Node
|
6
|
-
attr_accessor :relation, :wheres, :groups, :havings, :orders, :limit, :offset, :key
|
6
|
+
attr_accessor :relation, :wheres, :groups, :havings, :orders, :limit, :offset, :comment, :key
|
7
7
|
|
8
8
|
def initialize(relation = nil, wheres = [])
|
9
9
|
super()
|
@@ -14,6 +14,7 @@ module Arel # :nodoc: all
|
|
14
14
|
@orders = []
|
15
15
|
@limit = nil
|
16
16
|
@offset = nil
|
17
|
+
@comment = nil
|
17
18
|
@key = nil
|
18
19
|
end
|
19
20
|
|
@@ -24,7 +25,7 @@ module Arel # :nodoc: all
|
|
24
25
|
end
|
25
26
|
|
26
27
|
def hash
|
27
|
-
[self.class, @relation, @wheres, @orders, @limit, @offset, @key].hash
|
28
|
+
[self.class, @relation, @wheres, @orders, @limit, @offset, @comment, @key].hash
|
28
29
|
end
|
29
30
|
|
30
31
|
def eql?(other)
|
@@ -36,6 +37,7 @@ module Arel # :nodoc: all
|
|
36
37
|
self.havings == other.havings &&
|
37
38
|
self.limit == other.limit &&
|
38
39
|
self.offset == other.offset &&
|
40
|
+
self.comment == other.comment &&
|
39
41
|
self.key == other.key
|
40
42
|
end
|
41
43
|
alias :== :eql?
|
data/lib/arel/nodes/function.rb
CHANGED
@@ -5,28 +5,22 @@ module Arel # :nodoc: all
|
|
5
5
|
class Function < Arel::Nodes::NodeExpression
|
6
6
|
include Arel::WindowPredications
|
7
7
|
include Arel::FilterPredications
|
8
|
-
attr_accessor :expressions, :alias, :distinct
|
9
8
|
|
10
|
-
|
9
|
+
attr_accessor :expressions, :distinct
|
10
|
+
|
11
|
+
def initialize(expr)
|
11
12
|
super()
|
12
13
|
@expressions = expr
|
13
|
-
@alias = aliaz && SqlLiteral.new(aliaz)
|
14
14
|
@distinct = false
|
15
15
|
end
|
16
16
|
|
17
|
-
def as(aliaz)
|
18
|
-
self.alias = SqlLiteral.new(aliaz)
|
19
|
-
self
|
20
|
-
end
|
21
|
-
|
22
17
|
def hash
|
23
|
-
[@expressions, @
|
18
|
+
[@expressions, @distinct].hash
|
24
19
|
end
|
25
20
|
|
26
21
|
def eql?(other)
|
27
22
|
self.class == other.class &&
|
28
23
|
self.expressions == other.expressions &&
|
29
|
-
self.alias == other.alias &&
|
30
24
|
self.distinct == other.distinct
|
31
25
|
end
|
32
26
|
alias :== :eql?
|
data/lib/arel/nodes/node.rb
CHANGED
@@ -6,7 +6,7 @@ module Arel # :nodoc: all
|
|
6
6
|
#
|
7
7
|
# Active Record uses Arel to compose SQL statements. Instead of building SQL strings directly, it's building an
|
8
8
|
# abstract syntax tree (AST) of the statement using various types of Arel::Nodes::Node. Each node represents a
|
9
|
-
# fragment of
|
9
|
+
# fragment of an SQL statement.
|
10
10
|
#
|
11
11
|
# The intermediate representation allows Arel to compile the statement into the database's specific SQL dialect
|
12
12
|
# only before sending it without having to care about the nuances of each database when building the statement.
|
@@ -3,7 +3,7 @@
|
|
3
3
|
module Arel # :nodoc: all
|
4
4
|
module Nodes
|
5
5
|
class UpdateStatement < Arel::Nodes::Node
|
6
|
-
attr_accessor :relation, :wheres, :values, :groups, :havings, :orders, :limit, :offset, :key
|
6
|
+
attr_accessor :relation, :wheres, :values, :groups, :havings, :orders, :limit, :offset, :comment, :key
|
7
7
|
|
8
8
|
def initialize(relation = nil)
|
9
9
|
super()
|
@@ -15,6 +15,7 @@ module Arel # :nodoc: all
|
|
15
15
|
@orders = []
|
16
16
|
@limit = nil
|
17
17
|
@offset = nil
|
18
|
+
@comment = nil
|
18
19
|
@key = nil
|
19
20
|
end
|
20
21
|
|
@@ -25,7 +26,7 @@ module Arel # :nodoc: all
|
|
25
26
|
end
|
26
27
|
|
27
28
|
def hash
|
28
|
-
[@relation, @wheres, @values, @orders, @limit, @offset, @key].hash
|
29
|
+
[@relation, @wheres, @values, @orders, @limit, @offset, @comment, @key].hash
|
29
30
|
end
|
30
31
|
|
31
32
|
def eql?(other)
|
@@ -38,6 +39,7 @@ module Arel # :nodoc: all
|
|
38
39
|
self.orders == other.orders &&
|
39
40
|
self.limit == other.limit &&
|
40
41
|
self.offset == other.offset &&
|
42
|
+
self.comment == other.comment &&
|
41
43
|
self.key == other.key
|
42
44
|
end
|
43
45
|
alias :== :eql?
|
data/lib/arel/nodes.rb
CHANGED
@@ -45,8 +45,6 @@ require "arel/nodes/cte"
|
|
45
45
|
require "arel/nodes/nary"
|
46
46
|
|
47
47
|
# function
|
48
|
-
# FIXME: Function + Alias can be rewritten as a Function and Alias node.
|
49
|
-
# We should make Function a Unary node and deprecate the use of "aliaz"
|
50
48
|
require "arel/nodes/function"
|
51
49
|
require "arel/nodes/count"
|
52
50
|
require "arel/nodes/extract"
|
data/lib/arel/select_manager.rb
CHANGED
@@ -74,8 +74,13 @@ module Arel # :nodoc: all
|
|
74
74
|
def group(*columns)
|
75
75
|
columns.each do |column|
|
76
76
|
# FIXME: backwards compat
|
77
|
-
|
78
|
-
|
77
|
+
case column
|
78
|
+
when Nodes::SqlLiteral
|
79
|
+
when String
|
80
|
+
column = Nodes::SqlLiteral.new(column)
|
81
|
+
when Symbol
|
82
|
+
column = Nodes::SqlLiteral.new(column.name)
|
83
|
+
end
|
79
84
|
|
80
85
|
@ctx.groups.push Nodes::Group.new column
|
81
86
|
end
|
@@ -250,8 +255,12 @@ module Arel # :nodoc: all
|
|
250
255
|
end
|
251
256
|
|
252
257
|
def comment(*values)
|
253
|
-
|
254
|
-
|
258
|
+
if values.any?
|
259
|
+
@ctx.comment = Nodes::Comment.new(values)
|
260
|
+
self
|
261
|
+
else
|
262
|
+
@ctx.comment
|
263
|
+
end
|
255
264
|
end
|
256
265
|
|
257
266
|
private
|
data/lib/arel/update_manager.rb
CHANGED
data/lib/arel/visitors/dot.rb
CHANGED
@@ -34,7 +34,6 @@ module Arel # :nodoc: all
|
|
34
34
|
def visit_Arel_Nodes_Function(o)
|
35
35
|
visit_edge o, "expressions"
|
36
36
|
visit_edge o, "distinct"
|
37
|
-
visit_edge o, "alias"
|
38
37
|
end
|
39
38
|
|
40
39
|
def visit_Arel_Nodes_Unary(o)
|
@@ -108,14 +107,12 @@ module Arel # :nodoc: all
|
|
108
107
|
|
109
108
|
def visit_Arel_Nodes_Extract(o)
|
110
109
|
visit_edge o, "expressions"
|
111
|
-
visit_edge o, "alias"
|
112
110
|
end
|
113
111
|
|
114
112
|
def visit_Arel_Nodes_NamedFunction(o)
|
115
113
|
visit_edge o, "name"
|
116
114
|
visit_edge o, "expressions"
|
117
115
|
visit_edge o, "distinct"
|
118
|
-
visit_edge o, "alias"
|
119
116
|
end
|
120
117
|
|
121
118
|
def visit_Arel_Nodes_InsertStatement(o)
|
@@ -153,6 +150,7 @@ module Arel # :nodoc: all
|
|
153
150
|
visit_edge o, "orders"
|
154
151
|
visit_edge o, "limit"
|
155
152
|
visit_edge o, "offset"
|
153
|
+
visit_edge o, "comment"
|
156
154
|
visit_edge o, "key"
|
157
155
|
end
|
158
156
|
|
@@ -162,6 +160,7 @@ module Arel # :nodoc: all
|
|
162
160
|
visit_edge o, "orders"
|
163
161
|
visit_edge o, "limit"
|
164
162
|
visit_edge o, "offset"
|
163
|
+
visit_edge o, "comment"
|
165
164
|
visit_edge o, "key"
|
166
165
|
end
|
167
166
|
|
@@ -4,6 +4,55 @@ module Arel # :nodoc: all
|
|
4
4
|
module Visitors
|
5
5
|
class PostgreSQL < Arel::Visitors::ToSql
|
6
6
|
private
|
7
|
+
def visit_Arel_Nodes_UpdateStatement(o, collector)
|
8
|
+
collector.retryable = false
|
9
|
+
o = prepare_update_statement(o)
|
10
|
+
|
11
|
+
collector << "UPDATE "
|
12
|
+
|
13
|
+
# UPDATE with JOIN is in the form of:
|
14
|
+
#
|
15
|
+
# UPDATE t1 AS __active_record_update_alias
|
16
|
+
# SET ..
|
17
|
+
# FROM t1 JOIN t2 ON t2.join_id = t1.join_id ..
|
18
|
+
# WHERE t1.id = __active_record_update_alias.id AND ..
|
19
|
+
if has_join_sources?(o)
|
20
|
+
collector = visit o.relation.left, collector
|
21
|
+
collect_nodes_for o.values, collector, " SET "
|
22
|
+
collector << " FROM "
|
23
|
+
collector = inject_join o.relation.right, collector, " "
|
24
|
+
else
|
25
|
+
collector = visit o.relation, collector
|
26
|
+
collect_nodes_for o.values, collector, " SET "
|
27
|
+
end
|
28
|
+
|
29
|
+
collect_nodes_for o.wheres, collector, " WHERE ", " AND "
|
30
|
+
collect_nodes_for o.orders, collector, " ORDER BY "
|
31
|
+
maybe_visit o.limit, collector
|
32
|
+
maybe_visit o.comment, collector
|
33
|
+
end
|
34
|
+
|
35
|
+
# In the simple case, PostgreSQL allows us to place FROM or JOINs directly into the UPDATE
|
36
|
+
# query. However, this does not allow for LIMIT, OFFSET and ORDER. To support
|
37
|
+
# these, we must use a subquery.
|
38
|
+
def prepare_update_statement(o)
|
39
|
+
if o.key && has_join_sources?(o) && !has_group_by_and_having?(o) && !has_limit_or_offset_or_orders?(o)
|
40
|
+
# Join clauses cannot reference the target table, so alias the
|
41
|
+
# updated table, place the entire relation in the FROM clause, and
|
42
|
+
# add a self-join (which requires the primary key)
|
43
|
+
stmt = o.clone
|
44
|
+
stmt.relation, stmt.wheres = o.relation.clone, o.wheres.clone
|
45
|
+
stmt.relation.right = [stmt.relation.left, *stmt.relation.right]
|
46
|
+
stmt.relation.left = stmt.relation.left.alias("__active_record_update_alias")
|
47
|
+
Array.wrap(o.key).each do |key|
|
48
|
+
stmt.wheres << key.eq(stmt.relation.left[key.name])
|
49
|
+
end
|
50
|
+
stmt
|
51
|
+
else
|
52
|
+
super
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
7
56
|
def visit_Arel_Nodes_Matches(o, collector)
|
8
57
|
op = o.case_sensitive ? " LIKE " : " ILIKE "
|
9
58
|
collector = infix_value o, collector, op
|
@@ -66,6 +115,12 @@ module Arel # :nodoc: all
|
|
66
115
|
grouping_parentheses o.expr, collector
|
67
116
|
end
|
68
117
|
|
118
|
+
def visit_Arel_Nodes_InnerJoin(o, collector)
|
119
|
+
return super if o.right
|
120
|
+
collector << "CROSS JOIN "
|
121
|
+
visit o.left, collector
|
122
|
+
end
|
123
|
+
|
69
124
|
def visit_Arel_Nodes_IsNotDistinctFrom(o, collector)
|
70
125
|
collector = visit o.left, collector
|
71
126
|
collector << " IS NOT DISTINCT FROM "
|