Precommit

Precommit

2023, Aug 31    

Better Git Hooks

Git hooks are scripts that are trigered when some action from Git is perfomed. They can run before or after common actions such creating a commit, push or pull.

This can be used to define a local quality gate for the code that is triggered before each commit, push or pull action. Therefore, they enable catching errors ahead and preventing defective code been commited or sended to the remote server.

One can create any hook by adding a executable script at the .git/hooks stored in the repository home. But there is a more pratical for managing and maintaining Git hooks by using the Python package Pre-commit.

It uses a yaml config file placed in the repository as source, install and run all hooks.

Installing Pre-commit

It can be installed using pip using the following command:

pip install pre-commit

Basic Configuration

Once pre-commit is installed, you can create a file .pre-commit-config.yaml in the repository root with the following content:

repos:
-   repo: https://github.com/pre-commit/pre-commit-hooks
    rev: v4.4.0
    hooks:
    -   id: end-of-file-fixer
    -   id: check-yaml
    -   id: check-added-large-files
    -   id: check-merge-conflict

The hooks end-of-file-fixer, check-yaml, check-added-large-files, check-merge-conflict are already defined inside the the repository https://github.com/pre-commit/pre-commit-hooks.

In order to install them, just run the following command

pre-commit install -c .pre-commit-config.yaml

This will download the repository and install the hooks inside the .git/hooks folder. Then, the pre-commit hooks registered will run everytime that you run git commit before the commit be generated.

Using hooks with Parameters

pre-commit enable also to use some tools

For instance, if you would like to use the following command with all python files inside the a repository:

yapf --style=pep8 --parallel --in-place

This can be translated to the following config lines:

repos:
# ...
-   repo: https://github.com/google/yapf                     # yapf repository url
    rev: v0.40.0                                             # repository version to be used
    hooks:
    -   id: yapf                                             # id (used to call hook individualy)
        name: Ensure PEP-8 complience for all Python files   # short description
        args:                                                # Define arguments to be passed whenever yapf is called 
          - '--style=pep8'                                   # yapf arg: Ensure PEP-8
          - '--parallel'                                     # yapf arg: Allow run in parallel
          - '--in-place'                                     # yapf arg: edit file and fix error found
        files: \.(py)$                                       # Apply hook to all *.py files and trigger hook whenever a python file is changed

Using local commands as custom hooks

Some tools might be available only localy. In this case, you can use the repo: local combined with language: system to translate the command required by the tool to hook in the yaml config file.

For instance, the following command

pytest --junitxml=pytest.xml --cov-report=xml --cov=src -v

Can be translated to

repos:
# ...
-   repo: local
    hooks:
    -   id: pytest
        name: Ensure all tests are passing
        entry: pytest
        language: system                    # Uses entry as direct system command
        args:
          - '--junitxml=pytest.xml'
          - '--cov-report=xml'
          - '--cov=src'
          - '-v'
        files: \.(py)$                      # Trigger hook whenever a python file is changed
        always_run: true                    # This will the command will always run
        pass_filenames: false               # This will prevent the files to be passed to command since some tools might not need.

Git++

There so much more! Go ahead check the documentation page at https://pre-commit.com/ to enable even more powerful hooks.