A brief overview of FastApi For prototyping, keep things simple, get a professional to check your code ... and delegate the hard stuff if you're not comfortable with it.
FastApi seems a decent http
server that's quick (other languages are faster). The book uses MongoDB in later chapters, but I found it uncomfortable and painful to setup and use. For that reason, I prefer SQLite.
The FastApi documentation can sometimes be unclear, or overly complicated to find what you need, and mould it to your requirements.
The book is good from a high-level view, but has many errors and continuity issues, so you'll have to fix things and check documentation. It's also fast becoming outdated (dependency hell) as FastApi evolves, so watch out for changes and updates. Authentication is easy(ish) to use (with guidance) but email notifications aren't baked in, and other nice-to-haves can be difficult:
- Backend servers require domain knowledge (and low-level detail)
- There's a lot that I'm personally not 100% comfortable with DIYing ...
- It's wise to find an experienced dev to help out and double-check code.
/auth
,/database
and/.env
are especially important to get right!- I'd prefer to treat parts of the server as a "black box" (set and forget)
- Extra "defensive coding" and security advice may be needed ...
- Error messaging is particularly frustrating (cryptic, verbose, incomplete)
Have a clear goal, a clear learning frame
I think it's wise to provide yourself a clear learning frame, by which I mean drawing a clear line between what you're prepared to learn, and what you're not. For example, getting a working and reliable email confirmation script is non-trivial!
Personally, I'd prefer someone else to handle things like that, so unless there's a well-documented and stable plugin, I'm going to hire a professional. I prefer things as simple as possible; SQLite and FastApi routes are easy enough to understand the basics, but there can be a lot of moving parts! There's many ways to build out your app architecture, and I'm not sure there's a book out there that covers the best way to do things for your app.
There's so much to learn with programming it's good to set your own boundaries!
uv run uvicorn api:app --port 8000 --reload
(or run from.venv
)uv run main.py
(if you've setup properly__main__.py
)- Or,
fastapi dev main.py
(seems to essentially be the same)
- Or,
uv run pyright main.py
(run in strict mode, Pylance in VS Code)
curl -X 'POST' \
'http://localhost:8000/user/signin' \
-H 'accept: application/json' \
-H 'Content-Type: application/x-www-form-urlencoded' \
-d 'grant_type=password&username=[email]&password=[password]&scope=&client_id=string&client_secret=string'
-- Chapter 07 code
SELECT u.email, e.title FROM user AS u
JOIN event AS e ON u.email = e.creator; -- inner join
/docs
gives a JSON Schema documentation .../redoc
provides alternative documentation.
To implement these properly leads to messy code! Things like Annotated[]
, "json_schema_extra"
metadata, and so on. I'm finding that Bruno is pretty nice to work with (as an alternative) and does most of what I'd need.
Some notes on using
uv
andvenv
setup1
- Hello World
- Routing (1.6.0 — 1.6.6)
- Response models and error handling (1.7.0 — 1.7.4)
- Templating with Jinja (1.8.0 — 1.8.2)
— 1.8.1 for
json
version - Structuring FastApi applications (1.9.0 — 1.9.1)
- Working with the database
- SQLModel (1.10.0 — 1.10.6)
MongoDB2 (I'm sticking with SQLite)
- Securing FastApi applications (1.11.0 — 1.11.9)
- Hash and compare passwords
- Generating JWT tokens
- Securing routes (with authentication)
- CORS policy (middleware)
The SQLModel documentation isn't always great, and some things that should be easy enough, don't seem to be. Compared to Elm lang, Python errors can be a bit cryptic and using the REPL for "practice" or "discovery" isn't always easy or possible with SQLModel setup.
:id
not added to the Bruno path parameters (gettingmethod not allowed
)count()
fails hard: the alternative isfirst()
with SQLModel
The major rule for writing is ... BE CONSISTENT!
- There's a lot of small mistakes and continuity errors ...
- So use
⚠️ #!
style comments for major breaking code!- A single source of truth for code (edits break things)
For example, pg.131:
NewUser
model is mentioned but not createdUser
fields are not yet usedUser.username
is used (curl
example) but not createdusers.py
is referred to asuser.py
Also
- Make sure any required dependencies are introduced clearly!
SQLModel
is imported, but no download is mentioned.jose
has the same problem. Whichjose
package do you mean?!
- Some "upgrades", such as
take time to learn@app.on_event("startup")
- The app lifecycle, for example, requires understanding of
contextlib
.
- The app lifecycle, for example, requires understanding of
grant_type=
missing thepassword
keyword in the authentication curl call.
You're going to need the following:
- Pydantic
- Uvicorn
- VS Code Python plugin
- PyRight (CLI in strict mode, Pylance in VS Code)
- ORM of some description
- Be extra careful with raw SQL.
A great API test kit for Mac. Much simpler than the alternatives (IMO)
The only downsides to using Bruno is you've got to manually write your documentation and tests. FastApi comes with /docs
and /redoc
which are pretty handy, but I prefer Bruno's way of writing documentation. Doing things in Bruno means we can easily switch to a different API framework and keep all our tests in place.
- Use OAuth2 with Bruno
- Import
openapi.json
to a new collection
Rest Client, Postman3, and Insomnia are other options.
My general opinion is:
- Simple is better
- Smaller is better
- Less is better (dependencies)
- Human readable is better (docs)
- Boring is better (in general, stable)
- How do I generate
UUID
s (and what about performance?) - How do I load secrets and environment variables? with Pydantic?
- I think SQLite setup is a lot easier than Postgres
- Is async desirable with SQLite? (NEVER pre-optimise, wait until there's a need)
- The package is now deprecated, but there are 3rd party tools available.
- You might also be better off with RabbitMQ or some other queuing/sharding.
- Why would I use Peewee over SQLAlchemy or SQLModel?
Possibly better to do data migrations simply and often? Also may have to consider the
json
and client code (w/ business logic) Also handy isuser_version
which you can do like this.
FastApi doesn't come with data migration, so it might be wise to do this manually with SQLite, or find a solid tool (or Ai) to help you. Alembic seems a bit difficult. In general the advice seems to be create new (column, table) and copy data over (from old column) before dropping the old. Practice on a dummy database first, and always backup first!!
- Change the
user version
- Use ORM tools if you prefer
- GUIs like Enso or Ai might be helpful too!
I really don't want to use Python's OOP style very much. The book is a bit sloppy in places with conflicting instructions. Elm Lang just "feels" nicer: documentation, error messaging, and so on.
json
is preferrable to.jinja
(at scale)4 (just use Elm?)- Try to avoid Python "magic" that isn't transferable
- Features like
@classmethod
,response_model=
are handy but not portable
- Features like
- Aim to keep your models, SQL, data, and code as simple as possible
- Pydantic documentation is kind of narly and confusing. Some examples in the book are (already) outdated.
- If you're unsure about something, possibly best to leave it out.
- Is it wise to use
SQLModel
classes for request body?- Search
"using fastapi SQLModel as request body"
on Brave browser ... - Understand if it's wiser to use
BaseModel
for your request body (and separate concerns). SQLModel
is only usable if all fields in request body are provided (other thanOptional
ones). These seems suboptimal if user preferences requires many fields!! Perhaps this could be handled client-side (enforce non-optional fields)?
- Search
- For
status_code=
the book usesstatus.HTTP_403_FORBIDDEN
but I'm just using the403
code by itself, as it's cleaner. This is debatable. Depends()
is an important function that injects dependencies into our routes, forcing our route to handle something (such asoauth3_scheme
) first.- Why use
response_model=
instead of a response type?
Whereas Elm has a central Model
(generally) to work from and uses modules and functions, Python has instances of classes which (I think) are stateful. It feels like Python adds a whole lot of mess to the code base.
A good example of this is FastAPI allows generating API examples along with your models. I feel the model and examples should be handled separately, as the code becomes messy. Better to let Bruno handle the documentation (and use /docs
as-is), rather than this:
class ToDo(BaseModel):
id: int
item: Item
model_config = {
"json_schema_extra": {
"examples": [
{
"id": "1",
"item": {
"item": "Grab some shopping for dinner",
"status": "to-do"
}
}
]
}
}
These are annoying and compared to Elm (where everything just works), not particularly user-friendly. You'd think you could just run commands once you're set up with a
venv
(virtual environment).
- Dot notation: how are values extracted?
dictionary.id
rather thandictionary["id"]
- Module naming clashes and
venv
:**uv
commands require calling from thevenv
parent directory01
numbers cannot come first for module naming (name_01
is ok)
uvicorn
command won't run- Preface it with
uv run
(equivalent topython -m
) - Or, make sure you've
source .venv/bin/activate
d your environment - Also take care with modules, folders, and names.5
- Preface it with
uvicorn
doesn't allow securehttps
(by default)- Using Thonny as an IDE
- I can get the version of Python running but the other stuff is harder
- You might want to return
html
instead ofjson
. You can do both!
- AGSI setup in Python Anywhere
Footnotes
-
"I would not recommend using directory names in
.python-version
files or using a custom virtual environment name in a project." ↩ -
MongoDB is a an arse to setup (especially for beginners), more hassle than SQLite (and possibly Postgres easier also). The code in chapters
06
/07
and above also starts to get Pythonic (class methods rather than functional style) and I see diminishing returns from learning in the MongoDB style. ↩ -
I find Postman too flippin' complicated to use. Insomnia is another option. ↩
-
jinja
code adds a bit more complexity to your API code. Seems great for small setups but could prove a liability with complicated forms and UI. For now, handle most of the complexity with Elm lang and look into HTMX or static site generators for blog posts and FAQs. Possibly handleUser
admin with FastApi. ↩ -
There's two ways to do this. Either call
uvicorn subfolder.file:app
and make sure your modules usefrom subfolder.file
names, ORcd
into the correct folder and run Uvicorn from there! ↩