This blog post will give you a comprehensive understanding how FastAPI works when the client disconnects.
Info
If you want to try the code I'll be presenting, you'll need to have some packages installed:
pip install httpx fastapi uvicorn httptools uvloop
httpx is going to be used as the HTTP client.
uvicorn is the ASGI server, and httptools and uvloop are packages used by uvicorn.
- You know about
fastapi... But it's an ASGI web framework.
Let's create a FastAPI application with an endpoint that takes a long time to finish its processing.
The following endpoint just sleeps for 10 seconds, and sends a response with a 204 status code:
main.pyfrom fastapi import FastAPI, Request
app = FastAPI()
@app.get("/", status_code=204)
async def home() -> None:
await anyio.sleep(10)
You can run this application with any ASGI server. We'll use Uvicorn because I maintain it,
and it's the most popular ASGI server.
Let's run it with uvicorn main:app --reload --log-level=trace.
Tip
The --log-level=trace is used to see ASGI messages, and changes in the connection status.
Let's call this endpoint with an HTTP client, and disconnect before the server is able to send the response back.
client.pyimport anyio
import httpx
async def main() -> None:
async with httpx.AsyncClient(base_url="http://localhost:8000") as client:
with anyio.fail_after(1):
await client.get("/")
if __name__ == "__main__":
anyio.run(main)
If you run the above with python client.py, you'll see the following logs on the server side:
TRACE: 127.0.0.1:50953 - HTTP connection made
TRACE: 127.0.0.1:50953 - ASGI [2] Started scope={'type': 'http', 'asgi': {'version': '3.0', 'spec_version': '2.4'}, 'http_version': '1.1', 'server': ('127.0.0.1', 8000), 'client': ('127.0.0.1', 50953), 'scheme': 'http', 'root_path': '', 'headers': '<...>', 'state': {}, 'method': 'GET', 'path': '/', 'raw_path': b'/', 'query_string': b''}
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.request', 'body': '<0 bytes>', 'more_body': False}
TRACE: 127.0.0.1:50953 - HTTP connection lost
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.disconnect'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.start', 'status': 204, 'headers': '<...>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.body', 'body': '<0 bytes>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Completed
This may look a bit too complex, so let's go step by step here...
The first two lines show that the client connected to the server, and that the ASGI application was called.
TRACE: 127.0.0.1:50953 - HTTP connection made
TRACE: 127.0.0.1:50953 - ASGI [2] Started scope={'type': 'http', 'asgi': {'version': '3.0', 'spec_version': '2.4'}, 'http_version': '1.1', 'server': ('127.0.0.1', 8000), 'client': ('127.0.0.1', 50953), 'scheme': 'http', 'root_path': '', 'headers': '<...>', 'state': {}, 'method': 'GET', 'path': '/', 'raw_path': b'/', 'query_string': b''}
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.request', 'body': '<0 bytes>', 'more_body': False}
TRACE: 127.0.0.1:50953 - HTTP connection lost
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.disconnect'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.start', 'status': 204, 'headers': '<...>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.body', 'body': '<0 bytes>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Completed
Info
The ASGI specification determines how the server and the web framework are going to interact to process the client's request.
When the server reads the body, it will send a http.request ASGI message to the ASGI application (in this case, FastAPI):
TRACE: 127.0.0.1:50953 - HTTP connection made
TRACE: 127.0.0.1:50953 - ASGI [2] Started scope={'type': 'http', 'asgi': {'version': '3.0', 'spec_version': '2.4'}, 'http_version': '1.1', 'server': ('127.0.0.1', 8000), 'client': ('127.0.0.1', 50953), 'scheme': 'http', 'root_path': '', 'headers': '<...>', 'state': {}, 'method': 'GET', 'path': '/', 'raw_path': b'/', 'query_string': b''}
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.request', 'body': '<0 bytes>', 'more_body': False}
TRACE: 127.0.0.1:50953 - HTTP connection lost
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.disconnect'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.start', 'status': 204, 'headers': '<...>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.body', 'body': '<0 bytes>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Completed
Then... Before the application finishes the execution, the client disconnected!
TRACE: 127.0.0.1:50953 - HTTP connection made
TRACE: 127.0.0.1:50953 - ASGI [2] Started scope={'type': 'http', 'asgi': {'version': '3.0', 'spec_version': '2.4'}, 'http_version': '1.1', 'server': ('127.0.0.1', 8000), 'client': ('127.0.0.1', 50953), 'scheme': 'http', 'root_path': '', 'headers': '<...>', 'state': {}, 'method': 'GET', 'path': '/', 'raw_path': b'/', 'query_string': b''}
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.request', 'body': '<0 bytes>', 'more_body': False}
TRACE: 127.0.0.1:50953 - HTTP connection lost
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.disconnect'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.start', 'status': 204, 'headers': '<...>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.body', 'body': '<0 bytes>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Completed
The server notices, and sends a http.disconnect message to the application.
TRACE: 127.0.0.1:50953 - HTTP connection made
TRACE: 127.0.0.1:50953 - ASGI [2] Started scope={'type': 'http', 'asgi': {'version': '3.0', 'spec_version': '2.4'}, 'http_version': '1.1', 'server': ('127.0.0.1', 8000), 'client': ('127.0.0.1', 50953), 'scheme': 'http', 'root_path': '', 'headers': '<...>', 'state': {}, 'method': 'GET', 'path': '/', 'raw_path': b'/', 'query_string': b''}
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.request', 'body': '<0 bytes>', 'more_body': False}
TRACE: 127.0.0.1:50953 - HTTP connection lost
TRACE: 127.0.0.1:50953 - ASGI [2] Receive {'type': 'http.disconnect'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.start', 'status': 204, 'headers': '<...>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Send {'type': 'http.response.body', 'body': '<0 bytes>'}
TRACE: 127.0.0.1:50953 - ASGI [2] Completed
Ok... Cool! Now we finally arrive to the important point of this blog post!
The client disconnected, the ASGI server communicated it to the application, but... Did the application stop?
The answer is: NO. Although the application is able to check the http.disconnect, Starlette only does it for
StreamingResponses, but it doesn't do it for all the other response classes by default.
I said above that the application is able to check, but it doesn't do it by default.
I'll teach you how you can check when a client is disconnected.
Note
The current way to check client disconnection is a bit complicated. But... We are working on new mechanism that
will be introduce in a future release with the goal of simplifying this flow.
Follow me on LinkedIn and Twitter, and sponsor me on GitHub for more information. 
Let's complicate a bit our application... I'll explain everything, don't worry.
main.pyfrom fastapi import FastAPI, Request
import anyio
import httpx
app = FastAPI()
async def disconnected(request: Request, cancel_scope: anyio.CancelScope) -> None:
while True:
message = await request.receive()
if message["type"] == "http.disconnect":
cancel_scope.cancel()
break
@app.get("/", status_code=204)
async def home(request: Request) -> None:
async with anyio.create_task_group() as tg:
tg.start_soon(disconnected, request, tg.cancel_scope)
await anyio.sleep(10)
Cool! But... What have I done? 
We've created the disconnected() task, that will await on request.receive(), and cancel the
anyio.TaskGroup when the message "http.disconnect" is found.
Is the logic above 100% correct? When I was writing this article, I actually thought it was, but then I remembered
that I forgot a small detail... What if the client doesn't disconnect?
Well... Then the task runs forever. So yeah, we need to actually stop the TaskClient when either:
1. the client disconnects or...
2. the endpoint finishes to process its logic, and is ready to send the response.
The right logic is a bit more complex, but would be...
main.pyfrom typing import Any, Awaitable
import anyio
import httpx
from fastapi import FastAPI, Request
app = FastAPI()
async def disconnected(request: Request) -> None:
while True:
message = await request.receive()
if message["type"] == "http.disconnect":
break # (1)!
async def wrap(call: Awaitable[Any], cancel_scope: anyio.CancelScope):
await call
cancel_scope.cancel() # (2)!
@app.get("/", status_code=204)
async def home(request: Request) -> None:
async with anyio.create_task_group() as tg:
tg.start_soon(wrap, disconnected(request), tg.cancel_scope)
await wrap(anyio.sleep(5), tg.cancel_scope)
- We removed the
cancel_scope.cancel() from here.
- We added the
cancel_scope.cancel() in the wrap() function.
Now, we achieving our goal. You can try calling the python client.py, and you'll see it will work.
You can also call the endpoint with a simple curl http://localhost:8000/ (without disconnecting).
After seeing all of the above, you may have some questions...
I don't recommend to do it in most of cases. I'm just presenting a behavior, and explaining how to overcome it
with the current mechanisms that are available.
For now, yes. As I said above, we are working on a new mechanism to detect if the client has disconnected.
If there's curiosity, I'll write a blog post about it as well. There are some subtle (but important) differences.
If you learned something useful with this blog post, consider [sponsoring me on GitHub], and/or share this
blog post among your colleagues.
If you have more ideas about what would be interesting to share, feel free to let me know on LinkedIn or Twitter.