A clean, extensible, python project template, built using Makefile, ruff, and uv.
Python tooling moves really fast, and python package management is notoriously difficult. How do you resolve dependencies, do you need a lockfile, how can you quickly test your configurations and ensure everything works as expected locally and in production are common and important questions.
There's no definitive answer to this, but I like the idea of a system that is lightweight and lets you move really quickly.
I'm not a fan of theoretical arguments for or against some piece of technology. I like to think about tooling with regards to how quickly and easily it allows you to do the thing it claims to enable. In that light, poetry, in my opinion, has 2 flaws that make it unusable for larger-scale projects:
- It is slow. Way too slow. I've used poetry for large scale production projects in the past, and have had to wait over 700 seconds for a
poetry lock --no-update
to complete. This is too slow, and can completely break the flow of a project. Adding a new dependency to a project, or updating an existing one, should take less than 10 seconds. - Poetry is too rigid, leading to unresolvable dependency conflicts. Poetry doesn't let you override conflicting dependencies. This, in theory, makes sense, but in practice is untenable. If any downstream dependency of any of my dependencies, accidentally has
tqdm==4.65.0
specified, no other dependency can be installed that conflicts. If another downstream repo somewhere hastqdm >= 4.66
, your entire tree breaks. This is silly. People building projects will not always list dependencies perfectly, and we need a system that can handle these imperfections.
There are many kinds of python-based repos, but two pretty common ones are:
- an SDK (ie installed by others)
- an API/Servie/Pipeline (internal, only installed by the creators).
An SDK needs to be as flexible as reasonable with their dependencies. Lots of people (hopefully) will be building with their repo, so the repo should have as loose of dependencies as can reasonably be tested. In this case, in my opinion, a lockfile is not required, and testing as wide a range of dependencies is suggested (ie testing your code with lower and upper bounds of each dependency).
For this, we have the CI run tests for 2 cases:
- Installing all dependencies at the low-end of their pins
- Installing all dependencies at the high-end of their pins
This ensures everything is as compatible as we can reasonably confirm.
NOTE: The installation of lowest
dependencies can be tricky to set up at first. Many project don't pin any lowest supported version, so it goes back to 0.0.0, or some version released in 2017 that truly isn't compatible. You'll need to add more packages to your pyproject.toml
. This is probably long-term good,
ensuring a more stable setup, but if it's causing more pain than it's worth, you can always comment it out of the matrix in ci.yaml
An API/Internal repo, that's perhaps built into a docker image and deployed, is in some ways the opposite. You don't need wide ranges, because no one else is installing it. In this case, a lockfile is critical, ensuring that the code you write and test in dev is identical to the code you run in prod.
For this, we have a lockfile that we validate on every PR to main, ensuring that no new packages have been added to the pyproject.toml
, making the lockfile out of date.
We try to keep the tools to a minimum, with exceptions for massively increasing speed or quality of life.
- Makefile: A ubiquitous tool, default installed on most systems, to ensure that everyone can run the identical commands and get set up as easily as possible
- ruff: An awesome linter, written in rust, that is so fast that you think it's lying
- uv: A python package manager, also built in rust (by the same amazing folks at @astral-sh, which makes managing python dependencies pretty easy and insanely fast. It's a rapidly growing project, moving very quickly. So it's possible that by the time you see this, they support something that would make this setup even easier.
- mypy: Standard type-safety checking for python.
- github actions: all CI/CD
There's no cd.yaml
workflow, which could either
- deploy an image
- deploy a package to pypi
- deploy a cloud function
- ...
Because those are pretty easy to find on the internet, and not specifically releavant to the goals of this repo.
See the readme in src for usage details.
I've left 2 PRs open to showcase differing behavior:
- PR that is all good. Packages are aligned, new functionality that is tested, and test coverage across a matrix of configurations: link
- PR that adds a package but doesn't update the lockfile, which causes a failed PR test and an automated comment to the PR telling the author to run
make lock
: link
This is a few steps, so if you're interested, screenshots are here.
This is entirely optional, and you can just have the test fail, without posting the PR comment, by removing this code.
I can't figure out how to do this with the Github CLI, but feel free to open a PR with those instructions:
Setup a PAT for leaving PR comments
Go to your github settings -> developer settings -> Personal Access Tokens -> Fine-grained tokens, or click here Fill in the sections with the details of your token, such as name, experiation date, etc. Pick your repo, and set the "Pull requests" permission, under "Repository Permissions" to read/write:
Click "Generate Token" at the bottom. Copy the token, because you won't see it again.
Now, go to your repo, click "Settings" at the top, then "Secrets and variables":
Then click "Actions", and then "New repository secret":
Name the secret "COMMENT_TOKEN", and paste the pat, exactly as you copied it, into the body.