This repository has been archived by the owner on Nov 12, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 51
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #28 from gssbzn/add-rails-5-2
Add support to rails 5.2
- Loading branch information
Showing
9 changed files
with
199 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,2 @@ | ||
# active_record_5.0_ruby_2/join_association.rb | ||
require 'polyamorous/activerecord_5.1_ruby_2/join_association' | ||
require 'polyamorous/activerecord_5.1_ruby_2/join_association' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,2 @@ | ||
# active_record_5.0_ruby_2/join_dependency.rb | ||
require 'polyamorous/activerecord_5.1_ruby_2/join_dependency' | ||
require 'polyamorous/activerecord_5.1_ruby_2/join_dependency' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
39 changes: 39 additions & 0 deletions
39
lib/polyamorous/activerecord_5.2_ruby_2/join_association.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
# active_record_5.2_ruby_2/join_association.rb | ||
|
||
module Polyamorous | ||
module JoinAssociationExtensions | ||
include SwappingReflectionClass | ||
def self.prepended(base) | ||
base.class_eval { attr_reader :join_type } | ||
end | ||
|
||
def initialize(reflection, children, alias_tracker, polymorphic_class = nil, | ||
join_type = Arel::Nodes::InnerJoin) | ||
@join_type = join_type | ||
if polymorphic_class && ::ActiveRecord::Base > polymorphic_class | ||
swapping_reflection_klass(reflection, polymorphic_class) do |reflection| | ||
super(reflection, children, alias_tracker) | ||
self.reflection.options[:polymorphic] = true | ||
end | ||
else | ||
super(reflection, children, alias_tracker) | ||
end | ||
end | ||
|
||
# Reference: https://github.com/rails/rails/commit/9b15db5 | ||
# NOTE: Not sure we still need it? | ||
# | ||
def ==(other) | ||
base_klass == other.base_klass | ||
end | ||
|
||
def build_constraint(klass, table, key, foreign_table, foreign_key) | ||
if reflection.polymorphic? | ||
super(klass, table, key, foreign_table, foreign_key) | ||
.and(foreign_table[reflection.foreign_type].eq(reflection.klass.name)) | ||
else | ||
super(klass, table, key, foreign_table, foreign_key) | ||
end | ||
end | ||
end | ||
end |
130 changes: 130 additions & 0 deletions
130
lib/polyamorous/activerecord_5.2_ruby_2/join_dependency.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
# active_record_5.2_ruby_2/join_dependency.rb | ||
|
||
module Polyamorous | ||
module JoinDependencyExtensions | ||
# Replaces ActiveRecord::Associations::JoinDependency#build | ||
# | ||
def build(associations, base_klass) | ||
associations.map do |name, right| | ||
if name.is_a? Join | ||
reflection = find_reflection base_klass, name.name | ||
reflection.check_validity! | ||
reflection.check_eager_loadable! if ActiveRecord::VERSION::MAJOR >= 5 | ||
|
||
klass = if reflection.polymorphic? | ||
name.klass || base_klass | ||
else | ||
reflection.klass | ||
end | ||
JoinAssociation.new(reflection, build(right, klass), alias_tracker, name.klass, name.type) | ||
else | ||
reflection = find_reflection base_klass, name | ||
reflection.check_validity! | ||
reflection.check_eager_loadable! if ActiveRecord::VERSION::MAJOR >= 5 | ||
|
||
if reflection.polymorphic? | ||
raise ActiveRecord::EagerLoadPolymorphicError.new(reflection) | ||
end | ||
JoinAssociation.new(reflection, build(right, reflection.klass), alias_tracker) | ||
end | ||
end | ||
end | ||
|
||
def find_join_association_respecting_polymorphism(reflection, parent, klass) | ||
if association = parent.children.find { |j| j.reflection == reflection } | ||
unless reflection.polymorphic? | ||
association | ||
else | ||
association if association.base_klass == klass | ||
end | ||
end | ||
end | ||
|
||
def build_join_association_respecting_polymorphism(reflection, parent, klass) | ||
if reflection.polymorphic? && klass | ||
JoinAssociation.new(reflection, self, alias_tracker, klass) | ||
else | ||
JoinAssociation.new(reflection, self, alias_tracker) | ||
end | ||
end | ||
|
||
# Replaces ActiveRecord::Associations::JoinDependency#join_constraints | ||
# | ||
# This internal method was changed in Rails 5.0 by commit | ||
# https://github.com/rails/rails/commit/e038975 which added | ||
# left_outer_joins (see #make_polyamorous_left_outer_joins below) and added | ||
# passing an additional argument, `join_type`, to #join_constraints. | ||
# | ||
def join_constraints(outer_joins, join_type) | ||
joins = join_root.children.flat_map { |child| | ||
if join_type == Arel::Nodes::OuterJoin | ||
make_polyamorous_left_outer_joins join_root, child | ||
else | ||
make_polyamorous_inner_joins join_root, child | ||
end | ||
} | ||
|
||
joins.concat outer_joins.flat_map { |oj| | ||
if join_root.match? oj.join_root | ||
walk(join_root, oj.join_root) | ||
else | ||
oj.join_root.children.flat_map { |child| | ||
make_outer_joins(oj.join_root, child) | ||
} | ||
end | ||
} | ||
end | ||
|
||
# Replaces ActiveRecord::Associations::JoinDependency#make_left_outer_joins, | ||
# a new method that was added in Rails 5.0 with the following commit: | ||
# https://github.com/rails/rails/commit/e038975 | ||
# | ||
def make_polyamorous_left_outer_joins(parent, child) | ||
tables = child.tables | ||
join_type = Arel::Nodes::OuterJoin | ||
info = make_constraints parent, child, tables, join_type | ||
|
||
[info] + child.children.flat_map { |c| | ||
make_polyamorous_left_outer_joins(child, c) | ||
} | ||
end | ||
|
||
# Replaces ActiveRecord::Associations::JoinDependency#make_inner_joins | ||
# | ||
def make_polyamorous_inner_joins(parent, child) | ||
tables = child.tables | ||
join_type = child.join_type || Arel::Nodes::InnerJoin | ||
info = make_constraints parent, child, tables, join_type | ||
|
||
[info] + child.children.flat_map { |c| | ||
make_polyamorous_inner_joins(child, c) | ||
} | ||
end | ||
|
||
private :make_polyamorous_inner_joins, :make_polyamorous_left_outer_joins | ||
|
||
module ClassMethods | ||
# Prepended before ActiveRecord::Associations::JoinDependency#walk_tree | ||
# | ||
def walk_tree(associations, hash) | ||
case associations | ||
when TreeNode | ||
associations.add_to_tree(hash) | ||
when Hash | ||
associations.each do |k, v| | ||
cache = | ||
if TreeNode === k | ||
k.add_to_tree(hash) | ||
else | ||
hash[k] ||= {} | ||
end | ||
walk_tree(v, cache) | ||
end | ||
else | ||
super(associations, hash) | ||
end | ||
end | ||
end | ||
|
||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters