Skip to main content

Plugin for Poetry to enable dynamic versioning based on VCS tags

Project description

Dynamic versioning plugin for Poetry

Build status Version License: MIT

This package is a plugin for Poetry to enable dynamic versioning based on tags in your version control system, powered by Dunamai.

Since Poetry does not yet officially support plugins (refer to this issue) as of the time of writing on 2019-10-19, this package takes some novel liberties to make the functionality possible. As soon as official support lands, this plugin will be updated to do things the official way.

Installation

Python 3.5 or newer and Poetry 1.0.1 or newer are required.

  • Run pip install poetry-dynamic-versioning
  • Add this to your pyproject.toml:
    [tool.poetry-dynamic-versioning]
    enable = true
    

Note that you must install the plugin in your global Python installation, not as a dependency in pyroject.toml, because the virtual environment that Poetry creates cannot see Poetry itself and therefore cannot patch it.

With the minimal configuration above, the plugin will automatically take effect when you run commands such as poetry build. It will update the version in pyproject.toml, then revert the change when the plugin deactivates. If you want to include a __version__ variable in your code, just put a placeholder in the appropriate file and configure the plugin to update it (see below) if it isn't one of the defaults. You are encouraged to use __version__ = "0.0.0" as a standard placeholder.

Configuration

In your pyproject.toml file, you may configure the following options:

  • [tool.poetry-dynamic-versioning]: General options.
    • enable: Boolean. Default: false. Since the plugin has to be installed globally, this setting is an opt-in per project. This setting will likely be removed once plugins are officially supported.

    • vcs: String. This is the version control system to check for a version. One of: any (default), git, mercurial, darcs, bazaar, subversion, fossil.

    • metadata: Boolean. Default: unset. If true, include the commit hash in the version, and also include a dirty flag if dirty is true. If unset, metadata will only be included if you are on a commit without a version tag.

    • dirty: Boolean. Default: false. If true, include a dirty flag in the metadata, indicating whether there are any uncommitted changes.

    • pattern: String. This is a regular expression which will be used to find a tag representing a version. There must be a named capture group base with the main part of the version, and optionally you can also have groups named stage and revision for prereleases. The default is ^v(?P<base>\d+\.\d+\.\d+)(-?((?P<stage>[a-zA-Z]+)\.?(?P<revision>\d+)?))?$.

    • format: String. Default: unset. This defines a custom output format for the version. Available substitutions:

      • {base}
      • {stage}
      • {revision}
      • {distance}
      • {commit}
      • {dirty}

      Example: v{base}+{distance}.{commit}

    • format-jinja: String. Default: unset. This defines a custom output format for the version, using a Jinja template. When this is set, format is ignored.

      Available variables:

      • base (string)
      • stage (string or None)
      • revision (integer or None)
      • distance (integer)
      • commit (string)
      • dirty (boolean)
      • env (dictionary of environment variables)

      Available functions:

      Simple example:

      format-jinja = "{% if distance == 0 %}{{ base }}{% else %}{{ base }}+{{ distance }}.{{ commit }}{% endif %}"
      

      Complex example:

      format-jinja = """
          {%- if distance == 0 -%}
              {{ serialize_pep440(base, stage, revision) }}
          {%- elif revision is not none -%}
              {{ serialize_pep440(base, stage, revision + 1, dev=distance, metadata=[commit]) }}
          {%- else -%}
              {{ serialize_pep440(bump_version(base), stage, revision, dev=distance, metadata=[commit]) }}
          {%- endif -%}
      """
      
    • format-jinja-imports: Array of tables. Default: empty. This defines additional things to import and make available to the format-jinja template. Each table must contain a module key and may also contain an item key. Consider this example:

      format-jinja-imports = [
          { module = "foo" },
          { module = "bar", item = "baz" },
      ]
      

      This is roughly equivalent to:

      import foo
      from bar import baz
      

      foo and baz would then become available in the Jinja formatting.

    • style: String. Default: unset. One of: pep440, semver, pvp. These are preconfigured output formats. If you set both a style and a format, then the format will be validated against the style's rules. If style is unset, the default output format will follow PEP 440, but a custom format will only be validated if style is set explicitly.

    • latest-tag: Boolean. Default: false. If true, then only check the latest tag for a version, rather than looking through all the tags until a suitable one is found to match the pattern.

  • [tool.poetry-dynamic-versioning.subversion]: Options specific to Subversion.
    • tag-dir: String. Default: tags. This is the location of tags relative to the root.
  • [tool.poetry-dynamic-versioning.substitution]: Insert the dynamic version into additional files other than just pyproject.toml. These changes will be reverted when the plugin deactivates.
    • files: List of globs for any files that need substitutions. Default: ["*.py", "*/__init__.py", "*/__version__.py", "*/_version.py"]. To disable substitution, set this to an empty list.
    • patterns: List of regular expressions for the text to replace. Each regular expression must have two capture groups, which are any text to preserve before and after the replaced text. Default: ["(^__version__\s*=\s*['\"])[^'\"]*(['\"])"].

Simple example:

[tool.poetry-dynamic-versioning]
enable = true
vcs = "git"
style = "semver"

Command line mode

The plugin also has a command line mode for execution on demand. This mode applies the dynamic version to all relevant files and leaves the changes in-place, allowing you to inspect the result. Your configuration will be detected from pyproject.toml as normal, but the enable option is not necessary.

To activate this mode, run the poetry-dynamic-versioning command in your console.

Caveats

  • The dynamic version is not available during poetry run because Poetry uses os.execvp().

Implementation

In order to side-load plugin functionality into Poetry, this package does the following:

  • Upon installation, it delivers a zzz_poetry_dynamic_versioning.pth file to your Python site-packages directory. This forces Python to automatically load the plugin after all other modules have been loaded (or at least those alphabetically prior to zzz).
  • It first tries to patch poetry.factory.Factory.create_poetry and poetry.console.commands.run.RunCommand directly. If they cannot be imported, then it patches builtins.__import__ so that, whenever those classes are first imported, then they will be patched. The reason we may have to wait for these to be imported is in case you've used the get-poetry.py script, in which case there is a gap between when Python is fully loaded and when ~/.poetry/bin/poetry adds the Poetry lib folder to the PYTHONPATH.
  • The patched version of Factory will compute and apply the dynamic version.
  • The patched version of RunCommand will deactivate the plugin before executing the passed command, because otherwise we will not be able to do any cleanup afterwards.

Development

Please refer to CONTRIBUTING.md.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

poetry-dynamic-versioning-0.8.0.tar.gz (10.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

poetry_dynamic_versioning-0.8.0-py3-none-any.whl (10.3 kB view details)

Uploaded Python 3

File details

Details for the file poetry-dynamic-versioning-0.8.0.tar.gz.

File metadata

  • Download URL: poetry-dynamic-versioning-0.8.0.tar.gz
  • Upload date:
  • Size: 10.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.0.0 CPython/3.7.3 Windows/10

File hashes

Hashes for poetry-dynamic-versioning-0.8.0.tar.gz
Algorithm Hash digest
SHA256 281234c148b8e8ee64d90c4b8a3c9c87a4cc05e7b6b37c3f0ee074ef342e0613
MD5 4103044327a9e624d2c7b5ba3fe74ca1
BLAKE2b-256 aa7972092a0b6181c0b6b2913d216826b56cd9efc48e3902abd83d44277394fd

See more details on using hashes here.

File details

Details for the file poetry_dynamic_versioning-0.8.0-py3-none-any.whl.

File metadata

File hashes

Hashes for poetry_dynamic_versioning-0.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 5fe252f6cabf723628527ef9c45d5b3f0eb4302cad31bcd5061b7f28b191d40a
MD5 b0a6cac01c62223913958a6d8f1f4e87
BLAKE2b-256 e3776d35c5b8e140be2d336f867323116ff1c83cdb0fa69a29c8064e9b40ee73

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page