Skip to content

Commit e5e2661

Browse files
Bug fix for corrupted hook state (#1254)
* Bug fix for corrupted hook state - change hook state to a contextvar --------- Co-authored-by: James Hutchison <[email protected]>
1 parent d5a897e commit e5e2661

File tree

12 files changed

+120
-57
lines changed

12 files changed

+120
-57
lines changed

docs/source/about/changelog.rst

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ Unreleased
5656
**Fixed**
5757

5858
- :pull:`1239` - Fixed a bug where script elements would not render to the DOM as plain text.
59+
- :pull:`1254` - Fixed a bug where ``RuntimeError("Hook stack is in an invalid state")`` errors would be provided when using a webserver that reuses threads.
5960

6061
v1.1.0
6162
------

pyproject.toml

-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,6 @@ commands = [
8080
]
8181
artifacts = []
8282

83-
8483
#############################
8584
# >>> Hatch Test Runner <<< #
8685
#############################

src/reactpy/core/_life_cycle_hook.py

+43-17
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,16 @@
11
from __future__ import annotations
22

33
import logging
4+
import sys
45
from asyncio import Event, Task, create_task, gather
6+
from contextvars import ContextVar, Token
57
from typing import Any, Callable, Protocol, TypeVar
68

79
from anyio import Semaphore
810

911
from reactpy.core._thread_local import ThreadLocal
1012
from reactpy.types import ComponentType, Context, ContextProviderType
13+
from reactpy.utils import Singleton
1114

1215
T = TypeVar("T")
1316

@@ -18,16 +21,39 @@ async def __call__(self, stop: Event) -> None: ...
1821

1922
logger = logging.getLogger(__name__)
2023

21-
_HOOK_STATE: ThreadLocal[list[LifeCycleHook]] = ThreadLocal(list)
2224

25+
class _HookStack(Singleton): # pragma: no cover
26+
"""A singleton object which manages the current component tree's hooks.
27+
Life cycle hooks can be stored in a thread local or context variable depending
28+
on the platform."""
2329

24-
def current_hook() -> LifeCycleHook:
25-
"""Get the current :class:`LifeCycleHook`"""
26-
hook_stack = _HOOK_STATE.get()
27-
if not hook_stack:
28-
msg = "No life cycle hook is active. Are you rendering in a layout?"
29-
raise RuntimeError(msg)
30-
return hook_stack[-1]
30+
_state: ThreadLocal[list[LifeCycleHook]] | ContextVar[list[LifeCycleHook]] = (
31+
ThreadLocal(list) if sys.platform == "emscripten" else ContextVar("hook_state")
32+
)
33+
34+
def get(self) -> list[LifeCycleHook]:
35+
return self._state.get()
36+
37+
def initialize(self) -> Token[list[LifeCycleHook]] | None:
38+
return None if isinstance(self._state, ThreadLocal) else self._state.set([])
39+
40+
def reset(self, token: Token[list[LifeCycleHook]] | None) -> None:
41+
if isinstance(self._state, ThreadLocal):
42+
self._state.get().clear()
43+
elif token:
44+
self._state.reset(token)
45+
else:
46+
raise RuntimeError("Hook stack is an ContextVar but no token was provided")
47+
48+
def current_hook(self) -> LifeCycleHook:
49+
hook_stack = self.get()
50+
if not hook_stack:
51+
msg = "No life cycle hook is active. Are you rendering in a layout?"
52+
raise RuntimeError(msg)
53+
return hook_stack[-1]
54+
55+
56+
HOOK_STACK = _HookStack()
3157

3258

3359
class LifeCycleHook:
@@ -37,7 +63,7 @@ class LifeCycleHook:
3763
a component is first rendered until it is removed from the layout. The life cycle
3864
is ultimately driven by the layout itself, but components can "hook" into those
3965
events to perform actions. Components gain access to their own life cycle hook
40-
by calling :func:`current_hook`. They can then perform actions such as:
66+
by calling :func:`HOOK_STACK.current_hook`. They can then perform actions such as:
4167
4268
1. Adding state via :meth:`use_state`
4369
2. Adding effects via :meth:`add_effect`
@@ -57,7 +83,7 @@ class LifeCycleHook:
5783
.. testcode::
5884
5985
from reactpy.core._life_cycle_hook import LifeCycleHook
60-
from reactpy.core.hooks import current_hook
86+
from reactpy.core.hooks import HOOK_STACK
6187
6288
# this function will come from a layout implementation
6389
schedule_render = lambda: ...
@@ -75,15 +101,15 @@ class LifeCycleHook:
75101
...
76102
77103
# the component may access the current hook
78-
assert current_hook() is hook
104+
assert HOOK_STACK.current_hook() is hook
79105
80106
# and save state or add effects
81-
current_hook().use_state(lambda: ...)
107+
HOOK_STACK.current_hook().use_state(lambda: ...)
82108
83109
async def my_effect(stop_event):
84110
...
85111
86-
current_hook().add_effect(my_effect)
112+
HOOK_STACK.current_hook().add_effect(my_effect)
87113
finally:
88114
await hook.affect_component_did_render()
89115
@@ -130,7 +156,7 @@ def __init__(
130156
self._scheduled_render = False
131157
self._rendered_atleast_once = False
132158
self._current_state_index = 0
133-
self._state: tuple[Any, ...] = ()
159+
self._state: list = []
134160
self._effect_funcs: list[EffectFunc] = []
135161
self._effect_tasks: list[Task[None]] = []
136162
self._effect_stops: list[Event] = []
@@ -157,7 +183,7 @@ def use_state(self, function: Callable[[], T]) -> T:
157183
if not self._rendered_atleast_once:
158184
# since we're not initialized yet we're just appending state
159185
result = function()
160-
self._state += (result,)
186+
self._state.append(result)
161187
else:
162188
# once finalized we iterate over each succesively used piece of state
163189
result = self._state[self._current_state_index]
@@ -232,13 +258,13 @@ def set_current(self) -> None:
232258
This method is called by a layout before entering the render method
233259
of this hook's associated component.
234260
"""
235-
hook_stack = _HOOK_STATE.get()
261+
hook_stack = HOOK_STACK.get()
236262
if hook_stack:
237263
parent = hook_stack[-1]
238264
self._context_providers.update(parent._context_providers)
239265
hook_stack.append(self)
240266

241267
def unset_current(self) -> None:
242268
"""Unset this hook as the active hook in this thread"""
243-
if _HOOK_STATE.get().pop() is not self:
269+
if HOOK_STACK.get().pop() is not self:
244270
raise RuntimeError("Hook stack is in an invalid state") # nocov

src/reactpy/core/_thread_local.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,10 @@
55
_StateType = TypeVar("_StateType")
66

77

8-
class ThreadLocal(Generic[_StateType]):
9-
"""Utility for managing per-thread state information"""
8+
class ThreadLocal(Generic[_StateType]): # pragma: no cover
9+
"""Utility for managing per-thread state information. This is only used in
10+
environments where ContextVars are not available, such as the `pyodide`
11+
executor."""
1012

1113
def __init__(self, default: Callable[[], _StateType]):
1214
self._default = default

src/reactpy/core/hooks.py

+8-8
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919
from typing_extensions import TypeAlias
2020

2121
from reactpy.config import REACTPY_DEBUG
22-
from reactpy.core._life_cycle_hook import current_hook
22+
from reactpy.core._life_cycle_hook import HOOK_STACK
2323
from reactpy.types import Connection, Context, Key, Location, State, VdomDict
2424
from reactpy.utils import Ref
2525

@@ -83,7 +83,7 @@ def __init__(
8383
else:
8484
self.value = initial_value
8585

86-
hook = current_hook()
86+
hook = HOOK_STACK.current_hook()
8787

8888
def dispatch(new: _Type | Callable[[_Type], _Type]) -> None:
8989
next_value = new(self.value) if callable(new) else new # type: ignore
@@ -139,7 +139,7 @@ def use_effect(
139139
Returns:
140140
If not function is provided, a decorator. Otherwise ``None``.
141141
"""
142-
hook = current_hook()
142+
hook = HOOK_STACK.current_hook()
143143
dependencies = _try_to_infer_closure_values(function, dependencies)
144144
memoize = use_memo(dependencies=dependencies)
145145
cleanup_func: Ref[_EffectCleanFunc | None] = use_ref(None)
@@ -212,7 +212,7 @@ def use_async_effect(
212212
Returns:
213213
If not function is provided, a decorator. Otherwise ``None``.
214214
"""
215-
hook = current_hook()
215+
hook = HOOK_STACK.current_hook()
216216
dependencies = _try_to_infer_closure_values(function, dependencies)
217217
memoize = use_memo(dependencies=dependencies)
218218
cleanup_func: Ref[_EffectCleanFunc | None] = use_ref(None)
@@ -280,7 +280,7 @@ def use_debug_value(
280280

281281
if REACTPY_DEBUG.current and old.current != new:
282282
old.current = new
283-
logger.debug(f"{current_hook().component} {new}")
283+
logger.debug(f"{HOOK_STACK.current_hook().component} {new}")
284284

285285

286286
def create_context(default_value: _Type) -> Context[_Type]:
@@ -308,7 +308,7 @@ def use_context(context: Context[_Type]) -> _Type:
308308
309309
See the full :ref:`Use Context` docs for more information.
310310
"""
311-
hook = current_hook()
311+
hook = HOOK_STACK.current_hook()
312312
provider = hook.get_context_provider(context)
313313

314314
if provider is None:
@@ -361,7 +361,7 @@ def __init__(
361361
self.value = value
362362

363363
def render(self) -> VdomDict:
364-
current_hook().set_context_provider(self)
364+
HOOK_STACK.current_hook().set_context_provider(self)
365365
return {"tagName": "", "children": self.children}
366366

367367
def __repr__(self) -> str:
@@ -554,7 +554,7 @@ def use_ref(initial_value: _Type) -> Ref[_Type]:
554554

555555

556556
def _use_const(function: Callable[[], _Type]) -> _Type:
557-
return current_hook().use_state(function)
557+
return HOOK_STACK.current_hook().use_state(function)
558558

559559

560560
def _try_to_infer_closure_values(

src/reactpy/core/serve.py

+16-11
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
from anyio.abc import TaskGroup
1010

1111
from reactpy.config import REACTPY_DEBUG
12+
from reactpy.core._life_cycle_hook import HOOK_STACK
1213
from reactpy.types import LayoutEventMessage, LayoutType, LayoutUpdateMessage
1314

1415
logger = getLogger(__name__)
@@ -63,18 +64,22 @@ async def _single_outgoing_loop(
6364
send: SendCoroutine,
6465
) -> None:
6566
while True:
66-
update = await layout.render()
67+
token = HOOK_STACK.initialize()
6768
try:
68-
await send(update)
69-
except Exception: # nocov
70-
if not REACTPY_DEBUG.current:
71-
msg = (
72-
"Failed to send update. More info may be available "
73-
"if you enabling debug mode by setting "
74-
"`reactpy.config.REACTPY_DEBUG.current = True`."
75-
)
76-
logger.error(msg)
77-
raise
69+
update = await layout.render()
70+
try:
71+
await send(update)
72+
except Exception: # nocov
73+
if not REACTPY_DEBUG.current:
74+
msg = (
75+
"Failed to send update. More info may be available "
76+
"if you enabling debug mode by setting "
77+
"`reactpy.config.REACTPY_DEBUG.current = True`."
78+
)
79+
logger.error(msg)
80+
raise
81+
finally:
82+
HOOK_STACK.reset(token)
7883

7984

8085
async def _single_incoming_loop(

src/reactpy/pyscript/utils.py

+14-11
Original file line numberDiff line numberDiff line change
@@ -145,6 +145,8 @@ def extend_pyscript_config(
145145

146146

147147
def reactpy_version_string() -> str: # pragma: no cover
148+
from reactpy.testing.common import GITHUB_ACTIONS
149+
148150
local_version = reactpy.__version__
149151

150152
# Get a list of all versions via `pip index versions`
@@ -170,14 +172,16 @@ def reactpy_version_string() -> str: # pragma: no cover
170172
symbol_postion = line.index(latest_version_symbol)
171173
latest_version = line[symbol_postion + len(latest_version_symbol) :].strip()
172174

173-
# Return early if local version of ReactPy is available on PyPi
174-
if local_version in known_versions:
175+
# Return early if the version is available on PyPi and we're not in a CI environment
176+
if local_version in known_versions and not GITHUB_ACTIONS:
175177
return f"reactpy=={local_version}"
176178

177-
# Begin determining an alternative method of installing ReactPy
178-
179-
if not latest_version:
180-
_logger.warning("Failed to determine the latest version of ReactPy on PyPi. ")
179+
# We are now determining an alternative method of installing ReactPy for PyScript
180+
if not GITHUB_ACTIONS:
181+
_logger.warning(
182+
"Your current version of ReactPy isn't available on PyPi. Since a packaged version "
183+
"of ReactPy is required for PyScript, we are attempting to find an alternative method..."
184+
)
181185

182186
# Build a local wheel for ReactPy, if needed
183187
dist_dir = Path(reactpy.__file__).parent.parent.parent / "dist"
@@ -202,19 +206,18 @@ def reactpy_version_string() -> str: # pragma: no cover
202206
)
203207
return f"reactpy=={latest_version}"
204208
_logger.error(
205-
"Failed to build a local wheel for ReactPy and could not determine the latest version on PyPi. "
209+
"Failed to build a local wheel for ReactPy, and could not determine the latest version on PyPi. "
206210
"PyScript functionality may not work as expected.",
207211
)
208212
return f"reactpy=={local_version}"
209213

210-
# Move the local file to the web modules directory, if needed
214+
# Move the local wheel file to the web modules directory, if needed
211215
wheel_file = Path(wheel_glob[0])
212216
new_path = REACTPY_WEB_MODULES_DIR.current / wheel_file.name
213217
if not new_path.exists():
214218
_logger.warning(
215-
"'reactpy==%s' is not available on PyPi. "
216-
"PyScript will utilize a local wheel of ReactPy instead.",
217-
local_version,
219+
"PyScript will utilize local wheel '%s'.",
220+
wheel_file.name,
218221
)
219222
shutil.copy(wheel_file, new_path)
220223
return f"{REACTPY_PATH_PREFIX.current}modules/{wheel_file.name}"

src/reactpy/testing/common.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
from typing_extensions import ParamSpec
1515

1616
from reactpy.config import REACTPY_TESTS_DEFAULT_TIMEOUT, REACTPY_WEB_MODULES_DIR
17-
from reactpy.core._life_cycle_hook import LifeCycleHook, current_hook
17+
from reactpy.core._life_cycle_hook import HOOK_STACK, LifeCycleHook
1818
from reactpy.core.events import EventHandler, to_event_handler_function
1919

2020

@@ -153,7 +153,7 @@ def wrapper(*args: Any, **kwargs: Any) -> Any:
153153
if self is None:
154154
raise RuntimeError("Hook catcher has been garbage collected")
155155

156-
hook = current_hook()
156+
hook = HOOK_STACK.current_hook()
157157
if self.index_by_kwarg is not None:
158158
self.index[kwargs[self.index_by_kwarg]] = hook
159159
self.latest = hook

src/reactpy/utils.py

+10
Original file line numberDiff line numberDiff line change
@@ -334,3 +334,13 @@ def import_dotted_path(dotted_path: str) -> Any:
334334
except AttributeError as error:
335335
msg = f'ReactPy failed to import "{component_name}" from "{module_name}"'
336336
raise AttributeError(msg) from error
337+
338+
339+
class Singleton:
340+
"""A class that only allows one instance to be created."""
341+
342+
def __new__(cls, *args, **kw):
343+
if not hasattr(cls, "_instance"):
344+
orig = super()
345+
cls._instance = orig.__new__(cls, *args, **kw)
346+
return cls._instance

tests/conftest.py

+17
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,23 @@ def rebuild():
4444
subprocess.run(["hatch", "build", "-t", "wheel"], check=True) # noqa: S607, S603
4545

4646

47+
@pytest.fixture(autouse=True, scope="function")
48+
def create_hook_state():
49+
"""This fixture is a bug fix related to `pytest_asyncio`.
50+
51+
Usually the hook stack is created automatically within the display fixture, but context
52+
variables aren't retained within `pytest_asyncio` async fixtures. As a workaround,
53+
this fixture ensures that the hook stack is created before each test is run.
54+
55+
Ref: https://github.com/pytest-dev/pytest-asyncio/issues/127
56+
"""
57+
from reactpy.core._life_cycle_hook import HOOK_STACK
58+
59+
token = HOOK_STACK.initialize()
60+
yield token
61+
HOOK_STACK.reset(token)
62+
63+
4764
@pytest.fixture
4865
async def display(server, page):
4966
async with DisplayFixture(server, page) as display:

0 commit comments

Comments
 (0)