Skip to content

Commit d45e569

Browse files
authored
Fix get_parent_state and get_root_state when using mixin=True (#4976)
If a state class inherits directly from a mixin class, find the proper parent class by looking through the `mro()` until a non-mixin class is found.
1 parent 5a45359 commit d45e569

File tree

2 files changed

+41
-1
lines changed

2 files changed

+41
-1
lines changed

reflex/state.py

+8-1
Original file line numberDiff line numberDiff line change
@@ -905,7 +905,14 @@ def get_parent_state(cls) -> Type[BaseState] | None:
905905
]
906906
if len(parent_states) >= 2:
907907
raise ValueError(f"Only one parent state is allowed {parent_states}.")
908-
return parent_states[0] if len(parent_states) == 1 else None
908+
# The first non-mixin state in the mro is our parent.
909+
for base in cls.mro()[1:]:
910+
if base._mixin or not issubclass(base, BaseState):
911+
continue
912+
if base is BaseState:
913+
break
914+
return base
915+
return None # No known parent
909916

910917
@classmethod
911918
@functools.lru_cache()

tests/units/test_state.py

+33
Original file line numberDiff line numberDiff line change
@@ -3424,6 +3424,18 @@ class ChildUsesMixinState(UsesMixinState):
34243424
pass
34253425

34263426

3427+
class ChildMixinState(ChildUsesMixinState, mixin=True):
3428+
"""A mixin state that inherits from a concrete state that uses mixins."""
3429+
3430+
pass
3431+
3432+
3433+
class GrandchildUsesMixinState(ChildMixinState):
3434+
"""A grandchild state that uses the mixin state."""
3435+
3436+
pass
3437+
3438+
34273439
def test_mixin_state() -> None:
34283440
"""Test that a mixin state works correctly."""
34293441
assert "num" in UsesMixinState.base_vars
@@ -3438,6 +3450,9 @@ def test_mixin_state() -> None:
34383450
is not UsesMixinState.backend_vars["_backend_no_default"]
34393451
)
34403452

3453+
assert UsesMixinState.get_parent_state() == State
3454+
assert UsesMixinState.get_root_state() == State
3455+
34413456

34423457
def test_child_mixin_state() -> None:
34433458
"""Test that mixin vars are only applied to the highest state in the hierarchy."""
@@ -3447,6 +3462,24 @@ def test_child_mixin_state() -> None:
34473462
assert "computed" in ChildUsesMixinState.inherited_vars
34483463
assert "computed" not in ChildUsesMixinState.computed_vars
34493464

3465+
assert ChildUsesMixinState.get_parent_state() == UsesMixinState
3466+
assert ChildUsesMixinState.get_root_state() == State
3467+
3468+
3469+
def test_grandchild_mixin_state() -> None:
3470+
"""Test that a mixin can inherit from a concrete state class."""
3471+
assert "num" in GrandchildUsesMixinState.inherited_vars
3472+
assert "num" not in GrandchildUsesMixinState.base_vars
3473+
3474+
assert "computed" in GrandchildUsesMixinState.inherited_vars
3475+
assert "computed" not in GrandchildUsesMixinState.computed_vars
3476+
3477+
assert ChildMixinState.get_parent_state() == ChildUsesMixinState
3478+
assert ChildMixinState.get_root_state() == State
3479+
3480+
assert GrandchildUsesMixinState.get_parent_state() == ChildUsesMixinState
3481+
assert GrandchildUsesMixinState.get_root_state() == State
3482+
34503483

34513484
def test_assignment_to_undeclared_vars():
34523485
"""Test that an attribute error is thrown when undeclared vars are set."""

0 commit comments

Comments
 (0)