General-purpose Git tag-based version manager that works with any language and any build system.
- ๐งโ๐ณ Highly customisable: It's an easy-to-read script. Literally a simple Python script in which you can customise the version format or anything you need.
- ๐ Runs with Python 3.8+
- โ๐ฆ No dependencies like package, config file etc. It runs with one Python file.
- โญ Works with any build backend with hooks. (Supports setuptools, hatchling, pdm)
- ๐ฆ Works with any language, not just Python.
- Version format
"digits"
generates digits-only version string which is useful for multi-language projects, Chrome Extension, etc. because their versioning standard is different. - CLI makes it easy to compute the version without vendoring anything in the project.
- Version format
- ๐งโโ๏ธ Auto-magically infer version even when the git info is missing.
- Downloaded from GitHub Releases? Read from the directory name.
- The
parentdir_prefix
is automatically resolved frompyproject.toml
's source URL etc.
- The
- sdist built without writing a resolved versionfile?
- Read from PKG-INFO.
- Downloaded from GitHub Releases? Read from the directory name.
- ๐ข New version formats:
"pep440-master"
: shows the distance from the tag to master/main, and the master to the current branch. (e.g. 1.2.3โ+4.gxxxxxxxโ.5.gxxxxxxx )"digits"
: the distance and dirty information compiled to the last digit. (e.g. 1.2.3โ.4)
- </> API provided for complete non-vendored mode support.
- With Versioneer you still had to install a
_version.py
script in your project, but Version-Pioneer is able to be installed as a package.
- With Versioneer you still had to install a
- ๐ป CLI tool to get version string, execute the
_version.py
versionscript, and test your setup.
-
Configure
pyproject.toml
.[tool.version-pioneer]
section is required.[tool.version-pioneer] versionscript = "src/my_project/_version.py" # Where to "read" the Version-Pioneer script (to execute `get_version_dict()`). versionfile-sdist = "src/my_project/_version.py" # Where to "write" the version dict for sdist. versionfile-wheel = "my_project/_version.py" # Where to "write" the version dict for wheel.
-
Create
src/my_project/_version.py
withget_version_dict()
in your project.# Example _version.py, completely non-vendored. from pathlib import Path from version_pioneer.api import get_version_dict_wo_exec def get_version_dict(): # NOTE: during installation, __file__ is not defined # When installed in editable mode, __file__ is defined # When installed in standard mode (when built), this file is replaced to a compiled versionfile. if "__file__" in globals(): cwd = Path(__file__).parent else: cwd = Path.cwd() return get_version_dict_wo_exec( cwd=cwd, style="pep440", tag_prefix="v", )
-
Put the following code in your project's
__init__.py
to use the version string.# src/my_project/__init__.py from ._version import get_version_dict __version__ = get_version_dict()["version"]
Tip
Use version-pioneer install --no-vendor
CLI command to perform the step 2 and 3 automatically.
- Configure your build backend to execute
_version.py
and use the version string. Setuptools, Hatchling and PDM are supported.
๐ฆ Setuptools:
# append to pyproject.toml
[build-system]
requires = ["setuptools", "version-pioneer"]
build-backend = "setuptools.build_meta"
setup.py
:
from setuptools import setup
from version_pioneer.build.setuptools import get_cmdclass, get_version
setup(
version=get_version(),
cmdclass=get_cmdclass(),
)
๐ฅ Hatchling:
# append to pyproject.toml
[build-system]
requires = ["hatchling", "version-pioneer"]
build-backend = "hatchling.build"
[tool.hatch.version]
source = "version-pioneer"
[tool.hatch.build.hooks.version-pioneer]
# section is empty because we read config from `[tool.version-pioneer]` section.
PDM:
# append to pyproject.toml
[build-system]
requires = ["pdm-backend", "version-pioneer"]
build-backend = "pdm.backend"
Voilร ! The version string is now dynamically generated from git tags, and the _version.py
file is replaced with a constant "versionfile" when building a wheel or source distribution.
Tip
The _version.py
gets replaced to a constant version file when you build your package, so version-pioneer
shouldn't be in your package dependencies.
Instead, you may put it as a "dev dependency" in your pyproject.toml
.
[project.optional-dependencies]
dev = ["version-pioneer"]
Your package could be installed with pip install -e '.[dev]'
for development.
If you don't want to add a dev dependency, you can simply vendor the "versionscript" in your project.
Copy-paste the entire versionscript.py
to your project, use it as is or customise it to your needs.
If you choose to modify the script, remember one rule: the versionscript file must contain get_version_dict()
function that returns a dictionary with a "version" key. (more precisely, the VersionDict
type in the script.)
# Valid _version.py
def get_version_dict():
# Your custom logic to get the version string.
return { "version": version, ... }
Tip
Use version-pioneer install
or version-pioneer print-versionscript-code
CLI commands that helps you install (vendor) the versionscript.py
file to your project.
Unlike Versioneer, the configuration is located in two places: pyproject.toml
and the "versionscript" (src/my_project/_version.py
). This is to make it less confusing, because in Versioneer, most of the pyproject.toml config were actually useless once you install versionscript.py
in your project.
The idea is that the toml config just tells you where the script is (for build backends to identify them), and the script has everything it needs.
versionscript
: Path to the versionscript to executeget_version_dict()
. (e.g.src/my_project/_version.py
)versionfile-sdist
: Path to save the resolved versionfile in the sdist build directory (e.g.src/my_project/_version.py
)versionfile-wheel
: Path to save the resolved versionfile in the wheel build directory (e.g.my_project/_version.py
)
The main idea is that when you build your project, "versionscript" is executed to write the "versionfile".
When you build a source distribution (sdist), the versionfile-sdist
gets replaced to a short constant file.
When you build a wheel, the versionfile-wheel
gets replaced to a short constant file.
Tip
Leave out the versionfile-sdist
and/or versionfile-wheel
setting if you don't want to write/replace the versionfile in the build directory.
You can modify the config in the script.
@dataclass(frozen=True)
class VersionPioneerConfig:
style: VersionStyle = VersionStyle.pep440
tag_prefix: str = "v"
parentdir_prefix: Optional[str] = None
verbose: bool = False
style
: similar to Versioneer'sstyle
option. Three major styles are:pep440
: "1.2.3+4.gxxxxxxx.dirty" (default)pep440-master
: "1.2.3+4.gxxxxxxx.5.gxxxxxxx.dirty"- Shows the distance from the tag to master/main, and the master to the current branch.
- Useful when you mainly work on a branch and merge to master/main.
digits
: "1.2.3.5"- Digits-only version string.
- The last number is the distance from the tag (dirty is counted as 1, thus 5 in this example).
- Useful for multi-language projects, Chrome Extension, etc.
- See Versioneer for more styles (or read documentation in _version.py).
tag_prefix
: tag to look for in git for the reference version.parentdir_prefix
: if there is no .git, like it's a source tarball downloaded from GitHub Releases, find version from the name of the parent directory. e.g. setting it to "github-repo-name-" will find the version from "github-repo-name-1.2.3"- ๐งโโ๏ธ Set to None to auto-magically infer from pyproject.toml's GitHub/GitLab URL or project name. (New in Version-Pioneer)
verbose
: print debug messages.
If you want to customise the logic, you can modify the entire script. However you modify the script, remember that this file has to be able to run like a standalone script without any other dependencies (like package, files, config, etc.).
This section explains how Version-Pioneer works, so you can customise it to your needs.
The core functionality is in one file: versionscript.py
. This code is either used as a script (python versionscript.py
) that prints a json of all useful information, or imported as a module (from my_project.versionscript import get_version_dict
), depending on your needs.
Run it in your project to see what it prints. Change git tags, commit, and see how it changes.
$ git tag v1.2.3
$ python versionscript.py
{"version": "1.2.3", "full_revisionid": "xxxxxx", "dirty": False, "error": None, "date": "2024-12-17T12:25:42+0900"}
$ git commit --allow-empty -m "commit"
$ python versionscript.py
{"version": "1.2.3+1.gxxxxxxx", "full_revisionid": "xxxxxx", "dirty": True, "error": None, "date": "2024-12-17T12:25:42+0900"}
You lose the git history during build, so you need to convert the versionscript.py
to a constant version string.
Just exec
the original versionscript.py
and save the result as you wish: text, json, etc.
# code to evaluate get_version_dict() from the version script
Path("src/my_project/_version.py").read_text()
module_globals = {}
exec(version_py, module_globals)
print(module_globals["get_version_dict"]())
Place versionscript.py
in your project source directory (like src/my_project/_version.py
). When you install your package like pip install -e .
, the code is unchanged, so it will always print up-to-date version string from git tags.
However, if you install like pip install .
or pyproject-build
, uv build
etc., you would lose the git history so the src/my_project/_version.py
should change.
The original file is replaced with this. This is generated by literally executing the above file and saving version_dict as a constant.
# pseudo code of _version.py "versionfile", generated.
def get_version_dict():
return { "version": "0.3.2+15.g2127fd3.dirty", "full_revisionid": "2127fd373d14ed5ded497fc18ac1c1b667f93a7d", "dirty": True, "error": None, "date": "2024-12-17T12:25:42+0900" }
Even if you are not familiar with Hatchling, hear me out. It is very straightforward.
Add hatchling configuration to pyproject.toml
.
Note
In this tutorial, we're assuming that versionscript
== versionfile-sdist
for the sake of simplicity.
This will replace the _version.py itself.
If you want to keep the original versionscript.py (different versionfile-sdist
), first exec versionfile-sdist
if it exists, otherwise exec versionscript
.
The reason is that once sdist is built, the version should have been already evaluated and the git information is removed, so versionfile-sdist
must take precedence.
[build-system]
requires = ["hatchling", "tomli ; python_version < '3.11'"]
build-backend = "hatchling.build"
# We assume versionscript == versionfile-sdist thus we can use what hatchling provides, and we don't need a metadata hook.
[tool.hatch.version]
source = "code"
path = "src/my_project/_version.py"
expression = "get_version_dict()['version']"
[tool.hatch.build.hooks.custom]
path = "hatch_build.py"
[tool.version-pioneer]
versionscript = "src/my_project/_version.py"
versionfile-sdist = "src/my_project/_version.py"
versionfile-wheel = "my_project/_version.py"
[project]
name = "my-project"
dynamic = ["version"]
Add hatch_build.py
to the project root.
from __future__ import annotations
import sys
import tempfile
import textwrap
from os import PathLike
from pathlib import Path
from typing import Any
from hatchling.builders.hooks.plugin.interface import BuildHookInterface
if sys.version_info < (3, 11):
import tomli as tomllib
else:
import tomllib
def load_toml(file: str | PathLike) -> dict[str, Any]:
with open(file, "rb") as f:
return tomllib.load(f)
class CustomPioneerBuildHook(BuildHookInterface):
def initialize(self, version: str, build_data: dict[str, Any]) -> None:
self.temp_versionfile = None
if version == "editable":
return
pyproject_toml = load_toml(Path(self.root) / "pyproject.toml")
# evaluate the original versionscript.py file to get the computed versionfile
versionscript = Path(
pyproject_toml["tool"]["version-pioneer"]["versionscript"]
)
version_py = versionscript.read_text()
module_globals = {}
exec(version_py, module_globals)
version_dict = module_globals["get_version_dict"]()
# replace the file with the constant version
# NOTE: Setting delete=True will delete too early on Windows
self.temp_versionfile = tempfile.NamedTemporaryFile(mode="w", delete=False) # noqa: SIM115
self.temp_versionfile.write(
textwrap.dedent(f"""
# THIS "versionfile" IS GENERATED BY version-pioneer
# by evaluating the original versionscript and storing the computed versions as a constant.
def get_version_dict():
return {version_dict}
""").strip()
)
self.temp_versionfile.flush()
build_data["force_include"][self.temp_versionfile.name] = Path(
pyproject_toml["tool"]["version-pioneer"]["versionfile-sdist"]
)
def finalize(
self,
version: str,
build_data: dict[str, Any],
artifact_path: str,
) -> None:
if self.temp_versionfile is not None:
# Delete the temporary version file
self.temp_versionfile.close()
Path(self.temp_versionfile.name).unlink()
It just replaces the _version.py
"versionscript" with a constant "versionfile", by executing the versionscript.
This is skipped when the project is installed in editable mode (pip install -e .
).
Now you can install your package with pip install .
, pip install -e .
, or build a wheel with hatch build
, pyproject-build
(python -m build
), or uv build
.
Important
Validate if uv build --sdist
, uv build --wheel
produces the same result as uv build
(both sdist and wheel are built at the same time).
We provide a CLI command version-pioneer build-consistency-test
to help you with this.
The idea is the same, but the PDM doesn't really evaluate a code to get a version string (or maybe it doesn't work in this case). So we do both in the hook.
๐ pyproject.toml:
requires = ["pdm-backend"]
build-backend = "pdm.backend"
[tool.pdm.build]
custom-hook = "pdm_build.py"
[tool.version-pioneer]
versionscript = "src/my_project/_version.py"
versionfile-sdist = "src/my_project/_version.py"
versionfile-wheel = "my_project/_version.py"
[project]
name = "my-project"
dynamic = ["version"]
๐ pdm_build.py:
import textwrap
from pathlib import Path
from pdm.backend.hooks.base import Context
def pdm_build_initialize(context: Context):
# Update metadata version
versionscript = Path(
context.config.data["tool"]["version-pioneer"]["versionscript"]
)
versionscript_code = versionscript.read_text()
version_module_globals = {}
exec(versionscript_code, version_module_globals)
version_dict = version_module_globals["get_version_dict"]()
context.config.metadata["version"] = version_dict["version"]
# Write the static version file
if context.target != "editable":
if context.target == "wheel":
versionscript = context.config.data["tool"]["version-pioneer"][
"versionfile-wheel"
]
context.ensure_build_dir()
versionscript = context.build_dir / Path(versionscript)
versionscript.parent.mkdir(parents=True, exist_ok=True)
versionscript.write_text(
textwrap.dedent(f"""
# THIS "versionfile" IS GENERATED BY version-pioneer
# by evaluating the original versionscript and storing the computed versions as a constant.
def get_version_dict():
return {version_dict}
""").strip()
)
The above usage should be completely fine, but we also provide a CLI tool to help you install and evaluate versionscript.py.
# Install with pip
pip install 'version-pioneer[cli]'
# Install with uv tool (in a separate environment, just for the CLI)
uv tool install 'version-pioneer[cli]'
$ version-pioneer
Usage: version-pioneer [OPTIONS] COMMAND [ARGS]...
๐ง Version-Pioneer: Dynamically manage project version with hatchling and pdm support.
โญโ Commands โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฎ
โ install Add _version.py, modify __init__.py and maybe setup.py. โ
โ print-versionscript-code Print the content of _version.py (versionscript.py) file (for manual installation). โ
โ exec-versionscript Resolve the _version.py file for build, and print the content. โ
โ get-version-wo-exec WITHOUT evaluating the _version.py file, get version from VCS with built-in Version-Pioneer logic. โ
โ build-consistency-test Check if builds are consistent with sdist, wheel, both, sdist -> sdist. โ
โฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ
- Configure
pyproject.toml
with[tool.version-pioneer]
section.
[tool.version-pioneer]
versionscript = "src/my_project/_version.py"
versionfile-sdist = "src/my_project/_version.py"
versionfile-wheel = "my_project/_version.py"
version-pioneer install
will copy-paste theversionscript.py
to theversionscript
path you specified, and define__version__
to your__init__.py
.
If you are using setuptools backend, it will also create a setup.py
file for you.
You can set --no-vendor
option to import version_pioneer
as a module to reduce the boilerplate code in your project. (This adds a dev dependency to your project.)
Examples:
$ version-pioneer exec-versionscript --output-format version-string
0.1.0+8.g6228bc4.dirty
$ version-pioneer exec-versionscript --output-format json
{"version": "0.1.0+8.g6228bc4.dirty", "full_revisionid": "6228bc46e14cfc4e238e652e56ccbf3f2cb1e91f", "dirty": true, "error": null, "date": "2024-12-21T21:03:48+0900"}
$ version-pioneer exec-versionscript --output-format python
#!/usr/bin/env python3
# THIS "versionfile" IS GENERATED BY version-pioneer-0.1.0
# by evaluating the original versionscript and storing the computed versions as a constant.
def get_version_dict():
return {'version': '0.1.0+8.g6228bc4.dirty', 'full_revisionid': '6228bc46e14cfc4e238e652e56ccbf3f2cb1e91f', 'dirty': True, 'error': None, 'date': '2024-12-21T21:03:48+0900'}
if __name__ == "__main__":
import json
print(json.dumps(__version_dict__))
This is useful when you want to get the version string without evaluating the versionscript file, like your project is probably not Python.
It's the same as running the versionscript.py
script (unchanged, not the vendored one), but with more options.
$ version-pioneer get-version-wo-exec
0.1.0+8.g6228bc4.dirty
$ version-pioneer get-version-wo-exec --output-format json
{"version": "0.1.0+8.g6228bc4.dirty", "full_revisionid": "6228bc46e14cfc4e238e652e56ccbf3f2cb1e91f", "dirty": true, "error": null, "date": "2024-12-21T21:03:48+0900"}
$ version-pioneer get-version-wo-exec --style digits
0.1.0.9
Useful to check if you have configured Version-Pioneer correctly. It builds the project with uv build
, uv build --sdist
, uv build --wheel
, and checks if the version strings and the package content are consistent. Also it builds sdist from sdist and perform the check.
$ version-pioneer build-consistency-test
06:35:25 INFO version_pioneer - Running with version-pioneer 0.1.0+41.g8b148ed.dirty __init__.py:202
INFO version_pioneer.api - Testing build consistency... api.py:212
INFO version_pioneer.api - Changing cwd to /Users/kiyoon/project/version-pioneer api.py:220
INFO version_pioneer.api - Building the project with `uv build` api.py:226
06:35:26 INFO version_pioneer.api - Building the project with `uv build --sdist` and `uv build --wheel` api.py:235
06:35:27 SUCCESS version_pioneer.api - โ
2 wheel builds are consistent. api.py:272
SUCCESS version_pioneer.api - โ
2 sdist builds are consistent. api.py:288
INFO version_pioneer.api - Building the project with `uv build --sdist` using the built sdist api.py:290
(chaining test).
INFO version_pioneer.api - Changing cwd to the built sdist directory: api.py:294
/var/folders/r5/9cpfjfjx3b73b6stl7_w712h0000gn/T/tmpzuq_uwdn/dist/version_pioneer-0.1.0+4
1.g8b148ed.dirty
SUCCESS version_pioneer.api - โ
Chained sdist builds are consistent. api.py:324
INFO version_pioneer.api - Build wheel using the sdist. api.py:327
06:35:28 SUCCESS version_pioneer.api - โ
sdist -> wheel chained build is consistent with the non-chained api.py:346
build.
INFO version_pioneer.api - Deleting temporary directory api.py:351
/var/folders/r5/9cpfjfjx3b73b6stl7_w712h0000gn/T/tmpzuq_uwdn
SUCCESS version_pioneer.api - ๐ All tests passed! 3 sdist builds and 3 wheel builds are api.py:354
consistent.
- Only supports git.
git archive
is not supported. Original Versioneer uses.gitattributes
to tell git to replace some strings in_version.py
when archiving. But this is not enough information (at least in my case) and the version string always becomes0+unknown
. So I dropped it.
It's good to note that, chaining building (project -> sdist -> sdist -> wheel) may result in different version strings if not configured correctly. We take the following strategy to make it consistent:
versionfile-sdist
is evaluated first, if it exists.
Most of the time your versionscript
and versionfile-sdist
would be the same. But for some reason you choose to have a seaparate file,
and imagine if we execute the versionscript again in a built sdist. It may produce a different version string because now we don't have git information.
Therefore, versionfile-sdist
takes precedence (if it exists) over versionscript
, for resolving version.
- Each backend works differently under the hood. Some things to note:
Setuptools:
- If
setup.cfg
doesn't exist, the sdist build will generate the file.- Thus, if you build sdist from sdist, the
*.egg-info/SOURCES.txt
will containsetup.cfg
so the result is slightly different.
- Thus, if you build sdist from sdist, the
version_pioneer.build.setuptools.get_version()
finds the PKG-INFO to look up the version.- It's the function you used in
setup(version=get_version())
. - Building from sdist wouldn't look at git tags, but the
PKG-INFO
file. So the version string is consistent after multiple builds.
- It's the function you used in
Hatchling:
- Once sdist is built, the PKG-INFO is present, and hatchling's version source plugin is ignored.
versionfile-wheel
doesn't really get used, but I would still configure it for consistency.
PDM Backend:
- Building with pdm removes
dynamic = ["version"]
frompyproject.toml
's[project]
section.- Instead, the
version="0.1.0"
(whatever it is during the build) is written.
- Instead, the
- However, the build hook can still change the metadata version, thus the versionfile / versionscript is still executed.
- It will be the versionfile that is already resolved, so the version string is consistent.
Versioneer finds the closest git tag like v1.2.3
and generates a version string like 1.2.3+4.gxxxxxxx.dirty
.
1.2.3
is the closest git tag.+4
is the number of commits since the tag.gxxxxxxx
is the git commit hash (without the leadingg
)..dirty
is appended if the working directory is dirty (i.e. has uncommitted changes).
setuptools-scm is a similar tool, but with some differences:
- How the version string is rendered:
1.2.3+4.gxxxxxxx.dirty
vs1.2.4.dev4+gxxxxxxx
- No
.dirty
in setuptools-scm. - Infer the next version number (i.e. 1.2.4 instead of 1.2.3).
- No
- The
_version.py
file is always a constant in setuptools-scm.- Versioneer can dynamically generate the version string at runtime, so it's always up-to-date. Useful for development (pip install -e .).
- Setuptools-scm won't ever change the version string after installation. You need to reinstall to update the version string.
I have used versioneer for years, and I like the format and dynamic resolution of versions for development. However,
- It doesn't support any build backends other than
setuptools
(likepdm
,hatchling
,poetry
,maturin
,scikit-build
, etc.) - It doesn't support projects that are not Python (like Rust, Chrome Extension, etc.).
Every time I had to figure out how to integrate a new VCS versioning plugin but they all work differently and produce different version strings. GitHub Actions and other tools may not work with all different version format. Different language usually expects different format, and it's especially hard to make it compatible for mixed language projects.
The original versioneer is 99% boilerplate code to make it work with all legacy setuptools configurations, trying to "generate" code depending on the configuration, etc.. But the core functionality is simple: just get version from git tag and format it. I had to leverage this logic to integrate Versioneer in every project I had.
Run tests:
# install uv (brew install uv, pip install uv, ...)
uv pip install deps/requirements-dev.in
pytest
uv
is required to run tests because we use uv build
.
Types of tests:
- install with setuptools, hatchling, pdm
- version after tag, commit, dirty
- invalid version-pioneer config
- build with
uv build
,uv build --sdist
,uv build --wheel
- Important: all three can produce different results if sdist isn't generated correctly from the first place.
- When building both at the same time (
uv build
), it seems to make sdist first and then wheel from the sdist (not directly from the source dir). - If the sdist doesn't contain resolved
_version.py
, the wheel build will get no version, because git information is gone.