Skip to content

Commit 386503f

Browse files
committedJan 30, 2025··
Support cache_classes: true descendants filtering
When cache_classes is true, we can no longer rely on the Active Support dependencies tracker, so we use our own separate tracker to filter out destroyed with_model models from Class#descendants and Class#subclasses. Fixes #35
1 parent dd38e8d commit 386503f

File tree

6 files changed

+144
-37
lines changed

6 files changed

+144
-37
lines changed
 

‎.rubocop.yml

+3
Original file line numberDiff line numberDiff line change
@@ -29,5 +29,8 @@ RSpec/Be:
2929
RSpec/BeforeAfterAll:
3030
Enabled: false
3131

32+
RSpec/ExpectInHook:
33+
Enabled: false
34+
3235
Style/Documentation:
3336
Enabled: false

‎Gemfile

+1
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@ end
1717
gem "bigdecimal"
1818
gem "bundler"
1919
gem "debug"
20+
gem "logger"
2021
gem "minitest"
2122
gem "mutex_m"
2223
gem "rake"

‎lib/with_model/descendants_tracker.rb

+87
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
require "active_support/descendants_tracker"
2+
3+
module WithModel
4+
# Based on https://github.com/rails/rails/blob/491afff27e2dd3d5f301b478b9a43d3c31709af8/activesupport/lib/active_support/descendants_tracker.rb
5+
module DescendantsTracker
6+
if RUBY_ENGINE == "ruby"
7+
# On MRI `ObjectSpace::WeakMap` keys are weak references.
8+
# So we can simply use WeakMap as a `Set`.
9+
class WeakSet < ObjectSpace::WeakMap # :nodoc:
10+
alias_method :to_a, :keys
11+
12+
def <<(object)
13+
self[object] = true
14+
end
15+
end
16+
else
17+
# On TruffleRuby `ObjectSpace::WeakMap` keys are strong references.
18+
# So we use `object_id` as a key and the actual object as a value.
19+
#
20+
# JRuby for now doesn't have Class#descendant, but when it will, it will likely
21+
# have the same WeakMap semantic than Truffle so we future proof this as much as possible.
22+
class WeakSet # :nodoc:
23+
def initialize
24+
@map = ObjectSpace::WeakMap.new
25+
end
26+
27+
def [](object)
28+
@map.key?(object.object_id)
29+
end
30+
alias_method :include?, :[]
31+
32+
def []=(object, _present)
33+
@map[object.object_id] = object
34+
end
35+
36+
def to_a
37+
@map.values
38+
end
39+
40+
def <<(object)
41+
self[object] = true
42+
end
43+
end
44+
end
45+
@excluded_descendants = WeakSet.new
46+
47+
class << self
48+
def clear(classes) # :nodoc:
49+
classes.each do |klass|
50+
@excluded_descendants << klass
51+
klass.descendants.each do |descendant|
52+
@excluded_descendants << descendant
53+
end
54+
end
55+
end
56+
57+
def reject!(classes) # :nodoc:
58+
if @excluded_descendants
59+
classes.reject! { |d| @excluded_descendants.include?(d) }
60+
end
61+
classes
62+
end
63+
end
64+
65+
module ReloadedClassesFiltering # :nodoc:
66+
def subclasses
67+
WithModel::DescendantsTracker.reject!(super)
68+
end
69+
70+
def descendants
71+
WithModel::DescendantsTracker.reject!(super)
72+
end
73+
end
74+
end
75+
end
76+
77+
class Class
78+
prepend WithModel::DescendantsTracker::ReloadedClassesFiltering
79+
end
80+
81+
module ActiveSupport
82+
module DescendantsTracker
83+
class << self
84+
attr_reader :clear_disabled
85+
end
86+
end
87+
end

‎lib/with_model/model.rb

+5-9
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
11
# frozen_string_literal: true
22

3+
require "logger"
34
require "active_record"
45
require "active_support/core_ext/string/inflections"
56
require "English"
67
require "with_model/constant_stubber"
8+
require "with_model/descendants_tracker"
79
require "with_model/methods"
810
require "with_model/table"
911

@@ -38,6 +40,7 @@ def destroy
3840
cleanup_descendants_tracking
3941
reset_dependencies_cache
4042
table.destroy
43+
WithModel::DescendantsTracker.clear([@model])
4144
@model = nil
4245
end
4346

@@ -54,15 +57,8 @@ def setup_model
5457
end
5558

5659
def cleanup_descendants_tracking
57-
if defined?(ActiveSupport::DescendantsTracker)
58-
if ActiveSupport::VERSION::MAJOR >= 7
59-
ActiveSupport::DescendantsTracker.clear([@model])
60-
else
61-
ActiveSupport::DescendantsTracker.class_variable_get(:@@direct_descendants).delete(ActiveRecord::Base)
62-
end
63-
elsif @model.superclass.respond_to?(:direct_descendants)
64-
@model.superclass.direct_descendants.delete(@model)
65-
end
60+
ActiveSupport::DescendantsTracker.clear([@model]) \
61+
unless ActiveSupport::DescendantsTracker.clear_disabled
6662
end
6763

6864
def reset_dependencies_cache

‎spec/descendants_tracking_spec.rb

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
# frozen_string_literal: true
2+
3+
require "spec_helper"
4+
5+
describe "Descendants tracking" do # rubocop:disable RSpec/DescribeClass
6+
with_model :BlogPost do
7+
model do
8+
def self.inspect
9+
"BlogPost class #{object_id}"
10+
end
11+
end
12+
end
13+
14+
def blog_post_classes
15+
ActiveRecord::Base.descendants.select do |c|
16+
c.table_name == BlogPost.table_name
17+
end
18+
end
19+
20+
shared_examples "clearing descendants between test runs" do
21+
it "includes the correct model class in descendants on the first test run" do
22+
expect(blog_post_classes).to eq [BlogPost]
23+
end
24+
25+
it "includes the correct model class in descendants on the second test run" do
26+
expect(blog_post_classes).to eq [BlogPost]
27+
end
28+
end
29+
30+
context "with ActiveSupport::DescendantsTracker (cache_classes: true)" do
31+
before do
32+
expect(ActiveSupport::DescendantsTracker.clear_disabled).to be_falsey
33+
expect { ActiveSupport::DescendantsTracker.clear([]) }.not_to raise_exception
34+
end
35+
36+
include_examples "clearing descendants between test runs"
37+
end
38+
39+
context "without ActiveSupport::DescendantsTracker (cache_classes: false)" do
40+
before do
41+
ActiveSupport::DescendantsTracker.disable_clear!
42+
expect(ActiveSupport::DescendantsTracker.clear_disabled).to be_truthy
43+
expect { ActiveSupport::DescendantsTracker.clear([]) }.to raise_exception(RuntimeError)
44+
end
45+
46+
include_examples "clearing descendants between test runs"
47+
end
48+
end

‎spec/with_model_spec.rb

-28
Original file line numberDiff line numberDiff line change
@@ -320,30 +320,6 @@ def my_method
320320
end
321321
end
322322

323-
context "with ActiveSupport::DescendantsTracker" do
324-
with_model :BlogPost do
325-
model do
326-
def self.inspect
327-
"BlogPost class #{object_id}"
328-
end
329-
end
330-
end
331-
332-
def blog_post_classes
333-
ActiveRecord::Base.descendants.select do |c|
334-
c.table_name == BlogPost.table_name
335-
end
336-
end
337-
338-
it "includes the correct model class in descendants on the first test run" do
339-
expect(blog_post_classes).to eq [BlogPost]
340-
end
341-
342-
it "includes the correct model class in descendants on the second test run" do
343-
expect(blog_post_classes).to eq [BlogPost]
344-
end
345-
end
346-
347323
context "with_model can be run within RSpec :all hook" do
348324
with_model :BlogPost, scope: :all do
349325
table do |t|
@@ -396,10 +372,6 @@ class ApplicationRecordInDifferentDatabase < ActiveRecord::Base # standard:disab
396372
establish_connection(ActiveRecord::Base.connection_pool.db_config.configuration_hash)
397373
end
398374

399-
after(:all) do
400-
Object.__send__(:remove_const, "ApplicationRecordInDifferentDatabase")
401-
end
402-
403375
with_model :BlogPost, superclass: ApplicationRecordInDifferentDatabase do
404376
table do |t|
405377
t.string "title"

0 commit comments

Comments
 (0)