mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
Use ARel's joins when building a query for finding records with included
associations.
This commit is contained in:
parent
b68ef73344
commit
9ac01fad19
5 changed files with 89 additions and 75 deletions
|
@ -1673,14 +1673,20 @@ module ActiveRecord
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def construct_finder_sql_with_included_associations(options, join_dependency)
|
def construct_finder_arel_with_included_associations(options, join_dependency)
|
||||||
scope = scope(:find)
|
scope = scope(:find)
|
||||||
|
|
||||||
relation = arel_table((scope && scope[:from]) || options[:from])
|
relation = arel_table((scope && scope[:from]) || options[:from])
|
||||||
|
|
||||||
joins = join_dependency.join_associations.collect{|join| join.association_join }.join
|
for association in join_dependency.join_associations
|
||||||
joins << construct_join(options[:joins], scope)
|
if association.relation.is_a?(Array)
|
||||||
relation.join(joins)
|
relation.join(association.relation.first, association.join_type).on(association.association_join.first)
|
||||||
|
relation.join(association.relation.last, association.join_type).on(association.association_join.last)
|
||||||
|
else
|
||||||
|
relation.join(association.relation, association.join_type).on(association.association_join)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
relation.join(construct_join(options[:joins], scope))
|
||||||
|
|
||||||
relation.where(construct_conditions(options[:conditions], scope))
|
relation.where(construct_conditions(options[:conditions], scope))
|
||||||
relation.where(construct_arel_limited_ids_condition(options, join_dependency)) if !using_limitable_reflections?(join_dependency.reflections) && ((scope && scope[:limit]) || options[:limit])
|
relation.where(construct_arel_limited_ids_condition(options, join_dependency)) if !using_limitable_reflections?(join_dependency.reflections) && ((scope && scope[:limit]) || options[:limit])
|
||||||
|
@ -1690,7 +1696,11 @@ module ActiveRecord
|
||||||
relation.order(construct_order(options[:order], scope))
|
relation.order(construct_order(options[:order], scope))
|
||||||
relation.take(construct_limit(options[:limit], scope)) if using_limitable_reflections?(join_dependency.reflections)
|
relation.take(construct_limit(options[:limit], scope)) if using_limitable_reflections?(join_dependency.reflections)
|
||||||
|
|
||||||
sanitize_sql(relation.to_sql)
|
relation
|
||||||
|
end
|
||||||
|
|
||||||
|
def construct_finder_sql_with_included_associations(options, join_dependency)
|
||||||
|
sanitize_sql(construct_finder_arel_with_included_associations(options, join_dependency).to_sql)
|
||||||
end
|
end
|
||||||
|
|
||||||
def construct_arel_limited_ids_condition(options, join_dependency)
|
def construct_arel_limited_ids_condition(options, join_dependency)
|
||||||
|
@ -1716,9 +1726,15 @@ module ActiveRecord
|
||||||
|
|
||||||
relation = arel_table(options[:from])
|
relation = arel_table(options[:from])
|
||||||
|
|
||||||
joins = join_dependency.join_associations.collect{|join| join.association_join }.join
|
for association in join_dependency.join_associations
|
||||||
joins << construct_join(options[:joins], scope)
|
if association.relation.is_a?(Array)
|
||||||
relation.join(joins)
|
relation.join(association.relation.first, association.join_type).on(association.association_join.first)
|
||||||
|
relation.join(association.relation.last, association.join_type).on(association.association_join.last)
|
||||||
|
else
|
||||||
|
relation.join(association.relation, association.join_type).on(association.association_join)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
relation.join(construct_join(options[:joins], scope))
|
||||||
|
|
||||||
relation.where(construct_conditions(options[:conditions], scope))
|
relation.where(construct_conditions(options[:conditions], scope))
|
||||||
relation.project(connection.distinct("#{connection.quote_table_name table_name}.#{primary_key}", construct_order(options[:order], scope(:find)).join(",")))
|
relation.project(connection.distinct("#{connection.quote_table_name table_name}.#{primary_key}", construct_order(options[:order], scope(:find)).join(",")))
|
||||||
|
@ -1907,12 +1923,6 @@ module ActiveRecord
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def join_for_table_name(table_name)
|
|
||||||
join = (@joins.select{|j|j.aliased_table_name == table_name.gsub(/^\"(.*)\"$/){$1} }.first) rescue nil
|
|
||||||
return join unless join.nil?
|
|
||||||
@joins.select{|j|j.is_a?(JoinAssociation) && j.aliased_join_table_name == table_name.gsub(/^\"(.*)\"$/){$1} }.first rescue nil
|
|
||||||
end
|
|
||||||
|
|
||||||
def joins_for_table_name(table_name)
|
def joins_for_table_name(table_name)
|
||||||
join = join_for_table_name(table_name)
|
join = join_for_table_name(table_name)
|
||||||
result = nil
|
result = nil
|
||||||
|
@ -2088,19 +2098,18 @@ module ActiveRecord
|
||||||
connection = reflection.active_record.connection
|
connection = reflection.active_record.connection
|
||||||
join = case reflection.macro
|
join = case reflection.macro
|
||||||
when :has_and_belongs_to_many
|
when :has_and_belongs_to_many
|
||||||
" #{join_type} %s ON %s.%s = %s.%s " % [
|
["%s.%s = %s.%s " % [
|
||||||
table_alias_for(options[:join_table], aliased_join_table_name),
|
|
||||||
connection.quote_table_name(aliased_join_table_name),
|
connection.quote_table_name(aliased_join_table_name),
|
||||||
options[:foreign_key] || reflection.active_record.to_s.foreign_key,
|
options[:foreign_key] || reflection.active_record.to_s.foreign_key,
|
||||||
connection.quote_table_name(parent.aliased_table_name),
|
connection.quote_table_name(parent.aliased_table_name),
|
||||||
reflection.active_record.primary_key] +
|
reflection.active_record.primary_key],
|
||||||
" #{join_type} %s ON %s.%s = %s.%s " % [
|
"%s.%s = %s.%s " % [
|
||||||
table_name_and_alias,
|
|
||||||
connection.quote_table_name(aliased_table_name),
|
connection.quote_table_name(aliased_table_name),
|
||||||
klass.primary_key,
|
klass.primary_key,
|
||||||
connection.quote_table_name(aliased_join_table_name),
|
connection.quote_table_name(aliased_join_table_name),
|
||||||
options[:association_foreign_key] || klass.to_s.foreign_key
|
options[:association_foreign_key] || klass.to_s.foreign_key
|
||||||
]
|
]
|
||||||
|
]
|
||||||
when :has_many, :has_one
|
when :has_many, :has_one
|
||||||
if reflection.options[:through]
|
if reflection.options[:through]
|
||||||
through_conditions = through_reflection.options[:conditions] ? "AND #{interpolate_sql(sanitize_sql(through_reflection.options[:conditions]))}" : ''
|
through_conditions = through_reflection.options[:conditions] ? "AND #{interpolate_sql(sanitize_sql(through_reflection.options[:conditions]))}" : ''
|
||||||
|
@ -2154,26 +2163,22 @@ module ActiveRecord
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
" #{join_type} %s ON (%s.%s = %s.%s%s%s%s) " % [
|
["(%s.%s = %s.%s%s%s%s) " % [
|
||||||
table_alias_for(through_reflection.klass.table_name, aliased_join_table_name),
|
|
||||||
connection.quote_table_name(parent.aliased_table_name),
|
connection.quote_table_name(parent.aliased_table_name),
|
||||||
connection.quote_column_name(parent.primary_key),
|
connection.quote_column_name(parent.primary_key),
|
||||||
connection.quote_table_name(aliased_join_table_name),
|
connection.quote_table_name(aliased_join_table_name),
|
||||||
connection.quote_column_name(jt_foreign_key),
|
connection.quote_column_name(jt_foreign_key),
|
||||||
jt_as_extra, jt_source_extra, jt_sti_extra
|
jt_as_extra, jt_source_extra, jt_sti_extra],
|
||||||
] +
|
"(%s.%s = %s.%s%s) " % [
|
||||||
" #{join_type} %s ON (%s.%s = %s.%s%s) " % [
|
|
||||||
table_name_and_alias,
|
|
||||||
connection.quote_table_name(aliased_table_name),
|
connection.quote_table_name(aliased_table_name),
|
||||||
connection.quote_column_name(first_key),
|
connection.quote_column_name(first_key),
|
||||||
connection.quote_table_name(aliased_join_table_name),
|
connection.quote_table_name(aliased_join_table_name),
|
||||||
connection.quote_column_name(second_key),
|
connection.quote_column_name(second_key),
|
||||||
as_extra
|
as_extra]
|
||||||
]
|
]
|
||||||
|
|
||||||
elsif reflection.options[:as] && [:has_many, :has_one].include?(reflection.macro)
|
elsif reflection.options[:as] && [:has_many, :has_one].include?(reflection.macro)
|
||||||
" #{join_type} %s ON %s.%s = %s.%s AND %s.%s = %s" % [
|
"%s.%s = %s.%s AND %s.%s = %s" % [
|
||||||
table_name_and_alias,
|
|
||||||
connection.quote_table_name(aliased_table_name),
|
connection.quote_table_name(aliased_table_name),
|
||||||
"#{reflection.options[:as]}_id",
|
"#{reflection.options[:as]}_id",
|
||||||
connection.quote_table_name(parent.aliased_table_name),
|
connection.quote_table_name(parent.aliased_table_name),
|
||||||
|
@ -2184,8 +2189,7 @@ module ActiveRecord
|
||||||
]
|
]
|
||||||
else
|
else
|
||||||
foreign_key = options[:foreign_key] || reflection.active_record.name.foreign_key
|
foreign_key = options[:foreign_key] || reflection.active_record.name.foreign_key
|
||||||
" #{join_type} %s ON %s.%s = %s.%s " % [
|
"%s.%s = %s.%s " % [
|
||||||
table_name_and_alias,
|
|
||||||
aliased_table_name,
|
aliased_table_name,
|
||||||
foreign_key,
|
foreign_key,
|
||||||
parent.aliased_table_name,
|
parent.aliased_table_name,
|
||||||
|
@ -2193,8 +2197,7 @@ module ActiveRecord
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
when :belongs_to
|
when :belongs_to
|
||||||
" #{join_type} %s ON %s.%s = %s.%s " % [
|
"%s.%s = %s.%s " % [
|
||||||
table_name_and_alias,
|
|
||||||
connection.quote_table_name(aliased_table_name),
|
connection.quote_table_name(aliased_table_name),
|
||||||
reflection.klass.primary_key,
|
reflection.klass.primary_key,
|
||||||
connection.quote_table_name(parent.aliased_table_name),
|
connection.quote_table_name(parent.aliased_table_name),
|
||||||
|
@ -2213,6 +2216,20 @@ module ActiveRecord
|
||||||
join
|
join
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def relation
|
||||||
|
if reflection.macro == :has_and_belongs_to_many
|
||||||
|
[Arel::Table.new(table_alias_for(options[:join_table], aliased_join_table_name)), Arel::Table.new(table_name_and_alias)]
|
||||||
|
elsif reflection.options[:through]
|
||||||
|
[Arel::Table.new(table_alias_for(through_reflection.klass.table_name, aliased_join_table_name)), Arel::Table.new(table_name_and_alias)]
|
||||||
|
else
|
||||||
|
Arel::Table.new(table_name_and_alias)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def join_type
|
||||||
|
Arel::OuterJoin
|
||||||
|
end
|
||||||
|
|
||||||
protected
|
protected
|
||||||
|
|
||||||
def aliased_table_name_for(name, suffix = nil)
|
def aliased_table_name_for(name, suffix = nil)
|
||||||
|
@ -2238,7 +2255,7 @@ module ActiveRecord
|
||||||
end
|
end
|
||||||
|
|
||||||
def table_alias_for(table_name, table_alias)
|
def table_alias_for(table_name, table_alias)
|
||||||
"#{reflection.active_record.connection.quote_table_name(table_name)} #{table_alias if table_name != table_alias}".strip
|
"#{table_name} #{table_alias if table_name != table_alias}".strip
|
||||||
end
|
end
|
||||||
|
|
||||||
def table_name_and_alias
|
def table_name_and_alias
|
||||||
|
@ -2248,11 +2265,6 @@ module ActiveRecord
|
||||||
def interpolate_sql(sql)
|
def interpolate_sql(sql)
|
||||||
instance_eval("%@#{sql.gsub('@', '\@')}@")
|
instance_eval("%@#{sql.gsub('@', '\@')}@")
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
|
||||||
def join_type
|
|
||||||
"LEFT OUTER JOIN"
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -2263,13 +2275,11 @@ module ActiveRecord
|
||||||
end
|
end
|
||||||
|
|
||||||
class InnerJoinAssociation < JoinAssociation
|
class InnerJoinAssociation < JoinAssociation
|
||||||
private
|
|
||||||
def join_type
|
def join_type
|
||||||
"INNER JOIN"
|
Arel::InnerJoin
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1741,8 +1741,7 @@ module ActiveRecord #:nodoc:
|
||||||
if array_of_strings?(merged_joins)
|
if array_of_strings?(merged_joins)
|
||||||
merged_joins.join(' ') + " "
|
merged_joins.join(' ') + " "
|
||||||
else
|
else
|
||||||
join_dependency = ActiveRecord::Associations::ClassMethods::InnerJoinDependency.new(self, merged_joins, nil)
|
build_association_joins(merged_joins)
|
||||||
" #{join_dependency.join_associations.collect { |assoc| assoc.association_join }.join} "
|
|
||||||
end
|
end
|
||||||
when String
|
when String
|
||||||
" #{merged_joins} "
|
" #{merged_joins} "
|
||||||
|
@ -1801,10 +1800,7 @@ module ActiveRecord #:nodoc:
|
||||||
if joins.any?{|j| j.is_a?(String) || array_of_strings?(j) }
|
if joins.any?{|j| j.is_a?(String) || array_of_strings?(j) }
|
||||||
joins = joins.collect do |join|
|
joins = joins.collect do |join|
|
||||||
join = [join] if join.is_a?(String)
|
join = [join] if join.is_a?(String)
|
||||||
unless array_of_strings?(join)
|
join = build_association_joins(join) unless array_of_strings?(join)
|
||||||
join_dependency = ActiveRecord::Associations::ClassMethods::InnerJoinDependency.new(self, join, nil)
|
|
||||||
join = join_dependency.join_associations.collect { |assoc| assoc.association_join }
|
|
||||||
end
|
|
||||||
join
|
join
|
||||||
end
|
end
|
||||||
joins.flatten.map{|j| j.strip}.uniq
|
joins.flatten.map{|j| j.strip}.uniq
|
||||||
|
@ -1813,6 +1809,19 @@ module ActiveRecord #:nodoc:
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def build_association_joins(joins)
|
||||||
|
join_dependency = ActiveRecord::Associations::ClassMethods::InnerJoinDependency.new(self, joins, nil)
|
||||||
|
relation = arel_table
|
||||||
|
join_dependency.join_associations.map { |association|
|
||||||
|
if association.relation.is_a?(Array)
|
||||||
|
[Arel::InnerJoin.new(relation, association.relation.first, association.association_join.first).joins(relation),
|
||||||
|
Arel::InnerJoin.new(relation, association.relation.last, association.association_join.last).joins(relation)].join()
|
||||||
|
else
|
||||||
|
Arel::InnerJoin.new(relation, association.relation, association.association_join).joins(relation)
|
||||||
|
end
|
||||||
|
}.join(" ")
|
||||||
|
end
|
||||||
|
|
||||||
# Object#to_a is deprecated, though it does have the desired behavior
|
# Object#to_a is deprecated, though it does have the desired behavior
|
||||||
def safe_to_array(o)
|
def safe_to_array(o)
|
||||||
case o
|
case o
|
||||||
|
|
|
@ -128,12 +128,6 @@ module ActiveRecord
|
||||||
scope = scope(:find)
|
scope = scope(:find)
|
||||||
|
|
||||||
merged_includes = merge_includes(scope ? scope[:include] : [], options[:include])
|
merged_includes = merge_includes(scope ? scope[:include] : [], options[:include])
|
||||||
joins = construct_join(options[:joins], scope)
|
|
||||||
|
|
||||||
if merged_includes.any?
|
|
||||||
join_dependency = ActiveRecord::Associations::ClassMethods::JoinDependency.new(self, merged_includes, joins)
|
|
||||||
joins << join_dependency.join_associations.collect{|join| join.association_join }.join
|
|
||||||
end
|
|
||||||
|
|
||||||
if operation == "count"
|
if operation == "count"
|
||||||
if merged_includes.any?
|
if merged_includes.any?
|
||||||
|
@ -148,17 +142,12 @@ module ActiveRecord
|
||||||
end
|
end
|
||||||
|
|
||||||
catch :invalid_query do
|
catch :invalid_query do
|
||||||
relation = arel_table((scope && scope[:from]) || options[:from])
|
relation = if merged_includes.any?
|
||||||
|
join_dependency = ActiveRecord::Associations::ClassMethods::JoinDependency.new(self, merged_includes, construct_join(options[:joins], scope))
|
||||||
relation.join(joins)
|
construct_finder_arel_with_included_associations(options, join_dependency)
|
||||||
|
else
|
||||||
relation.where(construct_conditions(options[:conditions], scope))
|
construct_finder_arel(options)
|
||||||
relation.where(construct_arel_limited_ids_condition(options, join_dependency)) if join_dependency && !using_limitable_reflections?(join_dependency.reflections) && ((scope && scope[:limit]) || options[:limit])
|
end
|
||||||
|
|
||||||
relation.order(options[:order])
|
|
||||||
relation.take(options[:limit])
|
|
||||||
relation.skip(options[:offset])
|
|
||||||
|
|
||||||
if options[:group]
|
if options[:group]
|
||||||
return execute_grouped_calculation(operation, column_name, options, relation)
|
return execute_grouped_calculation(operation, column_name, options, relation)
|
||||||
else
|
else
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
module ActiveRecord
|
module ActiveRecord
|
||||||
class Relation
|
class Relation
|
||||||
delegate :delete, :to_sql, :to => :relation
|
delegate :delete, :to_sql, :to => :relation
|
||||||
CLAUSES_METHODS = ["project", "group", "order", "take", "skip"].freeze
|
CLAUSES_METHODS = ["project", "group", "order", "take", "skip", "on"].freeze
|
||||||
attr_reader :relation, :klass
|
attr_reader :relation, :klass
|
||||||
|
|
||||||
def initialize(klass, table = nil)
|
def initialize(klass, table = nil)
|
||||||
|
@ -31,8 +31,14 @@ module ActiveRecord
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
def join(joins)
|
def join(joins, join_type = nil)
|
||||||
@relation = @relation.join(@klass.send(:construct_join, joins, nil)) if !joins.blank?
|
if !joins.blank?
|
||||||
|
if [String, Hash, Array, Symbol].include?(joins.class)
|
||||||
|
@relation = @relation.join(@klass.send(:construct_join, joins, nil))
|
||||||
|
else
|
||||||
|
@relation = @relation.join(joins, join_type)
|
||||||
|
end
|
||||||
|
end
|
||||||
self
|
self
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
2
arel
2
arel
|
@ -1 +1 @@
|
||||||
Subproject commit 808b9e90a38c6c19e109da8eb5f2a264fd780d9a
|
Subproject commit 3d747a56b76ae97645dd265cc75e73e5f7827193
|
Loading…
Reference in a new issue