Bugfix: Fix quart threadedrunner stop graceful shutdown#3824
Open
joschrag wants to merge 3 commits into
Open
Conversation
ThreadedRunner.stop() only had a graceful shutdown branch for FastAPI (keyed on `_uvicorn_server`). A Quart app fell through to the kill-based branch, which injects an async SystemExit via thread.kill() and then calls join() with no timeout. The server thread is parked in a blocking syscall (IOCP on Windows, epoll on POSIX), so the SystemExit is not delivered promptly and join() can hang indefinitely -- on Windows and Linux alike. Add a Quart branch that signals the backend's existing cooperative shutdown switch (backend._ws_shutdown_event) thread-safely on the server's own loop via call_soon_threadsafe, then joins bounded by stop_timeout. Flask/other backends keep the kill path unchanged.
Add a dash.testing regression test that starts a Quart app on ThreadedRunner and asserts stop() returns bounded by stop_timeout (run under a watchdog so a regression fails fast instead of wedging the suite). Verified it fails against the unpatched stop() and passes with the fix.
|
Contributor
|
Thanks for the PR! Our team will take a look and follow up. |
T4rk1n
reviewed
Jun 26, 2026
Comment on lines
+222
to
+234
| quart_shutdown_event = getattr( | ||
| getattr(self._app, "backend", None), "_ws_shutdown_event", None | ||
| ) | ||
| # For FastAPI apps with uvicorn, use graceful shutdown | ||
| if self._app and hasattr(self._app, "_uvicorn_server"): | ||
| server = self._app._uvicorn_server # pylint: disable=protected-access | ||
| server = self._app._uvicorn_server | ||
| server.should_exit = True | ||
| self.thread.join(timeout=self.stop_timeout) # type: ignore[reportOptionalMemberAccess] | ||
| # For Quart apps, signal hypercorn's cooperative shutdown event. Only the | ||
| # main-thread signal handler sets it, but in tests the server runs in a | ||
| # worker thread, so we set it ourselves -- thread-safely, on the server's | ||
| # own loop (the event binds its loop on first await) -- then join bounded. | ||
| elif quart_shutdown_event is not None: |
Contributor
There was a problem hiding this comment.
Could change these checks to be more clear and not dependent on private variables checks, we have the server_type attribute
server_type = getattr(self._app.backend, "server_type", "flask")
if server_type == "fastapi":
...
elif server_type == "quart":
...
else:
...
| from dash.testing.application_runners import ThreadedRunner | ||
|
|
||
|
|
||
| def test_quart_threaded_runner_stop_is_graceful_and_bounded(): |
Contributor
There was a problem hiding this comment.
Think we could make this test generic and parametrize with the three backends?
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Add this suggestion to a batch that can be applied as a single commit.This suggestion is invalid because no changes were made to the code.Suggestions cannot be applied while the pull request is closed.Suggestions cannot be applied while viewing a subset of changes.Only one suggestion per line can be applied in a batch.Add this suggestion to a batch that can be applied as a single commit.Applying suggestions on deleted lines is not supported.You must change the existing code in this line in order to create a valid suggestion.Outdated suggestions cannot be applied.This suggestion has been applied or marked resolved.Suggestions cannot be applied from pending reviews.Suggestions cannot be applied on multi-line comments.Suggestions cannot be applied while the pull request is queued to merge.Suggestion cannot be applied right now. Please check back later.



dash.testing'sThreadedRunner.stop()only has a graceful-shutdown branch for FastAPI — it keys offhasattr(self._app, "_uvicorn_server"). A Quart app has no such attribute, so it falls through to theelsebranch:thread.kill()injects an asynchronousSystemExit, thenthread.join()is called with no timeout.the Quart backend already serves with
serve(shutdown_trigger=...)awaitingbackend._ws_shutdown_event; only the main-thread signal handler ever sets it, and in tests the server runs in a worker thread.stop()now sets that event itself — thread-safely, vialoop.call_soon_threadsafe(event.set)on the server's own loop — then joins bounded bystop_timeout.Closes #3823
Contributor Checklist
optionals
CHANGELOG.md