24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
|
# File 'activerecord/lib/active_record/associations/join_dependency/join_association.rb', line 24
def join_constraints(foreign_table, foreign_klass, join_type, alias_tracker)
joins = []
chain = []
reflection_chain = reflection.chain
reflection_chain.each_with_index do |reflection, index|
table, terminated = yield reflection, reflection_chain[index..]
@table ||= table
if terminated
foreign_table, foreign_klass = table, reflection.klass
break
end
chain << [reflection, table]
end
chain.reverse_each do |reflection, table|
klass = reflection.klass
scope = reflection.join_scope(table, foreign_table, foreign_klass)
unless scope.references_values.empty?
associations = scope.eager_load_values | scope.includes_values
unless associations.empty?
scope.joins! scope.construct_join_dependency(associations, Arel::Nodes::OuterJoin)
end
end
arel = scope.arel(alias_tracker.aliases)
nodes = arel.constraints.first
if nodes.is_a?(Arel::Nodes::And)
others = nodes.children. do |node|
!Arel.fetch_attribute(node) { |attr| attr.relation.name == table.name }
end
end
joins << join_type.new(table, Arel::Nodes::On.new(nodes))
if others && !others.empty?
joins.concat arel.join_sources
append_constraints(joins.last, others)
end
foreign_table, foreign_klass = table, klass
end
joins
end
|