Templates¶
You can use any template engine you want with FastAPI.
A common choice is Jinja2, the same one used by Flask and other tools.
There are utilities to configure it easily that you can use directly in your FastAPI application (provided by Starlette).
Install dependencies¶
Install jinja2:
$ pip install jinja2
---> 100%
Using Jinja2Templates¶
- Import Jinja2Templates.
- Create a templatesobject that you can re-use later.
- Declare a Requestparameter in the path operation that will return a template.
- Use the templatesyou created to render and return aTemplateResponse, passing therequestas one of the key-value pairs in the Jinja2 "context".
from fastapi import FastAPI, Request
from fastapi.responses import HTMLResponse
from fastapi.staticfiles import StaticFiles
from fastapi.templating import Jinja2Templates
app = FastAPI()
app.mount("/static", StaticFiles(directory="static"), name="static")
templates = Jinja2Templates(directory="templates")
@app.get("/items/{id}", response_class=HTMLResponse)
async def read_item(request: Request, id: str):
    return templates.TemplateResponse("item.html", {"request": request, "id": id})
Note
Notice that you have to pass the request as part of the key-value pairs in the context for Jinja2. So, you also have to declare it in your path operation.
Tip
By declaring response_class=HTMLResponse the docs UI will be able to know that the response will be HTML.
Technical Details
You could also use from starlette.templating import Jinja2Templates.
FastAPI provides the same starlette.templating as fastapi.templating just as a convenience for you, the developer. But most of the available responses come directly from Starlette. The same with Request and StaticFiles.
Writing templates¶
Then you can write a template at templates/item.html with:
<html>
<head>
    <title>Item Details</title>
    <link href="{{ url_for('static', path='/styles.css') }}" rel="stylesheet">
</head>
<body>
    <h1>Item ID: {{ id }}</h1>
</body>
</html>
It will show the id taken from the "context" dict you passed:
{"request": request, "id": id}
Templates and static files¶
And you can also use url_for() inside of the template, and use it, for example, with the StaticFiles you mounted.
<html>
<head>
    <title>Item Details</title>
    <link href="{{ url_for('static', path='/styles.css') }}" rel="stylesheet">
</head>
<body>
    <h1>Item ID: {{ id }}</h1>
</body>
</html>
In this example, it would link to a CSS file at static/styles.css with:
h1 {
    color: green;
}
And because you are using StaticFiles, that CSS file would be served automatically by your FastAPI application at the URL /static/styles.css.
More details¶
For more details, including how to test templates, check Starlette's docs on templates.