Skip to content
/ wizard-template Public template

πŸ§™ A general-purpose template that aims to provide a magical start to any Python project

License

Notifications You must be signed in to change notification settings

fschuch/wizard-template

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 

History

40 Commits
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

The Wizard Template for Python Projects

Wizard template logo

Let the wizard do the heavy lifting so you can focus on your craft

CI/CD CI Docs CodeQL
Meta Wizard Template Checked with mypy Hatch project Ruff GitHub License EffVer Versioning

Overview

This is a general-purpose template that aims to provide a magical start to any Python project. It includes the initial configuration of quality assurance tools, documentation, and automated actions to deploy a Python package.

The template is powered by Hatch, which manages Python installations, virtual environments, dependencies, besides builds, and deploys the project to PyPI. See Why Hatch? for more details. To ensure code quality, several tools are suggested and pre-configured:

  • mypy for static type checking
  • ruff as the linter and code formatter
  • codespell to check spelling
  • pytest as the test engine

In addition, Git hooks can be used to guarantee consistency and leverage the aforementioned tools. The workflow ci.yaml runs them automatically for you.

The documentation is initialized with Jupyter Books, providing a promising approach for interactive tutorials.

You can check at any time the environments and scripts that are prepared to support your development workflow:

$ hatch env show --ascii
                                               Standalone
+---------+---------+-----------------------------+----------------------+------------------------------+
| Name    | Type    | Dependencies                | Scripts              | Description                  |
+=========+=========+=============================+======================+==============================+
| default | virtual | coverage[toml]>=7.5.3       | check                | Base development environment |
|         |         | pre-commit>=3.5.0           | format               |                              |
|         |         | pytest-cov>=5.0.0           | lint                 |                              |
|         |         | pytest>=8.2.2               | pre-commit-install   |                              |
|         |         |                             | pre-commit-uninstall |                              |
|         |         |                             | qa                   |                              |
|         |         |                             | test                 |                              |
|         |         |                             | test-no-cov          |                              |
|         |         |                             | type                 |                              |
+---------+---------+-----------------------------+----------------------+------------------------------+
| docs    | virtual | docutils==0.20.1            | build                | Documentation environment    |
|         |         | jupyter-book==1.0.0         | config               |                              |
|         |         | sphinx-autobuild==2024.4.16 | serve                |                              |
|         |         | sphinx==7.3.7               |                      |                              |
+---------+---------+-----------------------------+----------------------+------------------------------+
                                                 Matrices
+------+---------+-------------+-----------------------+----------------------+---------------------------+
| Name | Type    | Envs        | Dependencies          | Scripts              | Description               |
+======+=========+=============+=======================+======================+===========================+
| test | virtual | test.py3.9  | coverage[toml]>=7.5.3 | check                | Extended test environment |
|      |         | test.py3.10 | pre-commit>=3.5.0     | extended             |                           |
|      |         | test.py3.11 | pytest-cov>=5.0.0     | format               |                           |
|      |         | test.py3.12 | pytest-randomly       | lint                 |                           |
|      |         | test.py3.13 | pytest-rerunfailures  | pre-commit-install   |                           |
|      |         |             | pytest-xdist          | pre-commit-uninstall |                           |
|      |         |             | pytest>=8.2.2         | qa                   |                           |
|      |         |             |                       | test                 |                           |
|      |         |             |                       | test-no-cov          |                           |
|      |         |             |                       | type                 |                           |
+------+---------+-------------+-----------------------+----------------------+---------------------------+

Quick Start

Use the template as a start point for your own project

  1. Click on Use this template, creating a new project for you from it.

  2. If you don't have Hatch, download and install it following the instructions for your OS.

    • I like to keep the Python environments within the project I'm working on, Hatch can be set to do so:

      hatch config set dirs.env.virtual .venv
  3. Clone your repository and make it your working directory.

  4. Assert that everything is up and running:

    $ hatch run qa
    cmd [1] | pre-commit run  --all-files
    check for added large files..............................................Passed
    check for case conflicts.................................................Passed
    check docstring is first.................................................Passed
    check json...............................................................Passed
    check for merge conflicts................................................Passed
    check toml...............................................................Passed
    check yaml...............................................................Passed
    debug statements (python)................................................Passed
    detect private key.......................................................Passed
    fix end of files.........................................................Passed
    mixed line ending........................................................Passed
    trim trailing whitespace.................................................Passed
    ruff.....................................................................Passed
    ruff-format..............................................................Passed
    mypy.....................................................................Passed
    codespell................................................................Passed
    mdformat.................................................................Passed
    nbstripout...............................................................Passed
    cmd [2] | pytest --cov --cov-report=term
    ============================================================ test session starts =============================================================
    platform darwin -- Python 3.12.7, pytest-8.3.3, pluggy-1.5.0
    rootdir: /Users/fschuch/Documents/GitHub/wizard-template
    configfile: pyproject.toml
    plugins: cov-6.0.0
    collected 8 items
    
    src/wizard_template/core.py .                                                                                                          [ 12%]
    tests/test_core.py ......                                                                                                              [ 87%]
    tools/rename_project_content.py s                                                                                                      [100%]
    
    ---------- coverage: platform darwin, python 3.12.7-final-0 ----------
    Name                              Stmts   Miss Branch BrPart    Cover   Missing
    -------------------------------------------------------------------------------
    src/wizard_template/__init__.py       2      0      0      0  100.00%
    src/wizard_template/core.py           8      0      2      0  100.00%
    tests/__init__.py                     0      0      0      0  100.00%
    tests/test_core.py                   10      0      0      0  100.00%
    -------------------------------------------------------------------------------
    TOTAL                                20      0      2      0  100.00%
    
    Required test coverage of 90.0% reached. Total coverage: 100.00%
    
    ======================================================== 7 passed, 1 skipped in 0.05s ========================================================
    cmd [3] | echo 'βœ… QA passed'
    βœ… QA passed
    • On first invocation on the project folder, Hatch creates the virtual environments, install the dependencies, and gets ready to go.
  5. A helper script is included to rename the git username and project name from the template files to your own new project, try it with:

    hatch run _wizard
  6. Run hatch run qa another time to assert everything is still all right. You can now review the changes, stage, and commit them on your repo.

Dependencies

  1. The template is designed to work with Python 3.9 and later versions. It is recommended to use the latest stable version of Python.

  2. Project dependencies are managed on the file pyproject.toml, refer to Dependency configuration for more details on the topic.

  3. Hatch specific configuration covering the development dependencies, environments, and maintenance scripts are defined also on the file pyproject.toml. You can refer to Environment configuration for more details.

  4. To ensure quality standards on the codebase, pre-commit manages and runs the hooks configured on .pre-commit-config.yaml.

    • pre-commit handles the installation of ruff, mypy, codespell, and others, on an isolated environments.
    • It is a good pick since many of the fixes can be done automatically at commit time just on the changed files.
    • These tools are not declared as development dependencies on the project to avoid duplication.
    • The action update-pre-commits.yaml scheduled to run weekly to ensure the hooks are up-to-date.

Enforcing Code Quality

  1. As mentioned above, the pre-commit tool is used to enforce code quality standards. Even though it performs checks on the changes for every commit when installed (hatch run pre-commit-install), it is a good practice to run the checks on the whole codebase occasionally (when a new hook is added or on Pull Requests). You can do so by running hatch run check <hook-id>, for instance hatch run check nbstripout. Some of them are available as scripts as a syntax sugar, like hatch run lint, hatch run format, or hatch run type. They check the whole codebase using ruff, ruff-format, and mypy, respectively.

    • The file project.toml includes configuration for some of the tools, so they can be consumed from your IDE as well.
    • The file .pre-commit-config.yaml includes the configuration for the pre-commit hooks.
  2. The pytest test suite can be run from the default environment with hatch run test or hatch run test-no-cov (the latter without coverage check).

    Code examples on docstrings and documentation are tested by the doctest module (configured on the file pyproject.toml). It is integrated with pytest, so the previous test commands will also run the doctests.

  3. To run all the quality checks, you can use the command hatch run qa.

  4. To step up in the game, an extended test environment and the command hatch run test:extended are available to verify the package on different Python versions and under different conditions thanks to the pytest plugins:

    • pytest-randomly that randomizes the test order;
    • pytest-rerunfailures that re-runs tests to eliminate intermittent failures;
    • pytest-xdist that parallelizes the test suite and reduce runtime, to help the previous points that increase the workload;
    • The file pyproject.toml includes configuration for them.

Continuous Integration

  • The workflow ci.yaml performs the verifications on every push and pull request, and deploys the package if running from a valid tag.
  • The workflow update-pre-commits.yaml is scheduled to run weekly to ensure the pre-commit hooks are up-to-date.
  • Dependabot is enabled to keep the dependencies up-to-date (dependabot.yml).

Managing the Changelog

The template relies on Automatically generated release notes from GitHub to handle the changelog. Refer to Managing labels and add the labels from release.yml to automatically organize your entries. This is a nice fit for Managing releases in a repository.

Some may argue about the importance of Keeping a Changelog on a dedicated file, but that results in frequent conflicts and merge issues to be solved when working with feature branches in parallel. The GitHub release notes are a good compromise, as they are automatically generated and can be edited before the release. Notice good Pull Request titles and small incremental changes are key to a good changelog.

Managing the Version Number

The version in the project is set dynamically by hatch-vcs. At installation and build time, the version is recovered from the version control system and exported to the file src/wizard_template/_version.py. In this way, there is no need to keep the version hard-coded on the codebase. You can use the command hatch version to check the current version. On the deployment workflow, the version is recovered from the tag and used to build the package.

The downside is that the new version number must be manually entered for each release, rather than using the command hatch version to bump it up. However, this is a reasonable trade-off, since it does not impose any restrictions on the project's versioning scheme nor branching model.

Documentation

The template includes a documentation environment that uses Jupyter Books to provide a promising approach for interactive tutorials. The documentation source is on the docs folder and can be served locally with hatch run docs:serve, it will be available on http://127.0.0.1:8000. The documentation is also built automatically on the deployment workflow docs.yaml.

Publishing

The package can be published to PyPI in a general-purpose workflow that can be used for any branch model and versioning strategy.

The action ci.yaml is triggered to publish the package to PyPi when any valid tag is pushed to the repository. The tag matching pattern is set to v*.*.*, for instance, 1.2.3, 0.0.1rc2, 2023.2.0, etc. Notice all previous steps on CI are executed before the deployment, including static analysis and tests.

The process can also be triggered when a Release creates a new tag on your repo, so it sinergizes with the GitHub release notes cited above on the Changelog management.

Miscellaneous

VSCode Configuration

The template includes a .vscode folder with a extensions.json file that suggests the extensions to be installed on VSCode, in line with the quality assurance tools included in the template. It allows test, debug, auto-format, lint, and a few other functionalities to work directly on your IDE. It also includes a settings.json file that configures the Python extension to use the virtual environment created by Hatch. Remember to set hatch to use the virtual environment within the project folder hatch config set dirs.env.virtual .venv.

Communicate Type Annotations

PEP 561 is a guideline that explains how a Python package can show that it has inline type annotations. These annotations are useful for static type checkers, which are tools that help ensure your code has been error-free. Mypy checks for a file named py.typed in the root of the installed package.

Next steps

Copyright and License

Β© 2023 Felipe N. Schuch. All content is under MIT License.