Update (04/01/2023): A kind reddit user by the name of lanemik suggested I give ruff a go. So i decided to add it to the guide.
In the previous tutorial we have finished styling our Authentication pages. Before we go any further I would like to introduce you to an awesome developer tool that will improve your code, a lot!
Meet pre-commit.
Pre-commit is a tool that allows you to automate the process of checking your code for errors and issues before committing it to your repository, hence the name.
In simple terms, it will help you catch bugs and errors. Not only that, but it will also be able to sort your imports based on PEP recommendation as well as lint your code for better readability.
In this guide, I will show you how to set it up, and then I will share my favorite hooks that I use in all my projects.
.pre-commit-config.yaml
Before we continue, I would like to mention that you can look at all the code that we wrote in this tutorial by looking into the pre-commit PR on the basic-django repo.
All the previous tutorials have been applied to that repo too. If you have any questions or concerns, feel free to leave a comment below or on the PR. I'll try to respond as soon as I see the comment.
Finally, last comment before we begin is that you are following the whole series that you should see the same results as I will. However, if you are using a different repo, or you have wrote more code, then the pre-commit
messages that you will receive might be slightly different.
But don't worry, even if they are different, they should be clear and very actionable.
I am going to go through each step with you, but I encourage you to check out the official website. The instructions they have are very useful.
You can install it on your whole system (MacOS) with brew install pre-commit
, but we are also going to add it to our project explicitly with by running poetry add --group dev pre-commit
.
Please note, the
--group dev
, this will add the dependency to the dev section. This is useful because when we install libraries in the production setting those won't be installed, thus using less storage space and increasing installation/deployment steps.
Let's confirm that the library has been installed by running poetry run pre-commit --version
. If you don't know what this poetry run
stuff is, I recommend you check my Poetry guide.
pre-commit
Config File (^)Next, you will need to create a .pre-commit-config.yaml
file in the root of your repository. This is where the magic happens. Once you create the file add the following text to it.
repos:
- repo: https://github.com/pre-commit/pre-commit-hooks
rev: v4.4.0
hooks:
- id: check-yaml
- id: end-of-file-fixer
- id: trailing-whitespace
This file defines the pre-commit hooks that will be run. For example, in this specific case, we are using 3 hooks provided by pre-commit: check-yaml
, end-of-file-fixer
, trailing-whitespace
.
This is the structure we are going to follow when we want to add new hooks. Under the repos
array we will specify an object that has the following:
pre-commit-hooks
library you can see it in the releases section: These specific hooks are pretty self explanatory, so instead of describing each one of those, let's just run this and see what happens.
Please note, that if you are running it outside of your repo, it is not likely to do anything. You should set this up as part of some repo.
Once you have created your config file, you need to install the hooks specified in the file. You can do this by running: poetry run pre-commit install
Once this command is done, run poetry run pre-commit run --all-files
which will run these hooks against your repo. You should see something like this:
So, what happened is that pre-commit
checked all the files in the library and "fixed" them based on the hooks you have set up.
!! Warning !!
These first checks are harmless, so there is no worry about running them with --all-file.
However, later we will start adding hooks that format your code. If you have a large codebase, I don't recommend you run the pre-commit run --all-files
, since that will affect a lot of the code.
Instead, you may want to start testing that the new hooks are working by running on one file at a time like so pre-commit run --files YOUR_FILENAME
.
If you would still prefer to "fix" the whole codebase, I will recommend you do it slowly, step, by step, like I do in this tutorial.
In the future you will not be using poetry run pre-commit run --all-files
, instead pre-commit will automatically on all the files that you are committing to your repo.
Here is the general workflow that will now happen:
git add --all
(doesn't have to be --all
you could add files one by one, whatever is your preference)git commit -m "adding new feature"
. At that stage pre-commit starts. If all the checks have Passed, then the commit will go through and you can then run git push
. If, on the other hand, at least one check fails commit will not go through.git add --all
and re-try running the git commit -m "adding new feature"
. This time, if all things have been fixed the commit will go through.I know that this can sound a little tedious or slow. But once you go through this a couple of times, you will realize how quick, easy and useful this is.
Now that the setup is done, let go through some of my favorite hooks that I use in almost all my projects. The first three you have already seen, I do use them everywhere.
Black is a Python code formatter. This means that when this hook will run, all the files that you want to commit will be checked for any inconsistencies and bad styling (based on PEP 8 standard). This hook will automatically fix those issues.
Before we add this hook, let's do something first. Add exclude: ^migrations/
to the top of your pre-commit
config file. So that it will look like this:
# .pre-commit-config.yaml
exclude: .*migrations\/.*
repos:
- repo:
...
This will make sure that the files that Django migration generates are not touched. You don't have to do that. I, personally, I think it makes sense not to touch files that are autogenerated by Django.
Alright, now that this is out of the way, let's add black
to pre-commit. All you have to do is add the following lines to the pre-commit config file, right after the first "repo":
exclude: .*migrations\/.*
repos:
- repo: https://github.com/pre-commit/pre-commit-hooks
...
- repo: https://github.com/psf/black
rev: 22.12.0
hooks:
- id: black
language_version: python3.9
Keep in mind that at the time that you are reading, black
might have a newer version available. You should go to the repo and check the latest available version (just like described above).
Another thing to keep in mind is that you should change the language version depending on what you are targeting. For example, when setting up the basic-django
repo we specified that this project will be version 3.9.
Optional
If you are using poetry in your project, like I described in a previous post you can also add another layer of configuration. In you pyproject.toml
file you can add the following block:
[tool.black]
line-length = 120
target-version = ['py39']
include = '\.pyi?$'
Optional, continued
Black will you these configurations when being ran by pre-commit. To see the full list of available configurations, check out the official docs page.
Once you've added it, you can test by running poetry run pre-commit run --all-files
.
You should see something like this:
If you do, congratulations!!! 8 files were formatted to adhere to the PEP-8 standard 🤘
If something went wrong, please comment below, I'll take a look ASAP.
We can move on to other hooks.
isort is an awesome tool that will sort imports across your repo.
You know the drill. Let's add this hook to the pre-commit config file. It will look like so:
repos:
... pre-commmit stuff
... black stuff
- repo: https://github.com/pycqa/isort
rev: 5.12.0
hooks:
- id: isort
name: isort (python)
And optionally, but very much recommended add these configurations to the pyproject.toml
file:
[tool.isort]
profile = "django"
combine_as_imports = true
include_trailing_comma = true
line_length = 120
Let's go over at what these do.
1. isort has profiles, which let you choose what type of repo you have. This will slightly change the behaviour of the check.
2. The next too are something that isort recommends for the django profile. combine_as_imports
will that imports with as
statement are combined and the include_trailing_comma
will keep the comma after the last import in a statement that imports multiple items.
3. Next is the line length, which is important. I ran into this problem a few times. Make sure that this number is the same as the one configure in black. If that is not the case, often times, these two will conflict with each other, correcting each others behavior.
Let's try running pre-commit again with poetry run pre-commit run --all-files
. You should see something like this:
Please note,that black passed the test, since we habe already run it before, which is cool. Also, interesting to see the fix end of files
hook being triggered. Probably I added extra line in the pre-commit config file. Finally I can see that 2 files have been fixed by isort
, awesome.
Let's move on!
flake8 is yet another awesome tool that will help you check the style and quality of some python code.
This one is going to be a liiiitle bit different. Instead configuring the behaviour in the pyproject.toml file like before we will create a separate file for it.
But first, let's add to the pre-commit file first. As before check the latest version that is available in the github repo.
...
- repo: https://github.com/pycqa/flake8
rev: 6.0.0
hooks:
- id: flake8
Note create a .flake8
file at the root of your repository and add the following to it:
[flake8]
max-line-length=120
Now, let's try running it with poetry run pre-commit run --all-files
. Here is what I got:
Awesome. Couple of things to note.
First, I did not talk about this before, but you can see in the first lines where pre-commit
is setting up the environment for flake8 (essentially checking if the hooks exist). If something went wrong at this stage, then it is likely that you entered url incorrect or that you added a non-existent version.
Second, is that this plugin doesn't update the file for us. It just tells us what is currently wrong so that I can go and fix it myself before committing the new code, which is fantastic. Let's do exactly that.
First, it doesn't like that the HTML string in the utils.py
file is too long. Black did not make it smaller, because it doesn't know how to operate on strings. There are two approaches we can take:
1. Fix the line length by breaking up the svg code into multiple lines.
2. Tell flake that this exact case is fine.
I'm going to opt for #2, since I don't care about the readability of the svg tag (no one actually "inspects" those). Plus, would be good to show how to tell packages to ignore some lines.
I'm going to put the following comment # noqa: E501
on lines 16 and 17. The E501
comes from the message in the terminal. Same for the lines, flake8 tells me which lines are affected. After adding this comment to both lines, let's rerun the pre-commit
command.
Et Voila! This specific message is gone. Now let's fix others. Looks like it is about unused imports. That should be easy to fix. Let's remove those import and rerun.
Hooray! Let's move on to pylint.
pylint is a linter and a static code analyzer just like flake8. It will check your code for any formatting issue as well as any performance issues. The difference is that pylint is a little more thorough and more customizable.
I feel like those two complement each other. You don't have to set both of them up, but I sleep better when I know that two unrelated programs checked my code 🤣
Set up for pylint is a little different. Here is a quote from the official docs:
Since
pylint
needs to import modules and dependencies to work correctly, the hook only works with a local installation ofpylint
(in your environment).
So, we need to install pylint into our repo with poetry add --group dev pylint
.
Once pylint
was installed we will need to tell pre-commit
to use that specific version. Again, let's check the docs for that. They recommend we do it that way (I removed a couple of args):
...
- repo: local
hooks:
- id: pylint
name: pylint
entry: poetry run pylint
language: system
types: [python]
args:
[
"-rn", # Only display messages
"-sn", # Don't display the score
]
Note the change I made on the entry
line. Instead of running pylint
, we are telling pre-commit
to run poetry run pylint
, since that is the preferred way to invoke packages installed with poetry.
One last thing to do before running the hooks is to create a config file, just like we did with flake8
. For this you are going to create a pylintrc
file at the roor of your project and copy the contents of the pylintrc
file from the pylint
repo (here is the link to it).
These are the recommended setting for pylint that we can configure in the future if we want.
Let's run poetry run pre-commit run --all-files
to see what happens.
Here is what I got.
Good, that means it's working. Before going further, I would like to show you another cool feature of pylint. Extensions.
There is a Django extension that we can install and apply to the pylint check. Let's do that.
Install with poetry add --group dev pylint-django
. And add the 2 new lines to the args list, like so:
...
args:
[
"-rn",
"-sn",
"--load-plugins=pylint_django", # new
"--django-settings-module={NAME OF YOUR REPO}.settings", # new
]
Let's run poetry run pre-commit run --all-files
once again, to make sure everything is working correctly.
Looks like pylint found the same 3 errors. I'm not going to go over them one by one, since they are very detailed, and more importantly, you might be seeing something else.
The only note I'll make is that I will add
--ignore=manage.py
to the list of args, since this was generated by Django, and I don't want to lint that.
So, let's use the magic of reading and see what it looks like after I modified the code to the pylint's liking.
Nice. Another useful check added to the list.
I will tell you this... In my opinion, pylint is the hardest and the most annoying to setup and "please" in the future. However, it doesn't mean that you should do it. I think all (most) the error messages that it is showing are useful.
If you encounter those error, at setup or at commit don't worry. Spend some time trying to fix them, or googling about them. This will make you a better dev for sure.
Comment below, if something is not working for you. I'll try to help.
For Django users, which you presumably are, this will be a God send! Check out the repo. This is an HTML linter... but for Django templates 🤯
It will look for errors and inconsistencies in your HTML files. The setup for this one is straightforward.
Add this to the pre-commit-config file:
- repo: https://github.com/Riverside-Healthcare/djLint
rev: v1.19.16
hooks:
- id: djlint-django
Also, I will add the following conf to the pyproject.toml
file:
[tool.djlint]
profile="django"
For all the conf options, check out the official docs
One thing to note is that I would recommend you put this "repo", before the "local" one. It is preferable to keep the "local" one last. So, in our case we would add this one right after flake8
one.
And, as before make sure you are using the latest version. Let's give this a run with poetry run pre-commit run --all-files
.
Oof, there is a lot to fix. Thankfully, errors are pretty clear.
I've included the following ignores to my conf file:
[tool.djlint]
profile="django"
ignore = "H031"
Added H031 because keywords meta is no longer used. Everything else, fixed.
Final one that I like to use is the poetry export
hook. It used to be the case that we needed to add this one to the local repo, but no longer.
This is useful if you don't want to install poetry on your prod server or in your Dockerfile. That is to say very useful.
Instead of dealing with poetry in Docker or prod we will just have a nice and fresh "requirements.txt" file to use for out dependencies. Beautiful.
To make this work add the following to your pre-commit config file. I put it right after the djlint hook.
- repo: https://github.com/python-poetry/poetry
rev: '1.4.1'
hooks:
- id: poetry-export
args: [
"-f", "requirements.txt",
"-o", "requirements.txt",
"--without-hashes"
]
This one doesn't require any configurations ♥️. To see other poetry hooks, check out their docs.
A kind reddit user by the name of lanemik suggested I give ruff a go. I have heard of this tool before, but never actually gave it a go.
According to the README
Ruff can be used to replace Flake8 (plus dozens of plugins), isort, pydocstyle, yesqa, eradicate, pyupgrade, and autoflake, all while executing tens or hundreds of times faster than any individual tool.
I'm not going to replace isort and Flake8 for the purpose of this tutorial, but I will add ruff above those two to see it's performance.
Let's add the following block to the pre-commit config file:
- repo: https://github.com/charliermarsh/ruff-pre-commit
rev: 'v0.0.260'
hooks:
- id: ruff
I will also add the following configuration to the pyproject.toml
:
[tool.ruff]
line-length = 120
, but there are a ton of other things you can configure. Check out the README for that.
Now, let's run the poetry run pre-commit run --all-files
and see what happens.
I got the same error as in the flake8 about the long HTML lines in the utils.py. The syntax for ignoring error is the same as in Flake8. The only difference is that here I had to add "# noqa: E501" at the end of a multi-line string, as opposed to the end the line. Once added at the end of the string I can remove noqa
comments on lines 16 & 17. I personally think that looks nicer.
That's it. As easy as that.
Aaaah, it's hard to explain the joy I get from seeing all those "Passed" messages.
In conclusion, here is what we did:
- Install pre-commit as a dev dependency in you poetry project.
- Installed a bunch of pre-commit hooks to the pre-commit config file.
- Configured the behavior of new hooks through pyproject.toml
configurations or through rc file configurations.
Can't believe it took more than 3000 words to go over these 3 bullet points 🤷♂️. If you made it to here, a salute you 🖖, you are an incredible human being with a ton of patience and focus! You will do good in life.
If you have any questions or comments, please them leave below.