Browse Source
* ✨ Re-export main features used from Starlette to simplify developer's code * ♻️ Refactor Starlette exports * ♻️ Refactor tutorial examples to use re-exported utils from Starlette * 📝 Add examples for all middlewares * 📝 Add new docs for middlewares * 📝 Add examples for custom responses * 📝 Extend docs for custom responses * 📝 Update docs and add notes explaining re-exports from Starlette everywhere * 🍱 Update screenshot for HTTP status * 🔧 Update MkDocs config with new content * ♻️ Refactor tests to use re-exported utils from Starlette * ✨ Re-export WebSocketDisconnect from Starlette for tests * ✅ Add extra tests for extra re-exported middleware * ✅ Add tests for re-exported responses from Starlette * ✨ Add docs about mounting WSGI apps * ➕ Add Flask as a dependency to test WSGIMiddleware * ✅ Test WSGIMiddleware examplepull/1065/head
committed by
GitHub
267 changed files with 1000 additions and 403 deletions
@ -0,0 +1,97 @@ |
|||||
|
In the main tutorial you read how to add [Custom Middleware](../tutorial/middleware.md){.internal-link target=_blank} to your application. |
||||
|
|
||||
|
And then you also read how to handle [CORS with the `CORSMiddleware`](../tutorial/cors.md){.internal-link target=_blank}. |
||||
|
|
||||
|
In this section we'll see how to use other middlewares. |
||||
|
|
||||
|
## Adding ASGI middlewares |
||||
|
|
||||
|
As **FastAPI** is based on Starlette and implements the <abbr title="Asynchronous Server Gateway Interface">ASGI</abbr> specification, you can use any ASGI middleware. |
||||
|
|
||||
|
A middleware doesn't have to be made for FastAPI or Starlette to work, as long as it follows the ASGI spec. |
||||
|
|
||||
|
In general, ASGI middlewares are classes that expect to receive an ASGI app as the first argument. |
||||
|
|
||||
|
So, in the documentation for third-party ASGI middlewares they will probably tell you to do something like: |
||||
|
|
||||
|
```Python |
||||
|
from unicorn import UnicornMiddleware |
||||
|
|
||||
|
app = SomeASGIApp() |
||||
|
|
||||
|
new_app = UnicornMiddleware(app, some_config="rainbow") |
||||
|
``` |
||||
|
|
||||
|
But FastAPI (actually Starlette) provides a simpler way to do it that makes sure that the internal middlewares to handle server errors and custom exception handlers work properly. |
||||
|
|
||||
|
For that, you use `app.add_middleware()` (as in the example for CORS). |
||||
|
|
||||
|
```Python |
||||
|
from fastapi import FastAPI |
||||
|
from unicorn import UnicornMiddleware |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
app.add_middleware(UnicornMiddleware, some_config="rainbow") |
||||
|
``` |
||||
|
|
||||
|
`app.add_middleware()` receives a middleware class as the first argument and any additional arguments to be passed to the middleware. |
||||
|
|
||||
|
## Integrated middlewares |
||||
|
|
||||
|
**FastAPI** includes several middlewares for common use cases, we'll see next how to use them. |
||||
|
|
||||
|
!!! note "Technical Details" |
||||
|
For the next examples, you could also use `from starlette.middleware.something import SomethingMiddleware`. |
||||
|
|
||||
|
**FastAPI** provides several middlewares in `fastapi.middleware` just as a convenience for you, the developer. But most of the available middlewares come directly from Starlette. |
||||
|
|
||||
|
## `HTTPSRedirectMiddleware` |
||||
|
|
||||
|
Enforces that all incoming requests must either be `https` or `wss`. |
||||
|
|
||||
|
Any incoming requests to `http` or `ws` will be redirected to the secure scheme instead. |
||||
|
|
||||
|
```Python hl_lines="2 6" |
||||
|
{!./src/advanced_middleware/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
## `TrustedHostMiddleware` |
||||
|
|
||||
|
Enforces that all incoming requests have a correctly set `Host` header, in order to guard against HTTP Host Header attacks. |
||||
|
|
||||
|
```Python hl_lines="2 6 7 8" |
||||
|
{!./src/advanced_middleware/tutorial002.py!} |
||||
|
``` |
||||
|
|
||||
|
The following arguments are supported: |
||||
|
|
||||
|
* `allowed_hosts` - A list of domain names that should be allowed as hostnames. Wildcard domains such as `*.example.com` are supported for matching subdomains to allow any hostname either use `allowed_hosts=["*"]` or omit the middleware. |
||||
|
|
||||
|
If an incoming request does not validate correctly then a `400` response will be sent. |
||||
|
|
||||
|
## `GZipMiddleware` |
||||
|
|
||||
|
Handles GZip responses for any request that includes `"gzip"` in the `Accept-Encoding` header. |
||||
|
|
||||
|
The middleware will handle both standard and streaming responses. |
||||
|
|
||||
|
```Python hl_lines="2 6 7 8" |
||||
|
{!./src/advanced_middleware/tutorial002.py!} |
||||
|
``` |
||||
|
|
||||
|
The following arguments are supported: |
||||
|
|
||||
|
* `minimum_size` - Do not GZip responses that are smaller than this minimum size in bytes. Defaults to `500`. |
||||
|
|
||||
|
## Other middlewares |
||||
|
|
||||
|
There are many other ASGI middlewares. |
||||
|
|
||||
|
For example: |
||||
|
|
||||
|
* <a href="https://docs.sentry.io/platforms/python/asgi/" class="external-link" target="_blank">Sentry</a> |
||||
|
* <a href="https://github.com/encode/uvicorn/blob/master/uvicorn/middleware/proxy_headers.py" class="external-link" target="_blank">Uvicorn's `ProxyHeadersMiddleware`</a> |
||||
|
* <a href="https://github.com/florimondmanca/msgpack-asgi" class="external-link" target="_blank">MessagePack</a> |
||||
|
|
||||
|
To see other available middlewares check <a href="https://www.starlette.io/middleware/" class="external-link" target="_blank">Starlette's Middleware docs</a> and the <a href="https://github.com/florimondmanca/awesome-asgi" class="external-link" target="_blank">ASGI Awesome List</a>. |
@ -0,0 +1,35 @@ |
|||||
|
You can mount WSGI applications as you saw with [Sub Applications - Behind a Proxy, Mounts](./sub-applications-proxy.md){.internal-link target=_blank}. |
||||
|
|
||||
|
For that, you can use the `WSGIMiddleware` and use it to wrap your WSGI application, for example, Flask, Django, etc. |
||||
|
|
||||
|
## Using `WSGIMiddleware` |
||||
|
|
||||
|
You need to import `WSGIMiddleware`. |
||||
|
|
||||
|
Then wrap the WSGI (e.g. Flask) app with the middleware. |
||||
|
|
||||
|
And then mount that under a path. |
||||
|
|
||||
|
```Python hl_lines="1 3 22" |
||||
|
{!./src/wsgi/tutorial001.py!} |
||||
|
``` |
||||
|
|
||||
|
## Check it |
||||
|
|
||||
|
Now, every request under the path `/v1/` will be handled by the Flask application. |
||||
|
|
||||
|
And the rest will be handled by **FastAPI**. |
||||
|
|
||||
|
If you run it with Uvicorn and go to <a href="http://localhost:8000/v1/" class="external-link" target="_blank">http://localhost:8000/v1/</a> you will see the response from Flask: |
||||
|
|
||||
|
```txt |
||||
|
Hello, World from Flask! |
||||
|
``` |
||||
|
|
||||
|
And if you go to <a href="http://localhost:8000/v2" class="external-link" target="_blank">http://localhost:8000/v2</a> you will see the response from FastAPI: |
||||
|
|
||||
|
```JSON |
||||
|
{ |
||||
|
"message": "Hello World" |
||||
|
} |
||||
|
``` |
Before Width: | Height: | Size: 21 KiB After Width: | Height: | Size: 30 KiB |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.middleware.httpsredirect import HTTPSRedirectMiddleware |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
app.add_middleware(HTTPSRedirectMiddleware) |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def main(): |
||||
|
return {"message": "Hello World"} |
@ -0,0 +1,13 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.middleware.trustedhost import TrustedHostMiddleware |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
app.add_middleware( |
||||
|
TrustedHostMiddleware, allowed_hosts=["example.com", "*.example.com"] |
||||
|
) |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def main(): |
||||
|
return {"message": "Hello World"} |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.middleware.gzip import GZipMiddleware |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
app.add_middleware(GZipMiddleware, minimum_size=1000) |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def main(): |
||||
|
return "somebigcontent" |
@ -0,0 +1,9 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import PlainTextResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/", response_class=PlainTextResponse) |
||||
|
async def main(): |
||||
|
return "Hello World" |
@ -0,0 +1,9 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import RedirectResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/typer") |
||||
|
async def read_typer(): |
||||
|
return RedirectResponse("https://typer.tiangolo.com") |
@ -0,0 +1,14 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import StreamingResponse |
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
async def fake_video_streamer(): |
||||
|
for i in range(10): |
||||
|
yield b"some fake video bytes" |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def main(): |
||||
|
return StreamingResponse(fake_video_streamer()) |
@ -0,0 +1,11 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import StreamingResponse |
||||
|
|
||||
|
some_file_path = "large-video-file.mp4" |
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
def main(): |
||||
|
file_like = open(some_file_path, mode="rb") |
||||
|
return StreamingResponse(file_like, media_type="video/mp4") |
@ -0,0 +1,10 @@ |
|||||
|
from fastapi import FastAPI |
||||
|
from fastapi.responses import FileResponse |
||||
|
|
||||
|
some_file_path = "large-video-file.mp4" |
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/") |
||||
|
async def main(): |
||||
|
return FileResponse(some_file_path) |
@ -1,9 +1,8 @@ |
|||||
from fastapi import FastAPI |
from fastapi import FastAPI, status |
||||
from starlette.status import HTTP_201_CREATED |
|
||||
|
|
||||
app = FastAPI() |
app = FastAPI() |
||||
|
|
||||
|
|
||||
@app.post("/items/", status_code=HTTP_201_CREATED) |
@app.post("/items/", status_code=status.HTTP_201_CREATED) |
||||
async def create_item(name: str): |
async def create_item(name: str): |
||||
return {"name": name} |
return {"name": name} |
||||
|
@ -0,0 +1,22 @@ |
|||||
|
from flask import Flask, escape, request |
||||
|
from fastapi import FastAPI |
||||
|
from fastapi.middleware.wsgi import WSGIMiddleware |
||||
|
|
||||
|
flask_app = Flask(__name__) |
||||
|
|
||||
|
|
||||
|
@flask_app.route("/") |
||||
|
def flask_main(): |
||||
|
name = request.args.get("name", "World") |
||||
|
return f"Hello, {escape(name)} from Flask!" |
||||
|
|
||||
|
|
||||
|
app = FastAPI() |
||||
|
|
||||
|
|
||||
|
@app.get("/v2") |
||||
|
def read_main(): |
||||
|
return {"message": "Hello World"} |
||||
|
|
||||
|
|
||||
|
app.mount("/v1", WSGIMiddleware(flask_app)) |
@ -0,0 +1 @@ |
|||||
|
from starlette.background import BackgroundTasks # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware import Middleware |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware.cors import CORSMiddleware # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware.gzip import GZipMiddleware # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware.httpsredirect import HTTPSRedirectMiddleware # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware.trustedhost import TrustedHostMiddleware # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.middleware.wsgi import WSGIMiddleware # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.requests import Request # noqa |
@ -0,0 +1,8 @@ |
|||||
|
from starlette.responses import FileResponse # noqa |
||||
|
from starlette.responses import HTMLResponse # noqa |
||||
|
from starlette.responses import JSONResponse # noqa |
||||
|
from starlette.responses import PlainTextResponse # noqa |
||||
|
from starlette.responses import RedirectResponse # noqa |
||||
|
from starlette.responses import Response # noqa |
||||
|
from starlette.responses import StreamingResponse # noqa |
||||
|
from starlette.responses import UJSONResponse # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.staticfiles import StaticFiles # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.templating import Jinja2Templates # noqa |
@ -0,0 +1 @@ |
|||||
|
from starlette.testclient import TestClient # noqa |
@ -0,0 +1,2 @@ |
|||||
|
from starlette.websockets import WebSocket # noqa |
||||
|
from starlette.websockets import WebSocketDisconnect # noqa |
Some files were not shown because too many files changed in this diff
Loading…
Reference in new issue