1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Include Enumerable in ActiveRecord::Relation

After discussing, we've decided it makes more sense to include it. We're
already forwarding every conflicting method to `to_a`, and there's no
conflation of concerns. `Enumerable` has no mutating methods, and it
just allows us to simplify the code. No existing methods will have a
change in behavior. Un-overridden Enumerable methods will simply
delegate to `each`.

[Sean Griffin & bogdan]
This commit is contained in:
Sean Griffin 2015-06-19 15:35:35 -06:00
parent 7d14bd3ff5
commit b644964b2b
5 changed files with 21 additions and 37 deletions

View file

@ -1,3 +1,7 @@
* Include the `Enumerable` module in `ActiveRecord::Relation`
*Sean Griffin & bogdan*
* Use `Enumerable#sum` in `ActiveRecord::Relation` if a block is given. * Use `Enumerable#sum` in `ActiveRecord::Relation` if a block is given.
*Sean Griffin* *Sean Griffin*

View file

@ -15,6 +15,7 @@ module ActiveRecord
VALUE_METHODS = MULTI_VALUE_METHODS + SINGLE_VALUE_METHODS + CLAUSE_METHODS VALUE_METHODS = MULTI_VALUE_METHODS + SINGLE_VALUE_METHODS + CLAUSE_METHODS
include Enumerable
include FinderMethods, Calculations, SpawnMethods, QueryMethods, Batches, Explain, Delegation include FinderMethods, Calculations, SpawnMethods, QueryMethods, Batches, Explain, Delegation
attr_reader :table, :klass, :loaded, :predicate_builder attr_reader :table, :klass, :loaded, :predicate_builder
@ -275,38 +276,26 @@ module ActiveRecord
# Returns true if there are no records. # Returns true if there are no records.
def none? def none?
if block_given? return super if block_given?
to_a.none? { |*block_args| yield(*block_args) } empty?
else
empty?
end
end end
# Returns true if there are any records. # Returns true if there are any records.
def any? def any?
if block_given? return super if block_given?
to_a.any? { |*block_args| yield(*block_args) } !empty?
else
!empty?
end
end end
# Returns true if there is exactly one record. # Returns true if there is exactly one record.
def one? def one?
if block_given? return super if block_given?
to_a.one? { |*block_args| yield(*block_args) } limit_value ? to_a.one? : size == 1
else
limit_value ? to_a.one? : size == 1
end
end end
# Returns true if there is more than one record. # Returns true if there is more than one record.
def many? def many?
if block_given? return super if block_given?
to_a.many? { |*block_args| yield(*block_args) } limit_value ? to_a.many? : size > 1
else
limit_value ? to_a.many? : size > 1
end
end end
# Scope all queries to the current scope. # Scope all queries to the current scope.

View file

@ -70,12 +70,9 @@ module ActiveRecord
# +calculate+ for examples with options. # +calculate+ for examples with options.
# #
# Person.sum(:age) # => 4562 # Person.sum(:age) # => 4562
def sum(*args, &block) def sum(*args)
if block_given? return super if block_given?
to_a.sum(&block) calculate(:sum, *args)
else
calculate(:sum, *args)
end
end end
# This calculates aggregate values in the given column. Methods for count, sum, average, # This calculates aggregate values in the given column. Methods for count, sum, average,

View file

@ -62,11 +62,8 @@ module ActiveRecord
# Person.where(name: 'Spartacus', rating: 4).pluck(:field1, :field2) # Person.where(name: 'Spartacus', rating: 4).pluck(:field1, :field2)
# # returns an Array of the required fields. # # returns an Array of the required fields.
def find(*args) def find(*args)
if block_given? return super if block_given?
to_a.find(*args) { |*block_args| yield(*block_args) } find_with_ids(*args)
else
find_with_ids(*args)
end
end end
# Finds the first record matching the specified conditions. There # Finds the first record matching the specified conditions. There

View file

@ -242,12 +242,9 @@ module ActiveRecord
# Model.select(:field).first.other_field # Model.select(:field).first.other_field
# # => ActiveModel::MissingAttributeError: missing attribute: other_field # # => ActiveModel::MissingAttributeError: missing attribute: other_field
def select(*fields) def select(*fields)
if block_given? return super if block_given?
to_a.select { |*block_args| yield(*block_args) } raise ArgumentError, 'Call this with at least one field' if fields.empty?
else spawn._select!(*fields)
raise ArgumentError, 'Call this with at least one field' if fields.empty?
spawn._select!(*fields)
end
end end
def _select!(*fields) # :nodoc: def _select!(*fields) # :nodoc: