2.4 KiB
2.4 KiB
PyPI release runbook — wifi-densepose + ruview
Operations doc for the .github/workflows/pip-release.yml CI workflow.
Auth
The workflow uses one GitHub Actions secret named PYPI_API_TOKEN.
It's a project-token issued by the rUv PyPI account with upload
scope for both wifi-densepose and ruview.
Refreshing the token
The canonical copy of the token lives in GCP Secret Manager,
project cognitum-20260110, entry name PYPI_TOKEN. To push a
fresh copy into GitHub Actions:
gcloud secrets versions access latest \
--secret=PYPI_TOKEN \
--project=cognitum-20260110 \
| tr -d '\r\n\xef\xbb\xbf' \
| gh secret set PYPI_API_TOKEN --repo ruvnet/RuView
The tr step strips any BOM / CRLF that PowerShell pipes or
Windows editors may have introduced — without it, twine fails with
UnicodeEncodeError: 'latin-1' codec can't encode character ''.
Triggering a release
Two paths:
- Tag push —
git tag v2.X.Y-pip && git push origin v2.X.Y-pip— publishes the v2 wheel matrix.v1.99.0-piptriggers the tombstone job instead. - Manual dispatch —
gh workflow run pip-release.yml --ref <branch> -f target=v2-wheels -f publish_to=pypi. Usepublish_to=testpypifor a dry-run target if a TestPyPI token is also set asTESTPYPI_API_TOKEN.
Release-day sequence
Per ADR-117 §7.3, the tombstone publishes first so it claims the "current" slot in pip's resolver:
git tag v1.99.0-pip && git push origin v1.99.0-pip→ tombstone live athttps://pypi.org/project/wifi-densepose/1.99.0/- Verify:
pip install wifi-densepose==1.99.0; python -c "import wifi_densepose"→ ImportError with migration URL. git tag v2.0.0-pip && git push origin v2.0.0-pip→ v2 wheel matrix live athttps://pypi.org/project/wifi-densepose/2.0.0/.- (Optional, in lock-step) build + publish a matching
ruviewrelease frompython/ruview-meta/so the meta-package version stays pinned to the same wifi-densepose version.
Off-loop manual gates
- Q3 (ADR-117 §11.3) — generate
expected_features_v2.sha256from the v2 Rust pipeline before any v2 publish. - OIDC Trusted Publisher — not used. The workflow is token-based;
this is a deliberate choice to keep the secret refresh entirely in
GCP. If the project migrates to OIDC later, remove
password:frompypa/gh-action-pypi-publishcalls and add the publisher registration on pypi.org.