activerecord 7.2.3 → 8.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +612 -1055
- data/README.rdoc +1 -1
- data/lib/active_record/association_relation.rb +2 -1
- data/lib/active_record/associations/association.rb +35 -11
- data/lib/active_record/associations/builder/association.rb +23 -11
- 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 +1 -1
- data/lib/active_record/associations/collection_proxy.rb +22 -4
- data/lib/active_record/associations/deprecation.rb +88 -0
- data/lib/active_record/associations/disable_joins_association_scope.rb +1 -1
- data/lib/active_record/associations/errors.rb +3 -0
- data/lib/active_record/associations/has_many_through_association.rb +3 -2
- data/lib/active_record/associations/join_dependency.rb +4 -2
- data/lib/active_record/associations/preloader/association.rb +2 -2
- data/lib/active_record/associations/preloader/batch.rb +7 -1
- data/lib/active_record/associations/preloader/branch.rb +1 -0
- data/lib/active_record/associations/singular_association.rb +8 -3
- data/lib/active_record/associations.rb +192 -24
- data/lib/active_record/asynchronous_queries_tracker.rb +28 -24
- data/lib/active_record/attribute_methods/primary_key.rb +4 -8
- data/lib/active_record/attribute_methods/query.rb +34 -0
- data/lib/active_record/attribute_methods/serialization.rb +16 -3
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +12 -14
- data/lib/active_record/attributes.rb +3 -0
- data/lib/active_record/autosave_association.rb +69 -27
- data/lib/active_record/base.rb +1 -2
- data/lib/active_record/coders/json.rb +14 -5
- data/lib/active_record/connection_adapters/abstract/connection_handler.rb +35 -28
- data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +16 -4
- data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +51 -13
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +412 -88
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +137 -75
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +27 -5
- data/lib/active_record/connection_adapters/abstract/quoting.rb +16 -25
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +11 -7
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +32 -35
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +2 -1
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +122 -32
- data/lib/active_record/connection_adapters/abstract/transaction.rb +40 -8
- data/lib/active_record/connection_adapters/abstract_adapter.rb +150 -91
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +63 -52
- 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/quoting.rb +0 -8
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +2 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +41 -10
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +73 -46
- data/lib/active_record/connection_adapters/mysql2/database_statements.rb +89 -94
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +2 -10
- data/lib/active_record/connection_adapters/pool_config.rb +7 -7
- data/lib/active_record/connection_adapters/postgresql/column.rb +8 -2
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +76 -45
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +3 -3
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +10 -0
- 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/referential_integrity.rb +2 -4
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +9 -17
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +14 -33
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +71 -32
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +139 -63
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +78 -105
- data/lib/active_record/connection_adapters/schema_cache.rb +3 -5
- data/lib/active_record/connection_adapters/sqlite3/column.rb +8 -2
- data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +90 -98
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +0 -8
- data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +0 -6
- data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +27 -2
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +13 -14
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +102 -37
- data/lib/active_record/connection_adapters/statement_pool.rb +4 -2
- data/lib/active_record/connection_adapters/trilogy/database_statements.rb +38 -67
- data/lib/active_record/connection_adapters/trilogy_adapter.rb +1 -18
- data/lib/active_record/connection_adapters.rb +1 -56
- data/lib/active_record/connection_handling.rb +25 -2
- data/lib/active_record/core.rb +33 -17
- data/lib/active_record/counter_cache.rb +33 -8
- data/lib/active_record/database_configurations/database_config.rb +9 -1
- data/lib/active_record/database_configurations/hash_config.rb +67 -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 +1 -1
- data/lib/active_record/dynamic_matchers.rb +54 -69
- data/lib/active_record/encryption/config.rb +3 -1
- data/lib/active_record/encryption/encryptable_record.rb +8 -8
- data/lib/active_record/encryption/encrypted_attribute_type.rb +11 -2
- data/lib/active_record/encryption/encryptor.rb +28 -8
- data/lib/active_record/encryption/extended_deterministic_queries.rb +4 -2
- data/lib/active_record/encryption/scheme.rb +9 -2
- data/lib/active_record/enum.rb +33 -30
- data/lib/active_record/errors.rb +33 -9
- data/lib/active_record/explain.rb +1 -1
- data/lib/active_record/explain_registry.rb +51 -2
- data/lib/active_record/filter_attribute_handler.rb +73 -0
- data/lib/active_record/fixtures.rb +2 -4
- data/lib/active_record/future_result.rb +15 -9
- data/lib/active_record/gem_version.rb +2 -2
- data/lib/active_record/inheritance.rb +1 -1
- data/lib/active_record/insert_all.rb +14 -9
- data/lib/active_record/locking/optimistic.rb +8 -1
- data/lib/active_record/locking/pessimistic.rb +5 -0
- data/lib/active_record/log_subscriber.rb +3 -13
- data/lib/active_record/middleware/shard_selector.rb +34 -17
- data/lib/active_record/migration/command_recorder.rb +45 -12
- data/lib/active_record/migration/compatibility.rb +37 -24
- data/lib/active_record/migration/default_schema_versions_formatter.rb +30 -0
- data/lib/active_record/migration.rb +48 -42
- data/lib/active_record/model_schema.rb +38 -13
- data/lib/active_record/nested_attributes.rb +6 -6
- data/lib/active_record/persistence.rb +162 -133
- data/lib/active_record/query_cache.rb +22 -15
- data/lib/active_record/query_logs.rb +100 -52
- data/lib/active_record/query_logs_formatter.rb +17 -28
- data/lib/active_record/querying.rb +8 -8
- data/lib/active_record/railtie.rb +35 -30
- data/lib/active_record/railties/controller_runtime.rb +11 -6
- data/lib/active_record/railties/databases.rake +26 -38
- 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 +53 -21
- data/lib/active_record/relation/batches/batch_enumerator.rb +4 -3
- data/lib/active_record/relation/batches.rb +147 -73
- data/lib/active_record/relation/calculations.rb +52 -40
- data/lib/active_record/relation/delegation.rb +25 -15
- data/lib/active_record/relation/finder_methods.rb +40 -24
- data/lib/active_record/relation/merger.rb +8 -8
- data/lib/active_record/relation/predicate_builder/array_handler.rb +3 -1
- data/lib/active_record/relation/predicate_builder/association_query_value.rb +9 -9
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +8 -8
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +4 -3
- data/lib/active_record/relation/predicate_builder.rb +22 -7
- data/lib/active_record/relation/query_attribute.rb +3 -1
- data/lib/active_record/relation/query_methods.rb +140 -86
- data/lib/active_record/relation/spawn_methods.rb +7 -7
- data/lib/active_record/relation/where_clause.rb +2 -9
- data/lib/active_record/relation.rb +107 -75
- data/lib/active_record/result.rb +109 -24
- data/lib/active_record/runtime_registry.rb +42 -58
- data/lib/active_record/sanitization.rb +9 -6
- data/lib/active_record/schema_dumper.rb +18 -11
- data/lib/active_record/schema_migration.rb +2 -1
- data/lib/active_record/scoping/named.rb +5 -2
- data/lib/active_record/scoping.rb +0 -1
- data/lib/active_record/signed_id.rb +43 -15
- data/lib/active_record/statement_cache.rb +24 -20
- data/lib/active_record/store.rb +51 -22
- data/lib/active_record/structured_event_subscriber.rb +85 -0
- data/lib/active_record/table_metadata.rb +6 -23
- data/lib/active_record/tasks/abstract_tasks.rb +76 -0
- data/lib/active_record/tasks/database_tasks.rb +85 -85
- data/lib/active_record/tasks/mysql_database_tasks.rb +3 -42
- data/lib/active_record/tasks/postgresql_database_tasks.rb +7 -40
- data/lib/active_record/tasks/sqlite_database_tasks.rb +16 -28
- data/lib/active_record/test_databases.rb +14 -4
- data/lib/active_record/test_fixtures.rb +39 -2
- data/lib/active_record/testing/query_assertions.rb +8 -2
- data/lib/active_record/timestamp.rb +4 -2
- data/lib/active_record/token_for.rb +1 -1
- data/lib/active_record/transaction.rb +2 -5
- data/lib/active_record/transactions.rb +37 -16
- 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 +13 -2
- data/lib/active_record/type/serialized.rb +16 -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/validations/uniqueness.rb +8 -8
- data/lib/active_record.rb +84 -49
- data/lib/arel/alias_predication.rb +2 -0
- data/lib/arel/collectors/bind.rb +2 -2
- data/lib/arel/collectors/sql_string.rb +1 -1
- data/lib/arel/collectors/substitute_binds.rb +2 -2
- data/lib/arel/crud.rb +6 -11
- data/lib/arel/nodes/binary.rb +1 -1
- data/lib/arel/nodes/count.rb +2 -2
- data/lib/arel/nodes/function.rb +4 -10
- data/lib/arel/nodes/named_function.rb +2 -2
- data/lib/arel/nodes/node.rb +2 -2
- data/lib/arel/nodes/sql_literal.rb +1 -1
- data/lib/arel/nodes.rb +0 -2
- data/lib/arel/predications.rb +1 -3
- data/lib/arel/select_manager.rb +7 -2
- data/lib/arel/table.rb +3 -7
- data/lib/arel/visitors/dot.rb +0 -3
- data/lib/arel/visitors/postgresql.rb +55 -0
- data/lib/arel/visitors/sqlite.rb +55 -8
- data/lib/arel/visitors/to_sql.rb +3 -21
- data/lib/arel.rb +3 -1
- data/lib/rails/generators/active_record/application_record/USAGE +1 -1
- metadata +16 -13
- data/lib/active_record/explain_subscriber.rb +0 -34
- data/lib/active_record/normalization.rb +0 -163
- data/lib/active_record/relation/record_fetch_warning.rb +0 -52
|
@@ -5,11 +5,12 @@ module ActiveRecord
|
|
|
5
5
|
class BatchEnumerator
|
|
6
6
|
include Enumerable
|
|
7
7
|
|
|
8
|
-
def initialize(of: 1000, start: nil, finish: nil, relation:, order: :asc, use_ranges: nil) # :nodoc:
|
|
8
|
+
def initialize(of: 1000, start: nil, finish: nil, relation:, cursor:, order: :asc, use_ranges: nil) # :nodoc:
|
|
9
9
|
@of = of
|
|
10
10
|
@relation = relation
|
|
11
11
|
@start = start
|
|
12
12
|
@finish = finish
|
|
13
|
+
@cursor = cursor
|
|
13
14
|
@order = order
|
|
14
15
|
@use_ranges = use_ranges
|
|
15
16
|
end
|
|
@@ -52,7 +53,7 @@ module ActiveRecord
|
|
|
52
53
|
def each_record(&block)
|
|
53
54
|
return to_enum(:each_record) unless block_given?
|
|
54
55
|
|
|
55
|
-
@relation.to_enum(:in_batches, of: @of, start: @start, finish: @finish, load: true, order: @order).each do |relation|
|
|
56
|
+
@relation.to_enum(:in_batches, of: @of, start: @start, finish: @finish, load: true, cursor: @cursor, order: @order).each do |relation|
|
|
56
57
|
relation.records.each(&block)
|
|
57
58
|
end
|
|
58
59
|
end
|
|
@@ -105,7 +106,7 @@ module ActiveRecord
|
|
|
105
106
|
# relation.update_all(awesome: true)
|
|
106
107
|
# end
|
|
107
108
|
def each(&block)
|
|
108
|
-
enum = @relation.to_enum(:in_batches, of: @of, start: @start, finish: @finish, load: false, order: @order, use_ranges: @use_ranges)
|
|
109
|
+
enum = @relation.to_enum(:in_batches, of: @of, start: @start, finish: @finish, load: false, cursor: @cursor, order: @order, use_ranges: @use_ranges)
|
|
109
110
|
return enum.each(&block) if block_given?
|
|
110
111
|
enum
|
|
111
112
|
end
|
|
@@ -5,7 +5,7 @@ require "active_record/relation/batches/batch_enumerator"
|
|
|
5
5
|
module ActiveRecord
|
|
6
6
|
# = Active Record \Batches
|
|
7
7
|
module Batches
|
|
8
|
-
ORDER_IGNORE_MESSAGE = "Scoped order is ignored,
|
|
8
|
+
ORDER_IGNORE_MESSAGE = "Scoped order is ignored, use :cursor with :order to configure custom order."
|
|
9
9
|
DEFAULT_ORDER = :asc
|
|
10
10
|
|
|
11
11
|
# Looping through a collection of records from the database
|
|
@@ -35,11 +35,13 @@ module ActiveRecord
|
|
|
35
35
|
#
|
|
36
36
|
# ==== Options
|
|
37
37
|
# * <tt>:batch_size</tt> - Specifies the size of the batch. Defaults to 1000.
|
|
38
|
-
# * <tt>:start</tt> - Specifies the
|
|
39
|
-
# * <tt>:finish</tt> - Specifies the
|
|
38
|
+
# * <tt>:start</tt> - Specifies the cursor column value to start from, inclusive of the value.
|
|
39
|
+
# * <tt>:finish</tt> - Specifies the cursor column value to end at, inclusive of the value.
|
|
40
40
|
# * <tt>:error_on_ignore</tt> - Overrides the application config to specify if an error should be raised when
|
|
41
41
|
# an order is present in the relation.
|
|
42
|
-
# * <tt>:
|
|
42
|
+
# * <tt>:cursor</tt> - Specifies the column to use for batching (can be a column name or an array
|
|
43
|
+
# of column names). Defaults to primary key.
|
|
44
|
+
# * <tt>:order</tt> - Specifies the cursor column order (can be +:asc+ or +:desc+ or an array consisting
|
|
43
45
|
# of :asc or :desc). Defaults to +:asc+.
|
|
44
46
|
#
|
|
45
47
|
# class Order < ActiveRecord::Base
|
|
@@ -71,20 +73,25 @@ module ActiveRecord
|
|
|
71
73
|
#
|
|
72
74
|
# NOTE: Order can be ascending (:asc) or descending (:desc). It is automatically set to
|
|
73
75
|
# ascending on the primary key ("id ASC").
|
|
74
|
-
# This also means that this method only works when the
|
|
76
|
+
# This also means that this method only works when the cursor column is
|
|
75
77
|
# orderable (e.g. an integer or string).
|
|
76
78
|
#
|
|
79
|
+
# NOTE: When using custom columns for batching, they should include at least one unique column
|
|
80
|
+
# (e.g. primary key) as a tiebreaker. Also, to reduce the likelihood of race conditions,
|
|
81
|
+
# all columns should be static (unchangeable after it was set).
|
|
82
|
+
#
|
|
77
83
|
# NOTE: By its nature, batch processing is subject to race conditions if
|
|
78
84
|
# other processes are modifying the database.
|
|
79
|
-
def find_each(start: nil, finish: nil, batch_size: 1000, error_on_ignore: nil, order: DEFAULT_ORDER, &block)
|
|
85
|
+
def find_each(start: nil, finish: nil, batch_size: 1000, error_on_ignore: nil, cursor: primary_key, order: DEFAULT_ORDER, &block)
|
|
80
86
|
if block_given?
|
|
81
|
-
find_in_batches(start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, order: order) do |records|
|
|
87
|
+
find_in_batches(start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, cursor: cursor, order: order) do |records|
|
|
82
88
|
records.each(&block)
|
|
83
89
|
end
|
|
84
90
|
else
|
|
85
|
-
enum_for(:find_each, start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, order: order) do
|
|
91
|
+
enum_for(:find_each, start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, cursor: cursor, order: order) do
|
|
86
92
|
relation = self
|
|
87
|
-
|
|
93
|
+
cursor = Array(cursor)
|
|
94
|
+
apply_limits(relation, cursor, start, finish, build_batch_orders(cursor, order)).size
|
|
88
95
|
end
|
|
89
96
|
end
|
|
90
97
|
end
|
|
@@ -109,11 +116,13 @@ module ActiveRecord
|
|
|
109
116
|
#
|
|
110
117
|
# ==== Options
|
|
111
118
|
# * <tt>:batch_size</tt> - Specifies the size of the batch. Defaults to 1000.
|
|
112
|
-
# * <tt>:start</tt> - Specifies the
|
|
113
|
-
# * <tt>:finish</tt> - Specifies the
|
|
119
|
+
# * <tt>:start</tt> - Specifies the cursor column value to start from, inclusive of the value.
|
|
120
|
+
# * <tt>:finish</tt> - Specifies the cursor column value to end at, inclusive of the value.
|
|
114
121
|
# * <tt>:error_on_ignore</tt> - Overrides the application config to specify if an error should be raised when
|
|
115
122
|
# an order is present in the relation.
|
|
116
|
-
# * <tt>:
|
|
123
|
+
# * <tt>:cursor</tt> - Specifies the column to use for batching (can be a column name or an array
|
|
124
|
+
# of column names). Defaults to primary key.
|
|
125
|
+
# * <tt>:order</tt> - Specifies the cursor column order (can be +:asc+ or +:desc+ or an array consisting
|
|
117
126
|
# of :asc or :desc). Defaults to +:asc+.
|
|
118
127
|
#
|
|
119
128
|
# class Order < ActiveRecord::Base
|
|
@@ -140,21 +149,26 @@ module ActiveRecord
|
|
|
140
149
|
#
|
|
141
150
|
# NOTE: Order can be ascending (:asc) or descending (:desc). It is automatically set to
|
|
142
151
|
# ascending on the primary key ("id ASC").
|
|
143
|
-
# This also means that this method only works when the
|
|
152
|
+
# This also means that this method only works when the cursor column is
|
|
144
153
|
# orderable (e.g. an integer or string).
|
|
145
154
|
#
|
|
155
|
+
# NOTE: When using custom columns for batching, they should include at least one unique column
|
|
156
|
+
# (e.g. primary key) as a tiebreaker. Also, to reduce the likelihood of race conditions,
|
|
157
|
+
# all columns should be static (unchangeable after it was set).
|
|
158
|
+
#
|
|
146
159
|
# NOTE: By its nature, batch processing is subject to race conditions if
|
|
147
160
|
# other processes are modifying the database.
|
|
148
|
-
def find_in_batches(start: nil, finish: nil, batch_size: 1000, error_on_ignore: nil, order: DEFAULT_ORDER)
|
|
161
|
+
def find_in_batches(start: nil, finish: nil, batch_size: 1000, error_on_ignore: nil, cursor: primary_key, order: DEFAULT_ORDER)
|
|
149
162
|
relation = self
|
|
150
163
|
unless block_given?
|
|
151
|
-
return to_enum(:find_in_batches, start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, order: order) do
|
|
152
|
-
|
|
164
|
+
return to_enum(:find_in_batches, start: start, finish: finish, batch_size: batch_size, error_on_ignore: error_on_ignore, cursor: cursor, order: order) do
|
|
165
|
+
cursor = Array(cursor)
|
|
166
|
+
total = apply_limits(relation, cursor, start, finish, build_batch_orders(cursor, order)).size
|
|
153
167
|
(total - 1).div(batch_size) + 1
|
|
154
168
|
end
|
|
155
169
|
end
|
|
156
170
|
|
|
157
|
-
in_batches(of: batch_size, start: start, finish: finish, load: true, error_on_ignore: error_on_ignore, order: order) do |batch|
|
|
171
|
+
in_batches(of: batch_size, start: start, finish: finish, load: true, error_on_ignore: error_on_ignore, cursor: cursor, order: order) do |batch|
|
|
158
172
|
yield batch.to_a
|
|
159
173
|
end
|
|
160
174
|
end
|
|
@@ -183,11 +197,13 @@ module ActiveRecord
|
|
|
183
197
|
# ==== Options
|
|
184
198
|
# * <tt>:of</tt> - Specifies the size of the batch. Defaults to 1000.
|
|
185
199
|
# * <tt>:load</tt> - Specifies if the relation should be loaded. Defaults to false.
|
|
186
|
-
# * <tt>:start</tt> - Specifies the
|
|
187
|
-
# * <tt>:finish</tt> - Specifies the
|
|
200
|
+
# * <tt>:start</tt> - Specifies the cursor column value to start from, inclusive of the value.
|
|
201
|
+
# * <tt>:finish</tt> - Specifies the cursor column value to end at, inclusive of the value.
|
|
188
202
|
# * <tt>:error_on_ignore</tt> - Overrides the application config to specify if an error should be raised when
|
|
189
203
|
# an order is present in the relation.
|
|
190
|
-
# * <tt>:
|
|
204
|
+
# * <tt>:cursor</tt> - Specifies the column to use for batching (can be a column name or an array
|
|
205
|
+
# of column names). Defaults to primary key.
|
|
206
|
+
# * <tt>:order</tt> - Specifies the cursor column order (can be +:asc+ or +:desc+ or an array consisting
|
|
191
207
|
# of :asc or :desc). Defaults to +:asc+.
|
|
192
208
|
#
|
|
193
209
|
# class Order < ActiveRecord::Base
|
|
@@ -231,22 +247,25 @@ module ActiveRecord
|
|
|
231
247
|
#
|
|
232
248
|
# NOTE: Order can be ascending (:asc) or descending (:desc). It is automatically set to
|
|
233
249
|
# ascending on the primary key ("id ASC").
|
|
234
|
-
# This also means that this method only works when the
|
|
250
|
+
# This also means that this method only works when the cursor column is
|
|
235
251
|
# orderable (e.g. an integer or string).
|
|
236
252
|
#
|
|
253
|
+
# NOTE: When using custom columns for batching, they should include at least one unique column
|
|
254
|
+
# (e.g. primary key) as a tiebreaker. Also, to reduce the likelihood of race conditions,
|
|
255
|
+
# all columns should be static (unchangeable after it was set).
|
|
256
|
+
#
|
|
237
257
|
# NOTE: By its nature, batch processing is subject to race conditions if
|
|
238
258
|
# other processes are modifying the database.
|
|
239
|
-
def in_batches(of: 1000, start: nil, finish: nil, load: false, error_on_ignore: nil, order: DEFAULT_ORDER, use_ranges: nil, &block)
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
end
|
|
259
|
+
def in_batches(of: 1000, start: nil, finish: nil, load: false, error_on_ignore: nil, cursor: primary_key, order: DEFAULT_ORDER, use_ranges: nil, &block)
|
|
260
|
+
cursor = Array(cursor).map(&:to_s)
|
|
261
|
+
ensure_valid_options_for_batching!(cursor, start, finish, order)
|
|
243
262
|
|
|
244
263
|
if arel.orders.present?
|
|
245
264
|
act_on_ignored_order(error_on_ignore)
|
|
246
265
|
end
|
|
247
266
|
|
|
248
267
|
unless block
|
|
249
|
-
return BatchEnumerator.new(of: of, start: start, finish: finish, relation: self, order: order, use_ranges: use_ranges)
|
|
268
|
+
return BatchEnumerator.new(of: of, start: start, finish: finish, relation: self, cursor: cursor, order: order, use_ranges: use_ranges)
|
|
250
269
|
end
|
|
251
270
|
|
|
252
271
|
batch_limit = of
|
|
@@ -261,6 +280,7 @@ module ActiveRecord
|
|
|
261
280
|
relation: self,
|
|
262
281
|
start: start,
|
|
263
282
|
finish: finish,
|
|
283
|
+
cursor: cursor,
|
|
264
284
|
order: order,
|
|
265
285
|
batch_limit: batch_limit,
|
|
266
286
|
&block
|
|
@@ -271,6 +291,7 @@ module ActiveRecord
|
|
|
271
291
|
start: start,
|
|
272
292
|
finish: finish,
|
|
273
293
|
load: load,
|
|
294
|
+
cursor: cursor,
|
|
274
295
|
order: order,
|
|
275
296
|
use_ranges: use_ranges,
|
|
276
297
|
remaining: remaining,
|
|
@@ -281,28 +302,51 @@ module ActiveRecord
|
|
|
281
302
|
end
|
|
282
303
|
|
|
283
304
|
private
|
|
284
|
-
def
|
|
285
|
-
|
|
286
|
-
|
|
305
|
+
def ensure_valid_options_for_batching!(cursor, start, finish, order)
|
|
306
|
+
if start && Array(start).size != cursor.size
|
|
307
|
+
raise ArgumentError, ":start must contain one value per cursor column"
|
|
308
|
+
end
|
|
309
|
+
|
|
310
|
+
if finish && Array(finish).size != cursor.size
|
|
311
|
+
raise ArgumentError, ":finish must contain one value per cursor column"
|
|
312
|
+
end
|
|
313
|
+
|
|
314
|
+
if (Array(primary_key) - cursor).any?
|
|
315
|
+
indexes = model.schema_cache.indexes(table_name)
|
|
316
|
+
unique_index = indexes.find { |index| index.unique && index.where.nil? && (Array(index.columns) - cursor).empty? }
|
|
317
|
+
|
|
318
|
+
unless unique_index
|
|
319
|
+
raise ArgumentError, ":cursor must include a primary key or other unique column(s)"
|
|
320
|
+
end
|
|
321
|
+
end
|
|
322
|
+
|
|
323
|
+
if (Array(order) - [:asc, :desc]).any?
|
|
324
|
+
raise ArgumentError, ":order must be :asc or :desc or an array consisting of :asc or :desc, got #{order.inspect}"
|
|
325
|
+
end
|
|
326
|
+
end
|
|
327
|
+
|
|
328
|
+
def apply_limits(relation, cursor, start, finish, batch_orders)
|
|
329
|
+
relation = apply_start_limit(relation, cursor, start, batch_orders) if start
|
|
330
|
+
relation = apply_finish_limit(relation, cursor, finish, batch_orders) if finish
|
|
287
331
|
relation
|
|
288
332
|
end
|
|
289
333
|
|
|
290
|
-
def apply_start_limit(relation, start, batch_orders)
|
|
334
|
+
def apply_start_limit(relation, cursor, start, batch_orders)
|
|
291
335
|
operators = batch_orders.map do |_column, order|
|
|
292
336
|
order == :desc ? :lteq : :gteq
|
|
293
337
|
end
|
|
294
|
-
batch_condition(relation,
|
|
338
|
+
batch_condition(relation, cursor, start, operators)
|
|
295
339
|
end
|
|
296
340
|
|
|
297
|
-
def apply_finish_limit(relation, finish, batch_orders)
|
|
341
|
+
def apply_finish_limit(relation, cursor, finish, batch_orders)
|
|
298
342
|
operators = batch_orders.map do |_column, order|
|
|
299
343
|
order == :desc ? :gteq : :lteq
|
|
300
344
|
end
|
|
301
|
-
batch_condition(relation,
|
|
345
|
+
batch_condition(relation, cursor, finish, operators)
|
|
302
346
|
end
|
|
303
347
|
|
|
304
|
-
def batch_condition(relation,
|
|
305
|
-
cursor_positions =
|
|
348
|
+
def batch_condition(relation, cursor, values, operators)
|
|
349
|
+
cursor_positions = cursor.zip(Array(values), operators)
|
|
306
350
|
|
|
307
351
|
first_clause_column, first_clause_value, operator = cursor_positions.pop
|
|
308
352
|
where_clause = predicate_builder[first_clause_column, first_clause_value, operator]
|
|
@@ -316,9 +360,9 @@ module ActiveRecord
|
|
|
316
360
|
relation.where(where_clause)
|
|
317
361
|
end
|
|
318
362
|
|
|
319
|
-
def build_batch_orders(order)
|
|
320
|
-
|
|
321
|
-
[column,
|
|
363
|
+
def build_batch_orders(cursor, order)
|
|
364
|
+
cursor.zip(Array(order)).map do |column, order_|
|
|
365
|
+
[column, order_ || DEFAULT_ORDER]
|
|
322
366
|
end
|
|
323
367
|
end
|
|
324
368
|
|
|
@@ -327,34 +371,28 @@ module ActiveRecord
|
|
|
327
371
|
|
|
328
372
|
if raise_error
|
|
329
373
|
raise ArgumentError.new(ORDER_IGNORE_MESSAGE)
|
|
330
|
-
elsif logger
|
|
331
|
-
logger.warn(ORDER_IGNORE_MESSAGE)
|
|
374
|
+
elsif model.logger
|
|
375
|
+
model.logger.warn(ORDER_IGNORE_MESSAGE)
|
|
332
376
|
end
|
|
333
377
|
end
|
|
334
378
|
|
|
335
|
-
def
|
|
336
|
-
Array(primary_key).zip(Array(order))
|
|
337
|
-
end
|
|
338
|
-
|
|
339
|
-
def batch_on_loaded_relation(relation:, start:, finish:, order:, batch_limit:)
|
|
379
|
+
def batch_on_loaded_relation(relation:, start:, finish:, cursor:, order:, batch_limit:)
|
|
340
380
|
records = relation.to_a
|
|
381
|
+
order = build_batch_orders(cursor, order).map(&:second)
|
|
341
382
|
|
|
342
383
|
if start || finish
|
|
343
384
|
records = records.filter do |record|
|
|
344
|
-
|
|
385
|
+
values = record_cursor_values(record, cursor)
|
|
345
386
|
|
|
346
|
-
|
|
347
|
-
(
|
|
348
|
-
else
|
|
349
|
-
(start.nil? || id <= start) && (finish.nil? || id >= finish)
|
|
350
|
-
end
|
|
387
|
+
(start.nil? || compare_values_for_order(values, Array(start), order) >= 0) &&
|
|
388
|
+
(finish.nil? || compare_values_for_order(values, Array(finish), order) <= 0)
|
|
351
389
|
end
|
|
352
390
|
end
|
|
353
391
|
|
|
354
|
-
records.
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
392
|
+
records.sort! do |record1, record2|
|
|
393
|
+
values1 = record_cursor_values(record1, cursor)
|
|
394
|
+
values2 = record_cursor_values(record2, cursor)
|
|
395
|
+
compare_values_for_order(values1, values2, order)
|
|
358
396
|
end
|
|
359
397
|
|
|
360
398
|
records.each_slice(batch_limit) do |subrecords|
|
|
@@ -367,44 +405,79 @@ module ActiveRecord
|
|
|
367
405
|
nil
|
|
368
406
|
end
|
|
369
407
|
|
|
370
|
-
def
|
|
371
|
-
|
|
408
|
+
def record_cursor_values(record, cursor)
|
|
409
|
+
record.attributes.slice(*cursor).values
|
|
410
|
+
end
|
|
411
|
+
|
|
412
|
+
# This is a custom implementation of `<=>` operator,
|
|
413
|
+
# which also takes into account how the collection will be ordered.
|
|
414
|
+
def compare_values_for_order(values1, values2, order)
|
|
415
|
+
values1.each_with_index do |element1, index|
|
|
416
|
+
element2 = values2[index]
|
|
417
|
+
direction = order[index]
|
|
418
|
+
comparison = element1 <=> element2
|
|
419
|
+
comparison = -comparison if direction == :desc
|
|
420
|
+
return comparison if comparison != 0
|
|
421
|
+
end
|
|
422
|
+
|
|
423
|
+
0
|
|
424
|
+
end
|
|
425
|
+
|
|
426
|
+
def batch_on_unloaded_relation(relation:, start:, finish:, load:, cursor:, order:, use_ranges:, remaining:, batch_limit:)
|
|
427
|
+
batch_orders = build_batch_orders(cursor, order)
|
|
372
428
|
relation = relation.reorder(batch_orders.to_h).limit(batch_limit)
|
|
373
|
-
relation = apply_limits(relation, start, finish, batch_orders)
|
|
429
|
+
relation = apply_limits(relation, cursor, start, finish, batch_orders)
|
|
374
430
|
relation.skip_query_cache! # Retaining the results in the query cache would undermine the point of batching
|
|
375
431
|
batch_relation = relation
|
|
376
|
-
empty_scope = to_sql ==
|
|
432
|
+
empty_scope = to_sql == model.unscoped.all.to_sql
|
|
377
433
|
|
|
378
434
|
loop do
|
|
379
435
|
if load
|
|
380
436
|
records = batch_relation.records
|
|
381
|
-
|
|
382
|
-
|
|
437
|
+
values = records.pluck(*cursor)
|
|
438
|
+
values_size = values.size
|
|
439
|
+
values_last = values.last
|
|
440
|
+
yielded_relation = rewhere(cursor => values)
|
|
383
441
|
yielded_relation.load_records(records)
|
|
384
442
|
elsif (empty_scope && use_ranges != false) || use_ranges
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
443
|
+
# Efficiently peak at the last value for the next batch using offset and limit.
|
|
444
|
+
values_size = remaining ? [batch_limit, remaining].min : batch_limit
|
|
445
|
+
values_last = batch_relation.offset(values_size - 1).pick(*cursor)
|
|
446
|
+
|
|
447
|
+
# If the last value is not found using offset, there is at most one more batch of size < batch_limit.
|
|
448
|
+
# Retry by getting the whole list of remaining values so that we have the exact size and last value.
|
|
449
|
+
unless values_last
|
|
450
|
+
values = batch_relation.pluck(*cursor)
|
|
451
|
+
values_size = values.size
|
|
452
|
+
values_last = values.last
|
|
453
|
+
end
|
|
454
|
+
|
|
455
|
+
# Finally, build the yielded relation if at least one value found.
|
|
456
|
+
if values_last
|
|
457
|
+
yielded_relation = apply_finish_limit(batch_relation, cursor, values_last, batch_orders)
|
|
389
458
|
yielded_relation = yielded_relation.except(:limit, :order)
|
|
390
459
|
yielded_relation.skip_query_cache!(false)
|
|
391
460
|
end
|
|
392
461
|
else
|
|
393
|
-
|
|
394
|
-
|
|
462
|
+
values = batch_relation.pluck(*cursor)
|
|
463
|
+
values_size = values.size
|
|
464
|
+
values_last = values.last
|
|
465
|
+
yielded_relation = rewhere(cursor => values)
|
|
395
466
|
end
|
|
396
467
|
|
|
397
|
-
break if
|
|
468
|
+
break if values_size == 0
|
|
398
469
|
|
|
399
|
-
|
|
400
|
-
|
|
470
|
+
if [values_last].flatten.any?(nil)
|
|
471
|
+
raise ArgumentError, "Not all of the batch cursor columns were included in the custom select clause "\
|
|
472
|
+
"or some columns contain nil."
|
|
473
|
+
end
|
|
401
474
|
|
|
402
475
|
yield yielded_relation
|
|
403
476
|
|
|
404
|
-
break if
|
|
477
|
+
break if values_size < batch_limit
|
|
405
478
|
|
|
406
479
|
if limit_value
|
|
407
|
-
remaining -=
|
|
480
|
+
remaining -= values_size
|
|
408
481
|
|
|
409
482
|
if remaining == 0
|
|
410
483
|
# Saves a useless iteration when the limit is a multiple of the
|
|
@@ -422,7 +495,8 @@ module ActiveRecord
|
|
|
422
495
|
end
|
|
423
496
|
operators << (last_order == :desc ? :lt : :gt)
|
|
424
497
|
|
|
425
|
-
|
|
498
|
+
cursor_value = values_last
|
|
499
|
+
batch_relation = batch_condition(relation, cursor, cursor_value, operators)
|
|
426
500
|
end
|
|
427
501
|
|
|
428
502
|
nil
|
|
@@ -233,7 +233,7 @@ module ActiveRecord
|
|
|
233
233
|
if operation == "count"
|
|
234
234
|
unless distinct_value || distinct_select?(column_name || select_for_count)
|
|
235
235
|
relation.distinct!
|
|
236
|
-
relation.select_values = Array(
|
|
236
|
+
relation.select_values = Array(model.primary_key || table[Arel.star])
|
|
237
237
|
end
|
|
238
238
|
# PostgreSQL: ORDER BY expressions must appear in SELECT list when using DISTINCT
|
|
239
239
|
relation.order_values = [] if group_values.empty?
|
|
@@ -274,14 +274,23 @@ module ActiveRecord
|
|
|
274
274
|
# # SELECT people.id FROM people WHERE people.age = 21 LIMIT 5
|
|
275
275
|
# # => [2, 3]
|
|
276
276
|
#
|
|
277
|
-
# Comment.joins(:person).pluck(:id, person:
|
|
278
|
-
# # SELECT comments.id,
|
|
277
|
+
# Comment.joins(:person).pluck(:id, person: :id)
|
|
278
|
+
# # SELECT comments.id, person.id FROM comments INNER JOIN people person ON person.id = comments.person_id
|
|
279
279
|
# # => [[1, 2], [2, 2]]
|
|
280
280
|
#
|
|
281
|
+
# Comment.joins(:person).pluck(:id, person: [:id, :name])
|
|
282
|
+
# # SELECT comments.id, person.id, person.name FROM comments INNER JOIN people person ON person.id = comments.person_id
|
|
283
|
+
# # => [[1, 2, 'David'], [2, 2, 'David']]
|
|
284
|
+
#
|
|
281
285
|
# Person.pluck(Arel.sql('DATEDIFF(updated_at, created_at)'))
|
|
282
286
|
# # SELECT DATEDIFF(updated_at, created_at) FROM people
|
|
283
287
|
# # => ['0', '27761', '173']
|
|
284
288
|
#
|
|
289
|
+
# Be aware that #pluck ignores any previous select clauses
|
|
290
|
+
#
|
|
291
|
+
# Person.select(:name).pluck(:id)
|
|
292
|
+
# # SELECT people.id FROM people
|
|
293
|
+
#
|
|
285
294
|
# See also #ids.
|
|
286
295
|
def pluck(*column_names)
|
|
287
296
|
if @none
|
|
@@ -305,16 +314,16 @@ module ActiveRecord
|
|
|
305
314
|
relation = apply_join_dependency
|
|
306
315
|
relation.pluck(*column_names)
|
|
307
316
|
else
|
|
308
|
-
|
|
317
|
+
model.disallow_raw_sql!(flattened_args(column_names))
|
|
309
318
|
relation = spawn
|
|
310
319
|
columns = relation.arel_columns(column_names)
|
|
311
320
|
relation.select_values = columns
|
|
312
321
|
result = skip_query_cache_if_necessary do
|
|
313
|
-
if where_clause.contradiction?
|
|
322
|
+
if where_clause.contradiction? && !possible_aggregation?(column_names)
|
|
314
323
|
ActiveRecord::Result.empty(async: @async)
|
|
315
324
|
else
|
|
316
|
-
|
|
317
|
-
c.select_all(relation.arel, "#{
|
|
325
|
+
model.with_connection do |c|
|
|
326
|
+
c.select_all(relation.arel, "#{model.name} Pluck", async: @async)
|
|
318
327
|
end
|
|
319
328
|
end
|
|
320
329
|
end
|
|
@@ -390,8 +399,8 @@ module ActiveRecord
|
|
|
390
399
|
ActiveRecord::Result.empty
|
|
391
400
|
else
|
|
392
401
|
skip_query_cache_if_necessary do
|
|
393
|
-
|
|
394
|
-
c.select_all(relation, "#{
|
|
402
|
+
model.with_connection do |c|
|
|
403
|
+
c.select_all(relation, "#{model.name} Ids", async: @async)
|
|
395
404
|
end
|
|
396
405
|
end
|
|
397
406
|
end
|
|
@@ -407,16 +416,19 @@ module ActiveRecord
|
|
|
407
416
|
|
|
408
417
|
protected
|
|
409
418
|
def aggregate_column(column_name)
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
419
|
+
case column_name
|
|
420
|
+
when Arel::Expressions
|
|
421
|
+
column_name
|
|
422
|
+
when :all
|
|
423
|
+
Arel.star
|
|
424
|
+
else
|
|
425
|
+
arel_column(column_name.to_s)
|
|
414
426
|
end
|
|
415
427
|
end
|
|
416
428
|
|
|
417
429
|
private
|
|
418
430
|
def all_attributes?(column_names)
|
|
419
|
-
(column_names.map(&:to_s) -
|
|
431
|
+
(column_names.map(&:to_s) - model.attribute_names - model.attribute_aliases.keys).empty?
|
|
420
432
|
end
|
|
421
433
|
|
|
422
434
|
def has_include?(column_name)
|
|
@@ -454,6 +466,16 @@ module ActiveRecord
|
|
|
454
466
|
column_name.is_a?(::String) && /\bDISTINCT[\s(]/i.match?(column_name)
|
|
455
467
|
end
|
|
456
468
|
|
|
469
|
+
def possible_aggregation?(column_names)
|
|
470
|
+
column_names.all? do |column_name|
|
|
471
|
+
if column_name.is_a?(String)
|
|
472
|
+
column_name.include?("(")
|
|
473
|
+
else
|
|
474
|
+
Arel.arel_node?(column_name)
|
|
475
|
+
end
|
|
476
|
+
end
|
|
477
|
+
end
|
|
478
|
+
|
|
457
479
|
def operation_over_aggregate_column(column, operation, distinct)
|
|
458
480
|
operation == "count" ? column.count(distinct) : column.public_send(operation)
|
|
459
481
|
end
|
|
@@ -486,8 +508,8 @@ module ActiveRecord
|
|
|
486
508
|
end
|
|
487
509
|
else
|
|
488
510
|
skip_query_cache_if_necessary do
|
|
489
|
-
|
|
490
|
-
c.select_all(query_builder, "#{
|
|
511
|
+
model.with_connection do |c|
|
|
512
|
+
c.select_all(query_builder, "#{model.name} #{operation.capitalize}", async: @async)
|
|
491
513
|
end
|
|
492
514
|
end
|
|
493
515
|
end
|
|
@@ -505,10 +527,9 @@ module ActiveRecord
|
|
|
505
527
|
|
|
506
528
|
def execute_grouped_calculation(operation, column_name, distinct) # :nodoc:
|
|
507
529
|
group_fields = group_values
|
|
508
|
-
group_fields = group_fields.uniq if group_fields.size > 1
|
|
509
530
|
|
|
510
531
|
if group_fields.size == 1 && group_fields.first.respond_to?(:to_sym)
|
|
511
|
-
association =
|
|
532
|
+
association = model._reflect_on_association(group_fields.first)
|
|
512
533
|
associated = association && association.belongs_to? # only count belongs_to associations
|
|
513
534
|
group_fields = Array(association.foreign_key) if associated
|
|
514
535
|
end
|
|
@@ -516,7 +537,7 @@ module ActiveRecord
|
|
|
516
537
|
relation = except(:group).distinct!(false)
|
|
517
538
|
group_fields = relation.arel_columns(group_fields)
|
|
518
539
|
|
|
519
|
-
|
|
540
|
+
model.with_connection do |connection|
|
|
520
541
|
column_alias_tracker = ColumnAliasTracker.new(connection)
|
|
521
542
|
|
|
522
543
|
group_aliases = group_fields.map { |field|
|
|
@@ -528,13 +549,13 @@ module ActiveRecord
|
|
|
528
549
|
column = relation.aggregate_column(column_name)
|
|
529
550
|
column_alias = column_alias_tracker.alias_for("#{operation} #{column_name.to_s.downcase}")
|
|
530
551
|
select_value = operation_over_aggregate_column(column, operation, distinct)
|
|
531
|
-
select_value.as(adapter_class.quote_column_name(column_alias))
|
|
552
|
+
select_value = select_value.as(model.adapter_class.quote_column_name(column_alias))
|
|
532
553
|
|
|
533
554
|
select_values = [select_value]
|
|
534
555
|
select_values += self.select_values unless having_clause.empty?
|
|
535
556
|
|
|
536
557
|
select_values.concat group_columns.map { |aliaz, field|
|
|
537
|
-
aliaz = adapter_class.quote_column_name(aliaz)
|
|
558
|
+
aliaz = model.adapter_class.quote_column_name(aliaz)
|
|
538
559
|
if field.respond_to?(:as)
|
|
539
560
|
field.as(aliaz)
|
|
540
561
|
else
|
|
@@ -546,7 +567,7 @@ module ActiveRecord
|
|
|
546
567
|
relation.select_values = select_values
|
|
547
568
|
|
|
548
569
|
result = skip_query_cache_if_necessary do
|
|
549
|
-
connection.select_all(relation.arel, "#{
|
|
570
|
+
connection.select_all(relation.arel, "#{model.name} #{operation.capitalize}", async: @async)
|
|
550
571
|
end
|
|
551
572
|
|
|
552
573
|
result.then do |calculated_data|
|
|
@@ -588,7 +609,7 @@ module ActiveRecord
|
|
|
588
609
|
|
|
589
610
|
def type_for(field, &block)
|
|
590
611
|
field_name = field.respond_to?(:name) ? field.name.to_s : field.to_s.split(".").last
|
|
591
|
-
|
|
612
|
+
model.type_for_attribute(field_name, &block)
|
|
592
613
|
end
|
|
593
614
|
|
|
594
615
|
def lookup_cast_type_from_join_dependencies(name, join_dependencies = build_join_dependencies)
|
|
@@ -601,12 +622,12 @@ module ActiveRecord
|
|
|
601
622
|
|
|
602
623
|
def type_cast_pluck_values(result, columns)
|
|
603
624
|
cast_types = if result.columns.size != columns.size
|
|
604
|
-
|
|
625
|
+
model.attribute_types
|
|
605
626
|
else
|
|
606
627
|
join_dependencies = nil
|
|
607
628
|
columns.map.with_index do |column, i|
|
|
608
629
|
column.try(:type_caster) ||
|
|
609
|
-
|
|
630
|
+
model.attribute_types.fetch(name = result.columns[i]) do
|
|
610
631
|
join_dependencies ||= build_join_dependencies
|
|
611
632
|
lookup_cast_type_from_join_dependencies(name, join_dependencies) ||
|
|
612
633
|
result.column_types[i] || Type.default_value
|
|
@@ -635,22 +656,12 @@ module ActiveRecord
|
|
|
635
656
|
end
|
|
636
657
|
|
|
637
658
|
def select_for_count
|
|
638
|
-
if select_values.
|
|
639
|
-
return select_values.first if select_values.one?
|
|
640
|
-
|
|
641
|
-
select_values.map do |field|
|
|
642
|
-
column = arel_column(field.to_s) do |attr_name|
|
|
643
|
-
Arel.sql(attr_name)
|
|
644
|
-
end
|
|
645
|
-
|
|
646
|
-
if column.is_a?(Arel::Nodes::SqlLiteral)
|
|
647
|
-
column
|
|
648
|
-
else
|
|
649
|
-
"#{adapter_class.quote_table_name(column.relation.name)}.#{adapter_class.quote_column_name(column.name)}"
|
|
650
|
-
end
|
|
651
|
-
end.join(", ")
|
|
652
|
-
else
|
|
659
|
+
if select_values.empty?
|
|
653
660
|
:all
|
|
661
|
+
else
|
|
662
|
+
with_connection do |conn|
|
|
663
|
+
arel_columns(select_values).map { |column| conn.visitor.compile(column) }.join(", ")
|
|
664
|
+
end
|
|
654
665
|
end
|
|
655
666
|
end
|
|
656
667
|
|
|
@@ -665,6 +676,7 @@ module ActiveRecord
|
|
|
665
676
|
if column_name == :all
|
|
666
677
|
column_alias = Arel.star
|
|
667
678
|
relation.select_values = [ Arel.sql(FinderMethods::ONE_AS_ONE) ] unless distinct
|
|
679
|
+
relation.unscope!(:order)
|
|
668
680
|
else
|
|
669
681
|
column_alias = Arel.sql("count_column")
|
|
670
682
|
relation.select_values = [ relation.aggregate_column(column_name).as(column_alias) ]
|