Code for presentation @ PyMi Meetup (April 19th 2023)
The topic of the presentation was to show a comparison between typical web application stacks in the Python and the Typescript ecosystems.
There are two main comparisons:
- The capabilities in achieving end-to-end typesafety. By end-to-end, I mean that the typesafety should be enforced from the database (ORM-level) to the frontend of the application leveraging automatic linting or generated stubs.
- The performance of the two stacks, in particular in relation to their Async I/O capabilities.
A simple To-Do list application is built in both stacks to demonstrate the differences.
TODO
The Typescript stack is built using the following technologies:
- NodeJS as the back-end runtime
- tRPC as the API framework
- Prisma as the ORM
- React as the frontend framework
This stack is quite popular in the Typescript ecosystem.
It achieves both ORM-level and API-level typesafety. tRPC is also particularly nice to work with, as it syncs the type between the API layer and the frontend layer (React) with automatic linting.
The webserver (NodeJS) supports Async natively.
The Python stack is built in different steps to get to the final stack.
This is the vanilla stack which is commonly used in the python ecosystem.
It doesn't provide any typesafety in the database layer nor in the API layer.
The webserver (Flask) doesn't support Async.
This is a modern stack which is increasingly used in the python ecosystem.
It provides typesafety in the API layer, but not in the database layer.
We can easily leverage the automatic generation of the OpenAPI schema from FastAPI to generate frontend stubs in Typescript to interact with the API with precise types. This is done using the openapi-typescript-codegen
library.
This gets us a quite nice developer experience, slightly worse than tRPC, but still quite good.
A common suggestion is to use Pydantic models to achieve typesafety in the database layer, but this is not ideal as it doesn't play well with SQLAlchemy. SQLModel is a library which aims to solve this problem, but it is still in its early stages.
The webserver (FastAPI) supports Async natively. Since the 1.4 version, SQLAlchemy natively supports Async at the database layer, avoiding the need to use external packages such as databases
.
This is the final stack which is built in this repository.
Other than the improvements of the previous stack, this stack also provides typesafety in the database layer.
SQLAlchemy 2.0 introduced native support for typed ORM models, which is a huge improvement over the previous versions (which required Mypy plugins that weren't well-supported).
SQLAlchemy 2.0 also provides a series of additional performance improvements (2.0 release post)
I expect to see more and more python developers using the FastAPI + SQLAlchemy 2.0 stack in the future, or at least using SQLAlchemy 2.0 in their projects while leveraging libraries like Pydantic to achieve typesafety in the API layer.
There are likely tons of improvements coming to us in the future, such as:
- Future versions of Pydantic will likely integrate much better with SQLAlchemy 2.0 types and we have already hints of that.
- Pydantic 2.0 rewrite in Rust, which will likely provide a huge performance boost to parsing and validation.
- Let's watch for Tiangolo (FastAPI creator) projects and see what he has in store for us with SQLModel.