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 starlette.status import HTTP_201_CREATED |
|||
from fastapi import FastAPI, status |
|||
|
|||
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): |
|||
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