Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MONGOID-5836 Don't repeat callbacks for embedded grandchildren (backport to 9.0-stable) #5935

Open
wants to merge 2 commits into
base: 9.0-stable
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions .github/workflows/test.yml
Original file line number Diff line number Diff line change
Expand Up @@ -152,6 +152,13 @@ jobs:
uses: actions/checkout@v2
with:
submodules: recursive

# the default python 3.8 doesn't cut it, and causes mongo-orchestration
# to fail in mongodb-labs/drivers-evergreen-tools.
- uses: actions/setup-python@v5
with:
python-version: '3.13'

- id: start-mongodb
name: start mongodb
uses: mongodb-labs/drivers-evergreen-tools@master
Expand Down
11 changes: 6 additions & 5 deletions lib/mongoid/interceptable.rb
Original file line number Diff line number Diff line change
Expand Up @@ -152,9 +152,13 @@ def run_callbacks(kind, with_children: true, skip_if: nil, &block)
# @api private
def _mongoid_run_child_callbacks(kind, children: nil, &block)
if Mongoid::Config.around_callbacks_for_embeds
_mongoid_run_child_callbacks_with_around(kind, children: children, &block)
_mongoid_run_child_callbacks_with_around(kind,
children: children,
&block)
else
_mongoid_run_child_callbacks_without_around(kind, children: children, &block)
_mongoid_run_child_callbacks_without_around(kind,
children: children,
&block)
end
end

Expand Down Expand Up @@ -235,9 +239,6 @@ def _mongoid_run_child_before_callbacks(kind, children: [], callback_list: [])
return false if env.halted
env.value = !env.halted
callback_list << [next_sequence, env]
if (grandchildren = child.send(:cascadable_children, kind))
_mongoid_run_child_before_callbacks(kind, children: grandchildren, callback_list: callback_list)
end
end
callback_list
end
Expand Down
78 changes: 78 additions & 0 deletions spec/mongoid/interceptable_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -389,6 +389,84 @@ class TestClass
end
end
end

context 'with embedded grandchildren' do
IS = InterceptableSpec

context 'when creating' do
let(:registry) { IS::CallbackRegistry.new(only: %i[ before_save ]) }

let(:expected_calls) do
[
# the parent
[ IS::CbParent, :before_save ],

# the immediate child of the parent
[ IS::CbCascadedNode, :before_save ],

# the grandchild of the parent
[ IS::CbCascadedNode, :before_save ],
]
end

let!(:parent) do
parent = IS::CbParent.new(registry)
child = IS::CbCascadedNode.new(registry)
grandchild = IS::CbCascadedNode.new(registry)

child.cb_cascaded_nodes = [ grandchild ]
parent.cb_cascaded_nodes = [ child ]

parent.tap(&:save)
end

it 'should cascade callbacks to grandchildren' do
expect(registry.calls).to be == expected_calls
end
end

context 'when updating' do
let(:registry) { IS::CallbackRegistry.new(only: %i[ before_update ]) }

let(:expected_calls) do
[
# the parent
[ IS::CbParent, :before_update ],

# the immediate child of the parent
[ IS::CbCascadedNode, :before_update ],

# the grandchild of the parent
[ IS::CbCascadedNode, :before_update ],
]
end

let!(:parent) do
parent = IS::CbParent.new(nil)
child = IS::CbCascadedNode.new(nil)
grandchild = IS::CbCascadedNode.new(nil)

child.cb_cascaded_nodes = [ grandchild ]
parent.cb_cascaded_nodes = [ child ]

parent.save

parent.callback_registry = registry
child.callback_registry = registry
grandchild.callback_registry = registry

parent.name = 'updated'
child.name = 'updated'
grandchild.name = 'updated'

parent.tap(&:save)
end

it 'should cascade callbacks to grandchildren' do
expect(registry.calls).to be == expected_calls
end
end
end
end

describe ".before_destroy" do
Expand Down
154 changes: 39 additions & 115 deletions spec/mongoid/interceptable_spec_models.rb
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
# rubocop:todo all
module InterceptableSpec
class CallbackRegistry
def initialize
def initialize(only: [])
@calls = []
@only = only
end

def record_call(cls, cb)
return unless @only.empty? || @only.any? { |pat| pat == cb }
@calls << [cls, cb]
end

Expand All @@ -16,6 +18,8 @@ module CallbackTracking
extend ActiveSupport::Concern

included do
field :name, type: String

%i(
validation save create update
).each do |what|
Expand All @@ -35,199 +39,110 @@ module CallbackTracking
end
end
end
end

class CbHasOneParent
include Mongoid::Document

has_one :child, autosave: true, class_name: "CbHasOneChild", inverse_of: :parent
attr_accessor :callback_registry

def initialize(callback_registry)
def initialize(callback_registry, *args, **kwargs)
@callback_registry = callback_registry
super()
super(*args, **kwargs)
end
end

attr_accessor :callback_registry

module RootInsertable
def insert_as_root
@callback_registry&.record_call(self.class, :insert_into_database)
super
end
end

class CbHasOneParent
include Mongoid::Document
include CallbackTracking
include RootInsertable

has_one :child, autosave: true, class_name: "CbHasOneChild", inverse_of: :parent
end

class CbHasOneChild
include Mongoid::Document
include CallbackTracking

belongs_to :parent, class_name: "CbHasOneParent", inverse_of: :child

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

include CallbackTracking
end

class CbHasManyParent
include Mongoid::Document
include CallbackTracking
include RootInsertable

has_many :children, autosave: true, class_name: "CbHasManyChild", inverse_of: :parent

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

def insert_as_root
@callback_registry&.record_call(self.class, :insert_into_database)
super
end

include CallbackTracking
end

class CbHasManyChild
include Mongoid::Document
include CallbackTracking

belongs_to :parent, class_name: "CbHasManyParent", inverse_of: :children

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

include CallbackTracking
end

class CbEmbedsOneParent
include Mongoid::Document
include CallbackTracking
include RootInsertable

field :name

embeds_one :child, cascade_callbacks: true, class_name: "CbEmbedsOneChild", inverse_of: :parent

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

def insert_as_root
@callback_registry&.record_call(self.class, :insert_into_database)
super
end

include CallbackTracking
end

class CbEmbedsOneChild
include Mongoid::Document
include CallbackTracking

field :age

embedded_in :parent, class_name: "CbEmbedsOneParent", inverse_of: :child

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

include CallbackTracking
end

class CbEmbedsManyParent
include Mongoid::Document
include CallbackTracking
include RootInsertable

embeds_many :children, cascade_callbacks: true, class_name: "CbEmbedsManyChild", inverse_of: :parent

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

def insert_as_root
@callback_registry&.record_call(self.class, :insert_into_database)
super
end

include CallbackTracking
end

class CbEmbedsManyChild
include Mongoid::Document
include CallbackTracking

embedded_in :parent, class_name: "CbEmbedsManyParent", inverse_of: :children

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry

include CallbackTracking
end

class CbParent
include Mongoid::Document

def initialize(callback_registry)
@callback_registry = callback_registry
super()
end

attr_accessor :callback_registry
include CallbackTracking

embeds_many :cb_children
embeds_many :cb_cascaded_children, cascade_callbacks: true

include CallbackTracking
embeds_many :cb_cascaded_nodes, cascade_callbacks: true, as: :parent
end

class CbChild
include Mongoid::Document
include CallbackTracking

embedded_in :cb_parent

def initialize(callback_registry, options)
@callback_registry = callback_registry
super(options)
end

attr_accessor :callback_registry

include CallbackTracking
end

class CbCascadedChild
include Mongoid::Document
include CallbackTracking

embedded_in :cb_parent

def initialize(callback_registry, options)
@callback_registry = callback_registry
super(options)
end

attr_accessor :callback_registry

before_save :test_mongoid_state

include CallbackTracking

private

# Helps test that cascading child callbacks have access to the Mongoid
Expand All @@ -238,6 +153,15 @@ def test_mongoid_state
Mongoid::Threaded.stack('interceptable').push(self)
end
end

class CbCascadedNode
include Mongoid::Document
include CallbackTracking

embedded_in :parent, polymorphic: true

embeds_many :cb_cascaded_nodes, cascade_callbacks: true, as: :parent
end
end

class InterceptableBand
Expand Down
Loading