Version: "0.2.0"
- Change files according to overview in
TODO
below - Develop package...
- Install package:
- static/normal install:
pip install .
- editable install:
pip install -e .
- dev install:
pip install -e .[dev]
(some terminals require to escape brackets with\
as\[
, esp. zsh)
- static/normal install:
-
LICENSE
: license text -
README.md
: Github-flavored markdown file -
templatepy
: placeholder folder for any python package that is configured for install viasetup.cfg
andpyproject.toml
__init__.py
: contains basic package info and example function that is called by console entrypoint (seesetup.cfg
)example.data.file.config
: a file to demonstrate that data files are included based onsetup.cfg
criteriaexample.data.file.test-extension-yu48
: a file to demonstrate data exclusion viasetup.py
tests
: placeholder folder for unit/integration tests and associated datapytest.ini
: config for testing framework withpytest
andcoverage
plugin (pytest-cov
)
Packaging System (see: packaging and [pyproject.toml])
-
MANIFEST.in
: Defines additional files to include/exclude in the build (if not automatically detected). -
pyproject.toml
: Central configuration file that replaces the traditionalsetup.cfg
andsetup.py
files:- Build System: Specifies the build system requirements and configuration, as defined in PEP 518 and PEP 621.
- Package Metadata: Includes the project's metadata (name, version, dependencies, etc.).
- Code Formatting: Configuration for tools like black and [flake8] (if used).
- Optional Dependencies: Organizes extra dependencies for development or other environments.
-
setup.py
: Legacy file retained only for backward compatibility if needed (e.g., older tooling). New projects should avoid it entirely.
The CI workflow is triggered on push to main
or when a tag is created. It ensures code quality and automates the release process:
-
Linting and Testing:
lint
: Checks code style withblack
andflake8
, runs pre-commit hooks.test
: Runs tests withpytest
and generates coverage reports.
-
Tag Validation:
check-tag
: Verifies that the tag is valid (notdev
orrc
) before proceeding.
-
Release Creation:
release
: Creates a GitHub release when the tag is valid.
-
Deployment to PyPI:
deploy
: Builds and uploads the package to PyPI usingtwine
.
The pipeline ensures code quality, passing tests, and automated deployment on new releases.
- Install
act
:- macOS/Linux:
brew install act
/sudo apt install act
- Windows: Download from act GitHub Releases and follow the installation instructions.
- macOS/Linux:
- Ensure Docker is installed and running. Download from Docker Desktop.
- Run the CI workflow with
act
:act push act push --verbose # very verbose output !
-
.pre-commit-config.yaml
: use pre-commit to run code formatting (e.g. with black andflake8
) and PEP compliance checks- Install pre-commit hook with
pre-commit install
(Note: only installs it in the current virtual environment) - Run it manually with
pre-commit run --all
or leave it to run on commit (requires to re-stage changed files!)
- Install pre-commit hook with
-
.github
: folder that contains github automation workflows and issues templates -
.gitignore
: ignored files/folders in git tools -
.bumpversion.cfg
: config for bump2version
- Change package name:
- (1) Rename the
templatepy
folder. - (2) Update all occurrences in
README.md
. - (3) Update the
name
field inpyproject.toml
. - (4) Update
.github/workflows
files. - (5) Update version references in
pyproject.toml
andtemplatepy/__init__.py
.
- (1) Rename the
- Update project author and metadata details in
pyproject.toml
,README.md
, andtemplatepy/__init__.py
. - Update the license holder in the
LICENSE
file. - Update
README.md
badge paths at the top. - Verify inclusions/exclusions of installable files/folders in
MANIFEST.in
andpyproject.toml
under[tool.setuptools]
. - Ensure
.gitignore
contains relevant entries for the new project. - Add all version string locations to
[tool.bump2version]
inpyproject.toml
.- Use syntax like
[bumpversion:file:templatepy/__init__.py]
to specify locations for version updates.
- Use syntax like
- To upload to PyPI, follow the instructions in the section below.
- To upload to Zenodo (if the repository is for a publication):
- (1) Connect Zenodo to your GitHub account.
- (2) Enable Zenodo integration for the repository (Zenodo requires the repository to be public).
- (3) Create a new GitHub release (manually or via
.github/workflows/CI.yaml
). - (4) Wait for Zenodo to sync and assign a DOI (this usually takes about a minute).
- (5) Add the DOI badge to
README.md
.
Workflow for Automatically Uploading Package to PyPI or [Test PyPI]
-
Generate a PyPI API Key:
- Go to PyPI and create a new API key, either specific to the repository or a general-purpose key.
-
Add the API Key to GitHub:
- In your repository's settings on GitHub:
- Navigate to Settings > Secrets and variables > Actions.
- Add a new Actions secret with the name
TWINE_API_KEY
. - Paste the PyPI API key into the secret's value field.
- In your repository's settings on GitHub:
-
Create a New Release:
- On GitHub, create a new release manually via the Releases page.
- Use a version number without a release extension (e.g.,
x.y.z
).
- Use a version number without a release extension (e.g.,
- Alternatively, trigger the GitHub workflow configured for releasing by incrementing the version with
bump2version
(see info below) - Desperately, manually update the version strings in the relevant files and add git commit tag to trigger the release workflow on push
- On GitHub, create a new release manually via the Releases page.
The package will then be automatically uploaded to PyPI or Test PyPI as configured in your CI/CD workflow.
bump2version
is used to increment version numbers based on semantic versioning. Here’s how you can use it with the current setup to trigger a release:
- Bumping the Minor Version:
Increment the minor version (e.g., from
1.2.3
to1.3.0
):# Bumping the minor Version: bump2version minor # Bumping the major version (e.g., from 1.2.3 to 2.0.0): bump2version major # Bumping for a release Version (e.g., from 1.0.0.dev to 1.0.0.rc or 1.0.0): bump2version release
Historically, packaging in Python was governed by standards such as PEP-426
, PEP-517
, and PEP-518
. These PEPs introduced various mechanisms for packaging and building Python projects, but with certain limitations, especially regarding flexibility and future-proofing.
- PEP-426: Introduced the
setup.cfg
andsetup.py
files as the standard way to define package metadata and build configuration. - PEP-517: Introduced a standardized interface for building Python projects, separating the build process from the packaging process and allowing for more flexible build systems.
- PEP-518: Defined how
pyproject.toml
should be used to declare build dependencies and system requirements, allowing tools likepip
to know which backend to use for the build process.
While these PEPs were important milestones, the latest changes to the packaging ecosystem make pyproject.toml
the preferred way to configure projects going forward.
-
pip
:- Tested with
pip install . --use-feature=in-tree-build
for forward compatibility withpip 21.3
and later. - While
setup.py
is technically optional, an emptysetup.py
is still kept for enabling editable installs (pip install -e .
), as this requires such a file for now. wheel
is added as a build-system dependency to maintain compatibility with versions ofpip
that do not yet fully implementPEP 517
.
- Tested with
-
Replacing
setup.cfg
andsetup.py
withpyproject.toml
:- In the modern packaging ecosystem,
pyproject.toml
is increasingly the standard for declaring build systems, dependencies, and metadata. This configuration file simplifies the process and eliminates the need for separatesetup.py
andsetup.cfg
files in many cases.
- In the modern packaging ecosystem,
For further reading on the transition to pyproject.toml
and the removal of setup.py
/setup.cfg
, see the following discussions:
- check a signing key is configured:
git config --global user.signingkey
(see below for generating a key) - repo is a detached HEAD? check out a branch:
git checkout main
- there are changes that need to be committed?
git commit -am "commit message"
- tag already exists?
git tag
to list tags,git tag -d <tag>
to delete a tag
- generate a key with
gpg --full-generate-key
- list keys with
gpg --list-secret-keys --keyid-format LONG
- set the key for signing commits with
git config --global user.signingkey <key-id>
- enable signing commits with
git config --global commit.gpgsign true
- set the key for signing tags with
git config --global tag.gpgSign true
- export the public key with
gpg --armor --export <key-id>
- and add it to your GitHub account: select 'New key' -> paste output of command above
Which one is the key id you might ask?
/home/user/.gnupg/secring.gpg
------------------------------
sec 4096R/<KEY_ID> 2024-11-22 [expires: 2025-11-22]
uid Your Name <youremail@example.com>
ssb 4096R/<KEY_SUBKEY_ID> 2024-11-22
- try:
export GPG_TTY=$(tty)
Contributions are very welcome! Please see the contribution guidelines or check out the issues
This software is released under the BSD 3-Clause License