Ifan Johnston

Ifan Johnston

fastbooks - FastAPI, sqlmodel, alembic and Jinja

We will be building a fastapi app using sqlmodel to handle a connection to the database, alembic for migrations, and Jinja2 templates for the front end. Let's get straight into it!

To see the full code, see this git repo.

Package requirements

We will start by installing all the packages we will need into a virtual environment.

python3 -m venv fastenv
source fastenv/bin/activate
pip install fastapi sqlmodel uvicorn Jinja2 aiofiles

Those packages are enough to get off the ground for local development. We will be using an sqlite database for local development, but when we deploy to Heroku we'll use Postgres so let's install some additional things we need for later.

pip install psycopg2 alembic gunicorn

We also installed alembic for database migrations, and gunicorn to use as a production server. The psycopg2 package allows sqlmodel (which uses SQLAlchemy underneath) to talk to Postgres databases.

Those are all the packages we are going to need, so let's save them to a requirements.txt file before we forget.

pip freeze > requirements.txt

Fastapi app

Start by making the files main.py and __init__.py inside a directory called app, so that the file structure looks like this:

.
├── app
│   ├── __init__.py
│   ├── main.py
├── bookenv
│   ├── bin
│   ├── include
│   ├── lib
│   └── pyvenv.cfg
└── requirements.txt
# app/main.py
from fastapi import FastAPI
import uvicorn

app = FastAPI()

@app.get("/")
def home_page():
    return {"message": "Hello!"}

if __name__=="__main__":
    uvicorn.run('app.main:app', reload=True)

Running this with python -m app.main should start up a server on port 8000 on your localhost, so going to 127.0.0.1:8000 should show you the message

{"message": "Hello!"}

Making it look nice

Before setting up our database and models, let's make the first page look nice using Jinja templates.

from fastapi import FastAPI
import uvicorn

from starlette.staticfiles import StaticFiles
from starlette.templating import Jinja2Templates
from starlette.requests import Request

templates = Jinja2Templates('app/templates')

app = FastAPI()

def configure():
    app.mount('/static', StaticFiles(directory='app/static'), name='static')

@app.get("/")
def home_page(request: Request):
    return templates.TemplateResponse('index.html', {'request': request})

if __name__=="__main__":
    configure()
    uvicorn.run('app.main:app', reload=True)
else:
    configure()

After importing the required packages, we set up the templates with Jinja2Templates('app/templates') and also mount the static files so that fastapi can find them with app.mount('/static', StaticFiles(directory='app/static', name='static'), so we need to make 2 directories:

mkdir app/static
mkdir app/templates

Inside templates, we'll create the following index.html file,

<!DOCTYPE html>

<html lang="en-us">
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="description" content="A website showcasing fastapi and sqlmodel">
    <meta name="author" content="idjotherwise">

    <title>Fastsqlmodel showcase</title>

    <link href="//oss.maxcdn.com/libs/twitter-bootstrap/3.0.3/css/bootstrap.min.css" rel="stylesheet">
</head>

<body>
    <h1>
        A super stylish home page
    </h1>
    <p>
        Really interesting content, full of buzzwords like AI and MACHINE LEARNING!
    </p>
</body>
</html>

Now we can run the app to check that things are working so far,

python -m main.app

If everything has gone well, after opening up localhost:8000, you should be greeted with

You should be greeted with this lovely webpage

We'll refactor the jinja templates later, but that will do for now.

Next we'll set up some database stuff, including some models to hold our data.

Database: sqlmodel

In a new file called database.py, we'll set up the database engine and session handler.

# app/database.py

from sqlmodel import SQLModel, create_engine, Session

from .settings import settings

connect_args = {"check_same_thread": False}
engine = create_engine(settings.database_url), connect_args=connect_args)


def create_db_and_tables():
    SQLModel.metadata.create_all(engine)


def get_session():
    with Session(engine) as session:
        yield session

and in the file settings.py we'll define the DATABASE_URL,

# app/settings.py

from pydantic import BaseSettings

class Settings(BaseSettings):
    database_url: str = 'sqlite:///database.db'

This may seem uncessarily complex, but it will make things easier later. When your class Settings inherits from BaseSettings, it will first try to get the attributes from your environment but if it doesn't find anything it will set it to the default you've given. So in our case (assuming you haven't set DATABASE_URL to anything else in your terminal) it will simply be the sqlite database url. On Heroku, environment variables like DATABASE_URL are set automatically (if you have installed a Postgres add-on).

Before we actually create the database, we should create a model that will go into the database. In a new file models.py,

# app/models.py

from sqlmodel import SQLModel, Field
from typing import Optional


class BookBase(SQLModel):
    title: str
    author: Optional[str]


class Book(BookBase, table=True):
    id: Optional[int] = Field(default=None, primary_key=True)

In sqlmodel, models can be treated as both Pydantic models and SQLAlchemy models. That's all it takes to create the table called Book in the database. Now let's actually create the table, we add some code to main.py

# app/main.py

from fastapi import FastAPI, Depends
import uvicorn

from starlette.staticfiles import StaticFiles
from starlette.templating import Jinja2Templates
from starlette.requests import Request

from sqlmodel import Session, select
from .database import create_db_and_tables, get_session
from .models import Book

templates = Jinja2Templates('app/templates')

app = FastAPI()


def configure():
    create_db_and_tables()
    app.mount('/static', StaticFiles(directory='app/static'), name='static')

@app.get("/")
def home_page(*, session: Session = Depends(get_session), request: Request):
    books = session.exec(select(Book)).all()
    return templates.TemplateResponse('index.html', {'request': request})

if __name__=="__main__":
    configure()
    uvicorn.run('app.main:app', reload=True)
else:
    configure()

Finally add the following bit of code to the index.html file (see the Jinja documentation for more information),

<!DOCTYPE html>

<html lang="en-us">
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="description" content="A website showcasing fastapi and sqlmodel">
    <meta name="author" content="idjotherwise">

    <title>Fastsqlmodel showcase</title>

    <link href="//oss.maxcdn.com/libs/twitter-bootstrap/3.0.3/css/bootstrap.min.css" rel="stylesheet">
</head>

<body style="text-align: center">
    <h1>
        A super stylish home page
    </h1>
    <p>
        Really interesting content, full of buzzwords like AI and MACHINE LEARNING!
    </p>
    {% if books %}
    <ul>
        {% for book in books %}
        <li key="{{book.id}}">Title: {{book.title}}, Author: {{book.author}}</li>
        {% endfor %}
    </ul>
    {% endif %}
</body>
</html>

At this point, if you navigate localhost:8000, you won't see anything has changed - that's because our database is empty. But if the database did have books in it, then in the get request the books = session.exec(select(Book)).all() would contain a list of book items, and so our Jinja template index.html would get passed a non-empty list.

There are many ways to add data into our database. You could use SQLite Browser to open the database.db file, PyCharm's own database reader, or any other database browser. Instead, we will add a POST route so that we can add in data from the browser.

Add the following code the main.py, just below the @app.get('/') route.

@app.post("/book")
def add_book(*, session: Session = Depends(get_session), request: Request, book: BookCreate) -> BookRead:
    db_book = Book.from_orm(book)
    session.add(db_book)
    session.commit()
    session.refresh(db_book)
    return db_book

We've added in two new models here, BookCreate and BookRead. Add them to the models.py file.

# app/models.py
from sqlmodel import SQLModel, Field
from typing import Optional


class BookBase(SQLModel):
    title: str
    author: Optional[str]


class Book(BookBase, table=True):
    id: Optional[int] = Field(default=None, primary_key=True)


class BookCreate(BookBase):
    pass


class BookRead(BookBase):
    id: int

Now, navigate to localhost:8000/docs in your browser, and you should see the POST method right there:

You should see this in the docs

Click on the post request and add a few books in. These will now be saved into the database. Thanks to the way we wrote the index.html file, if you navigate again to the home page (localost:8000), you should see the books that you just added. Since we didn't implement @app.delete or @app.patch methods, we can't delete or update the entries. To start afresh, you can just delete the database.db file and re-add the books.

Adding a description column with Alembic

Now let's say that we want to add a description field to the book items. We could just delete the database.db file, make the changes to the Book model, and restart the app. However, if we were in the situation where the database had a bunch of entries that you don't want to lose, then using migrations is the way to go. We'll be using alembic, see the documentation for more information. We need to do a few changes to the setup since we are using sqlmodel and not sqlalchemy. TestDriven.io has a nice tutorial showing how to do this, so we'll follow that here (although we won't be using the async version) Start by going to the terminal, from root directory of the project, run

alembic init alembic

This will make a directory called alembic with some files that we need to modify. The project structure should now look like this

.
|-- alembic
|   |-- README
|   |-- env.py
|   |-- script.py.mako
|   `-- versions
|-- alembic.ini
|-- app
|   |-- __init__.py
|   |-- database.py
|   |-- main.py
|   |-- models.py
|   |-- settings.py
|   `-- static/
|   `-- templates
|       `-- index.html
|-- database.db
`-- requirements.txt

In the script.py.mako file, we need to import sqlmodel:

# alembic/script.py.mako
# code above ommited

from alembic import op
import sqlalchemy as sa
import sqlmodel
${imports if imports else ""}

# code below ommited

and finally in the alembic/env.py add the following imports,

from logging.config import fileConfig

from sqlalchemy import engine_from_config
from sqlalchemy import pool

from alembic import context

from sqlmodel import SQLModel
from app.settings import settings
from app.models import Book

# this is the Alembic Config object, which provides
# access to the values within the .ini file in use.
config = context.config

config.set_main_option("sqlalchemy.url", settings.database_url)

# Interpret the config file for Python logging.
# This line sets up loggers basically.
fileConfig(config.config_file_name)

# add your model's MetaData object here
# for 'autogenerate' support

# target_metadata = mymodel.Base.metadata
target_metadata = SQLModel.metadata

# ---- code below omitted ---

All we've done here is import the SQLModel from sqlmodel, which holds all the metadata about our database (after also importing our Book model), then we've grabbed the database_url from the settings file.

Now delete the database.db file, and then generate the first alembic migration with alembic revision -autogenerate -m "Add book table" to verify that everything sets up properly. If successful, you will find a new file in the alembic/revisions directory which looks something like this (the revision ID might be different):

"""Add book table

Revision ID: 7071875a0907
Revises:
Create Date: 2021-09-12 22:16:26.595677

"""
from alembic import op
import sqlalchemy as sa
import sqlmodel


# revision identifiers, used by Alembic.
revision = '7071875a0907'
down_revision = None
branch_labels = None
depends_on = None


def upgrade():
    # ### commands auto generated by Alembic - please adjust! ###
    op.create_table('book',
    sa.Column('title', sqlmodel.sql.sqltypes.AutoString(), nullable=False),
    sa.Column('author', sqlmodel.sql.sqltypes.AutoString(), nullable=True),
    sa.Column('id', sa.Integer(), nullable=True),
    sa.PrimaryKeyConstraint('id')
    )
    op.create_index(op.f('ix_book_author'), 'book', ['author'], unique=False)
    op.create_index(op.f('ix_book_id'), 'book', ['id'], unique=False)
    op.create_index(op.f('ix_book_title'), 'book', ['title'], unique=False)
    # ### end Alembic commands ###


def downgrade():
    # ### commands auto generated by Alembic - please adjust! ###
    op.drop_index(op.f('ix_book_title'), table_name='book')
    op.drop_index(op.f('ix_book_id'), table_name='book')
    op.drop_index(op.f('ix_book_author'), table_name='book')
    op.drop_table('book')
    # ### end Alembic commands ###

Apply the migration (which just creates the table in the database) with alembic upgrade head.

Now to add a new column to the book table in the database, add the following to the Book model in app/models.py,

...
class BookBase(SQLModel):
    title: str
    author: Optional[str]
    description: Optional[str]
...

and then generate another migration file with

alembic revision --autogenerate -m "Add description column"

This gives us the following migration file,

"""Add description column

Revision ID: 0b26b04f3af4
Revises: 7071875a0907
Create Date: 2021-09-12 22:55:29.481379

"""
from alembic import op
import sqlalchemy as sa
import sqlmodel

# revision identifiers, used by Alembic.
revision = '0b26b04f3af4'
down_revision = '7071875a0907'
branch_labels = None
depends_on = None


def upgrade():
    # ### commands auto generated by Alembic - please adjust! ###
    op.add_column('book', sa.Column('description', sqlmodel.sql.sqltypes.AutoString(), nullable=True))
    op.create_index(op.f('ix_book_description'), 'book', ['description'], unique=False)
    # ### end Alembic commands ###


def downgrade():
    # ### commands auto generated by Alembic - please adjust! ###
    op.drop_index(op.f('ix_book_description'), table_name='book')
    op.drop_column('book', 'description')
    # ### end Alembic commands ###

Apply the migration with alembic upgrade head.

Update Jinja template

Now we are going to modify the jinja template in index.html to use the new description column. While we're at it, lets add some more styling to the page with a header and a footer. For this, we'll create a new file templates/layout.html. In there we will put all the things from index.html that should go on every page (if there were more pages).

<!DOCTYPE html>

<html lang="en-us">
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta name="description" content="A website showcasing fastapi and sqlmodel">
    <meta name="author" content="idjotherwise">

    <title>Fastsqlmodel showcase</title>

    <link href="//oss.maxcdn.com/libs/twitter-bootstrap/3.0.3/css/bootstrap.min.css" rel="stylesheet">
    <link href="/static/css/styling.css" rel="stylesheet">
</head>

<body >
<div class="content-wrapper">
    <div class="container">
        <div class="row">
                    <h1>
                          Book list
                    </h1>
        </div>
    </div>
    <div class="row">
        <div class="col-md-10">
            <div>
                {% block content %}
                    <div class="content">
                                THIS PAGE HAS NO CONTENT
                    </div>
                {% endblock %}
            </div>
        </div>
    </div>
    <div class="footer">
        <div class="footer links">
            <ul>
                <li><i class="glyphicon glyphicon-cog icon-muted"></i><a
                                href="https://github.com/idjotherwise"
                                target="_blank">Github</a>
                </li>
                <li><i class="glyphicon glyphicon-globe icon-muted"></i><a href="https://twitter.com/johnstonifan"
                                                                                   target="_blank">Twitter</a>
                </li>
            </ul>
        </div>
    </div>
</div>


</body>
</html>

Now our app/templates/index.html will extend the app/templates/layout.html file:

{% extends "layout.html" %}
{% block content %}

    <h1>
        A super stylish home page
    </h1>
    <p>
        Really interesting content, full of buzzwords like AI and MACHINE LEARNING!
    </p>
    {% if books %}
    <ul style="text-align: left">
        {% for book in books %}
        <li key="{{book.id}}"><b>Title</b>: {{book.title}}<br/>
        <b>Author</b>: {{book.author}}</li>
        <p style="padding-left: 6rem;">{{book.description}}</p>
        {% endfor %}
    </ul>
    {% endif %}

{% endblock %}

Finally, we'll add some simple styling in the file app/static/css/styling.css. You can also just put in your own style file here if you like.

@import url(//fonts.googleapis.com/css?family=Open+Sans:300,400,600,700);

body {
    font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
    font-weight: 300;
    color: black;
    background: lightcyan;
    width: 100%;
}

h1,
h2,
h3,
h4,
h5,
h6 {
    font-family: "Open Sans", "Helvetica Neue", Helvetica, Arial, sans-serif;
    font-weight: 400;
}

p {
    font-weight: 300;
}

.content-wrapper {
    display: flex;
    flex-direction: row;
    flex-wrap: wrap;
    margin: 2rem auto 0 auto;
    height: 85vh;
    text-align: center;
    width: 56rem;
    justify-content: center;
}

Opening up the browser again to localhost:8000 should greet you with something like

This fancy looking webpage

As you can see, I used the locahost:8000/docs page to add a very interesting sounding book to the database and it showed up on the home page. Try adding a bunch more books to see the page filling up. Also go ahead and modify the styling to make it look actually stylish!

Conclusion

We've done quite a bit here! In another post, we'll clean things up a bit by adding tests with pytest and then deploy to Heroku. We'll also refactor some of the endpoints (and add some more) to use the powerful response model stuff from fastapi/sqlmodel.

Thanks for reading!