mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
Add ActiveSupport::DescendantsTracker.
This commit is contained in:
parent
a186431414
commit
8db8c6f4ce
3 changed files with 109 additions and 0 deletions
|
@ -51,6 +51,7 @@ module ActiveSupport
|
||||||
autoload :Concern
|
autoload :Concern
|
||||||
autoload :Configurable
|
autoload :Configurable
|
||||||
autoload :Deprecation
|
autoload :Deprecation
|
||||||
|
autoload :DescendantsTracker
|
||||||
autoload :Gzip
|
autoload :Gzip
|
||||||
autoload :Inflector
|
autoload :Inflector
|
||||||
autoload :JSON
|
autoload :JSON
|
||||||
|
|
36
activesupport/lib/active_support/descendants_tracker.rb
Normal file
36
activesupport/lib/active_support/descendants_tracker.rb
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
require 'active_support/dependencies'
|
||||||
|
|
||||||
|
module ActiveSupport
|
||||||
|
# This module provides an internal implementation to track descendants
|
||||||
|
# which is faster than iterating through ObjectSpace.
|
||||||
|
module DescendantsTracker
|
||||||
|
mattr_accessor :descendants
|
||||||
|
@@descendants = Hash.new { |h, k| h[k] = [] }
|
||||||
|
|
||||||
|
def self.clear
|
||||||
|
@@descendants.each do |klass, descendants|
|
||||||
|
if ActiveSupport::Dependencies.autoloaded?(klass)
|
||||||
|
@@descendants.delete(klass)
|
||||||
|
else
|
||||||
|
descendants.reject! { |v| ActiveSupport::Dependencies.autoloaded?(v) }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def inherited(base)
|
||||||
|
self.direct_descendants << base
|
||||||
|
super
|
||||||
|
end
|
||||||
|
|
||||||
|
def direct_descendants
|
||||||
|
@@descendants[self]
|
||||||
|
end
|
||||||
|
|
||||||
|
def descendants
|
||||||
|
@@descendants[self].inject([]) do |descendants, klass|
|
||||||
|
descendants << klass
|
||||||
|
descendants.concat klass.descendants
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
72
activesupport/test/descendants_tracker_test.rb
Normal file
72
activesupport/test/descendants_tracker_test.rb
Normal file
|
@ -0,0 +1,72 @@
|
||||||
|
require 'abstract_unit'
|
||||||
|
require 'test/unit'
|
||||||
|
require 'active_support'
|
||||||
|
|
||||||
|
class DescendantsTrackerTest < Test::Unit::TestCase
|
||||||
|
class Parent
|
||||||
|
extend ActiveSupport::DescendantsTracker
|
||||||
|
end
|
||||||
|
|
||||||
|
class Child1 < Parent
|
||||||
|
end
|
||||||
|
|
||||||
|
class Child2 < Parent
|
||||||
|
end
|
||||||
|
|
||||||
|
class Grandchild1 < Child1
|
||||||
|
end
|
||||||
|
|
||||||
|
class Grandchild2 < Child1
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_descendants
|
||||||
|
assert_equal [Child1, Grandchild1, Grandchild2, Child2], Parent.descendants
|
||||||
|
assert_equal [Grandchild1, Grandchild2], Child1.descendants
|
||||||
|
assert_equal [], Child2.descendants
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_direct_descendants
|
||||||
|
assert_equal [Child1, Child2], Parent.direct_descendants
|
||||||
|
assert_equal [Grandchild1, Grandchild2], Child1.direct_descendants
|
||||||
|
assert_equal [], Child2.direct_descendants
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_clear_with_autoloaded_parent_children_and_granchildren
|
||||||
|
mark_as_autoloaded Parent, Child1, Child2, Grandchild1, Grandchild2 do
|
||||||
|
ActiveSupport::DescendantsTracker.clear
|
||||||
|
assert_equal Hash.new, ActiveSupport::DescendantsTracker.descendants
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_clear_with_autoloaded_children_and_granchildren
|
||||||
|
mark_as_autoloaded Child1, Grandchild1, Grandchild2 do
|
||||||
|
ActiveSupport::DescendantsTracker.clear
|
||||||
|
assert_equal [Child2], Parent.descendants
|
||||||
|
assert_equal [], Child2.descendants
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def test_clear_with_autoloaded_granchildren
|
||||||
|
mark_as_autoloaded Grandchild1, Grandchild2 do
|
||||||
|
ActiveSupport::DescendantsTracker.clear
|
||||||
|
assert_equal [Child1, Child2], Parent.descendants
|
||||||
|
assert_equal [], Child1.descendants
|
||||||
|
assert_equal [], Child2.descendants
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
protected
|
||||||
|
|
||||||
|
def mark_as_autoloaded(*klasses)
|
||||||
|
old_autoloaded = ActiveSupport::Dependencies.autoloaded_constants.dup
|
||||||
|
ActiveSupport::Dependencies.autoloaded_constants = klasses.map(&:name)
|
||||||
|
|
||||||
|
old_descendants = ActiveSupport::DescendantsTracker.descendants.dup
|
||||||
|
old_descendants.each { |k, v| old_descendants[k] = v.dup }
|
||||||
|
|
||||||
|
yield
|
||||||
|
ensure
|
||||||
|
ActiveSupport::Dependencies.autoloaded_constants = old_autoloaded
|
||||||
|
ActiveSupport::DescendantsTracker.descendants = old_descendants
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue