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.
|
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