FastAPI
FastAPI⚑
(https://fastapi.tiangolo.com/) is a modern, fast (high-performance), web framework for building HTTP APIs based on standard Python type hints.
Example (main.py):
from typing import Optional
from fastapi import FastAPI
app = FastAPI()
@app.get("/")
def read_root():
return {"Hello": "World"}
@app.get("/items/{item_id}")
def read_item(item_id: int, q: Optional = None):
return {"item_id": item_id, "q": q}
Run it with uvicorn main:app --reload
and then open your browser at http://127.0.0.1:8000/items/5?q=somequery.
Usage⚑
There are two main ways of using this module, with the decorator @app.get("{{route}}")
or with FastAPI().add_route("{{route}}", {{function}})
.
The first alternative is by far the most common, but if you want to separate the functions from the HTTP API entrypoint, you have to use the second option. For example:
from fastapi import FastAPI
api = FastAPI()
def hello(name: str) -> str:
return "Hello " + name + "!"
api.add_route('/hello', hello)
The advantage of this alternative is that we can import the function from another module.
If you need to change the default value of some argument of the original function, you can use (python_basics.md#functools). This might be necessary if an argument is a list, since you'll need to set its default value to Query()
you can change it with functools
instead of modifying the original function (which would be pretty ugly). Make sure of copying the original docstring.
Nevertheless, this approach has some caveats such as not handling exceptions raised from the imported function. The best solution for this appears to be defining a wrapper function that handles the arguments, calls the imported function or method and handles the exceptions raising the proper HTTPException
.
Dependency injection⚑
A FastAPI decorated function can import the necessary arguments for other functions or objects.
For example:
from typing import Optional
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(q: Optional = None, skip: int = 0, limit: int = 100):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
In this example, the function read_items
will get the parameters for the function common_parameters
(i.e., q
, skip
, limit
) allowing the optional ones to be set or not. Then, the passed parameters will be passed as a dictionary as the commons
argument value of read_itmes
. This is thanks to Depends
.
Depends
can also be used with classes, and we could get an instance of a class directly to our function. For example by setting the argument: car: Car = Depends(Car)
or car: Car = Depends()
(which assumes that it depends on the class defined in the type hint).
JSON encoder⚑
The default the responses will be serialized to JSON. You can use a different JSON encoder than the default one, for example (https://github.com/ijl/orjson) which is more efficient and natively support serialization of dataclasses and more.
To set the default encoder do:
from fastapi import FastAPI
from fastapi.responses import ORJSONResponse
FastAPI(default_response_class=ORJSONResponse)
Testing⚑
Testing startup and shutdown events⚑
When you need your event handlers (startup
and shutdown
) to run in your tests, you can use the TestClient
with a with statement:
import pytest
from fastapi.testclient import TestClient
from api.main import api
@pytest.fixture
def client():
with TestClient(api) as client:
yield client
def test_endpoint(client):
response = client.get("/endpoint")
assert response.status_code == 200
Dynamically getting the endpoint URLs⚑
If you want to get the URL of an endpoint dynamically, you can use:
# get url for endpoint
url = app.url_path_for("endpoint_name")
Performance⚑
Performance can be hugely improved with some minor changes:
- Use
orjson
(faster JSON serialization):
from fastapi import FastAPI
from fastapi.responses import ORJSONResponse
app = FastAPI(default_response_class=ORJSONResponse)
...
response_model=...
, directly return a Response
object if you've already done the pydantic
model validation (or if you don't want to validate the object at all). Otherwise the validation will happen twice which is computationally expensive. ...
from fastapi.encoders import jsonable_encoder
class Thing(BaseModel):
name: str
@app.get("/thing', response_model=Thing)
def get_thing():
thing: Thing = Thing(name="something")
return ORJSONResponse(content=jsonable_encoder(result))
ETag
and Cache-Control
headers. You will need to use the PickleCoder
coder if you want to cache responses directly. Otherwise if you directly return a dict
or similar (not a Response
) the cache will store that object and then the model validation will be executed unnecessarily every time the endpoint is called..
All together:
from fastapi import FastAPI
from fastapi.responses import ORJSONResponse
from fastapi.encoders import jsonable_encoder
from fastapi_cache.backends.inmemory import InMemoryBackend
from fastapi_cache.coder import PickleCoder
app = FastAPI(default_response_class=ORJSONResponse)
@app.on_event("startup")
async def startup():
FastAPICache.init(InMemoryBackend())
class Thing(BaseModel):
name: str
@app.get("/thing', response_model=Thing)
@cache(expire=7 * 24 * 60 * 60, coder=PickleCoder) # 7 days
def get_thing():
thing: Thing = Thing(name="something")
return ORJSONResponse(content=jsonable_encoder(result))
Logging⚑
To add custom log messages to the uvicorn
output, the dirty way, get the logger called uvicorn
:
logger = logging.getLogger("uvicorn")