Release helper for Anki add-ons.
Project description
anki-addon-release
anki-addon-release is a small release helper for Anki add-ons.
It provides deterministic local release prep:
- read release config from
pyproject.toml - validate the add-on source tree and
manifest.json - build a clean
.ankiaddonarchive - inspect archive contents before upload
For AnkiWeb, it can either write a regular-browser handoff bundle or drive the
upload form with Playwright. Browser publishing is review-first by default: the
final AnkiWeb save/submit button is clicked only when --submit is passed.
Status
Early public release. It has been dogfooded against the Study Triage add-on release flow, including AnkiWeb login, create/update form filling, support URL filling, branch compatibility fields, and local browser-flow regression tests. The first PyPI upload is pending the one-time PyPI Trusted Publishing setup.
Install
Until the first PyPI release is uploaded, run from GitHub:
uvx --from "anki-addon-release @ git+https://github.com/elvis-sik/anki-addon-release.git" \
anki-addon-release --help
After a PyPI release is available, the shorter PyPI form will work:
uvx anki-addon-release --help
For browser publishing support from GitHub, include the browser extra and install the Playwright browser runtime:
uvx --from "anki-addon-release[browser] @ git+https://github.com/elvis-sik/anki-addon-release.git" \
playwright install chromium
Install For Local Development
python3 -m venv .venv
. .venv/bin/activate
python -m pip install -e .
No runtime dependencies are required for the initial local package/check workflow.
For browser publishing:
python -m pip install -e ".[browser]"
python -m playwright install chromium
Use sfw when installing from public registries in this workspace.
Credentials
The framework never calls 1Password, reads password-manager references, or stores AnkiWeb credentials in config. If you want automated login, resolve secrets outside the process and expose plain environment variables only for the command that needs them:
export ANKIWEB_EMAIL="$(op read op://Private/AnkiWeb/email)"
export ANKIWEB_PASSWORD="$(op read op://Private/AnkiWeb/password)"
Then reference the variable names in project config:
[tool.anki-addon-release.ankiweb]
login_email_env = "ANKIWEB_EMAIL"
login_password_env = "ANKIWEB_PASSWORD"
or pass them on the command line:
anki-addon-release login \
--email-env ANKIWEB_EMAIL \
--password-env ANKIWEB_PASSWORD \
--submit-login
Without --submit-login, the login command fills the fields and leaves the form
for review. In --headless mode, --submit-login is required.
Separate Publishing Account
For real publishing, prefer a dedicated AnkiWeb account used only for add-ons. This is an isolation pattern, not a framework requirement: it keeps release automation separate from your personal synced collection and makes it easier to reason about which account a browser profile is logged into.
Two AnkiWeb account lifecycle rules matter for that pattern:
- New-account publishing guard: AnkiWeb may send very new accounts to its Account Too New page when they try to share add-ons. That public page says new accounts must meet certain criteria and can continue once the account is older, but it does not publish an exact wait period. Create the publishing account before you need it, or use an older dedicated account.
- Inactivity expiry: AnkiWeb's terms and account-removal article say account data may be deleted if the account is not accessed or synced in the last 6 months. Log in to the dedicated account at least every few months to keep it active. The account-removal article notes that shared add-ons are not subject to the usual data expiry, but keeping the publishing account active avoids losing account access or release ownership context.
From GitHub, uv can run or install the package without a PyPI release once the
repository is public or otherwise accessible to the local Git credentials:
uvx --from "anki-addon-release @ git+https://github.com/elvis-sik/anki-addon-release.git" \
anki-addon-release --help
Configure An Add-on
Add a section like this to the add-on repository's pyproject.toml:
[tool.anki-addon-release]
source_dir = "."
manifest = "manifest.json"
artifact_dir = "dist"
artifact_name = "study-triage.ankiaddon"
include = ["__init__.py", "manifest.json", "README.md"]
exclude = [
".git",
"__pycache__",
".mypy_cache",
".pytest_cache",
".ruff_cache",
".uv-cache",
"dist",
"tests",
"user_files",
]
[tool.anki-addon-release.ankiweb]
# Omit addon_id for a first publish; set it for updates.
addon_id = "1234567890"
title = "Study Triage"
support_url = "https://github.com/example/study-triage"
description_file = "README.md"
changelog_file = "CHANGELOG.md"
login_email_env = "ANKIWEB_EMAIL"
login_password_env = "ANKIWEB_PASSWORD"
include is optional. When omitted, the whole source_dir is considered and exclude filters out development files.
Commands
From an add-on repository:
anki-addon-release check
anki-addon-release package
anki-addon-release inspect dist/study-triage.ankiaddon
Prepare an AnkiWeb publish plan without opening a browser:
anki-addon-release publish --dry-run
Build a regular-browser handoff bundle for Codex or a human:
anki-addon-release handoff
The handoff bundle is written to .anki-addon-release/handoff/ by default and
contains:
release-handoff.json: machine-readable release metadatabrowser-checklist.md: human browser checklistcodex-browser-prompt.md: prompt for a Codex browser-operator sessiondescription.txtandchangelog.txtwhen configured
This path does not require Playwright. Use it when you want Codex to operate your regular logged-in Chrome session.
Open an AnkiWeb login page using a persistent project-local browser profile:
anki-addon-release login
Prepare an upload with Playwright without clicking the final submit button:
anki-addon-release publish --diagnostics-dir out/release-diagnostics
Click the final submit button only when the flow is trusted:
anki-addon-release publish --submit --diagnostics-dir out/release-diagnostics
Use --mode create for first-publish testing and --mode update for updating a configured addon_id. --mode auto uses update when ankiweb.addon_id is present and create otherwise.
The final AnkiWeb save/submit button is not clicked unless --submit is passed.
That review-first behavior is the default. In headed browser mode, the prepared
form stays open until you press Enter in the terminal.
Without installation, from this repository:
PYTHONPATH=src python -m anki_addon_release --project /path/to/addon check
Development
make check
Browser-flow stress tests are opt-in because they need Playwright and a local fake AnkiWeb server:
ANKI_ADDON_RELEASE_BROWSER_TESTS=1 python -m unittest tests.test_browser_flows -v
or:
make test-browser
Those tests exercise separate create and update forms against a local HTTP server and verify that Playwright uploads the .ankiaddon artifact plus the expected metadata.
Releasing
Releases publish to PyPI via GitHub Actions using Trusted Publishing (OIDC); no API token is stored. To cut a release:
# 1. bump `version` in pyproject.toml, commit
# 2. tag and push -- the tag must match the version
git tag v0.1.0
git push origin v0.1.0
The release.yml workflow checks that the tag
matches pyproject.toml, builds the sdist + wheel with uv build, and publishes.
One-time setup on PyPI (Account -> Publishing -> Add a pending publisher):
- PyPI Project Name:
anki-addon-release - Owner / Repository:
elvis-sik/anki-addon-release - Workflow name:
release.yml - Environment name:
pypi
Roadmap
- Add saved HTML/screenshot diagnostics on every browser failure.
- Add public artifact verification by downloading/installing the published add-on into a disposable Anki profile, likely composed with
anki-addon-workbench. - Expand real-world compatibility coverage across more AnkiWeb form variants.
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file anki_addon_release-0.1.0.tar.gz.
File metadata
- Download URL: anki_addon_release-0.1.0.tar.gz
- Upload date:
- Size: 25.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6f864de84daad694eb15014a3e3072514762e92a05378bb0eeaac90c7ceb4c4d
|
|
| MD5 |
eb9f04e5bc1b8a19c8d851ea757f8c0d
|
|
| BLAKE2b-256 |
644f0bc0f4e8ac42a0ddd98a403eb8b7658bb5558b6bba29e068c9250f5bf5d7
|
Provenance
The following attestation bundles were made for anki_addon_release-0.1.0.tar.gz:
Publisher:
release.yml on elvis-sik/anki-addon-release
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
anki_addon_release-0.1.0.tar.gz -
Subject digest:
6f864de84daad694eb15014a3e3072514762e92a05378bb0eeaac90c7ceb4c4d - Sigstore transparency entry: 2051623675
- Sigstore integration time:
-
Permalink:
elvis-sik/anki-addon-release@9acf4203b6f51a1d1733ce700fc7c21071c123c3 -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/elvis-sik
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@9acf4203b6f51a1d1733ce700fc7c21071c123c3 -
Trigger Event:
push
-
Statement type:
File details
Details for the file anki_addon_release-0.1.0-py3-none-any.whl.
File metadata
- Download URL: anki_addon_release-0.1.0-py3-none-any.whl
- Upload date:
- Size: 20.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
070399bfed1e2e47bf1dcf8ea280601c8c0a8c1f1f59f9a1e34356a524f90783
|
|
| MD5 |
96a2ca75602f09ed8e713b1f5576a5aa
|
|
| BLAKE2b-256 |
93b41146c23e325f3e2ba384647de2931f8878d8bfd613f576db96b3533674f8
|
Provenance
The following attestation bundles were made for anki_addon_release-0.1.0-py3-none-any.whl:
Publisher:
release.yml on elvis-sik/anki-addon-release
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
anki_addon_release-0.1.0-py3-none-any.whl -
Subject digest:
070399bfed1e2e47bf1dcf8ea280601c8c0a8c1f1f59f9a1e34356a524f90783 - Sigstore transparency entry: 2051623824
- Sigstore integration time:
-
Permalink:
elvis-sik/anki-addon-release@9acf4203b6f51a1d1733ce700fc7c21071c123c3 -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/elvis-sik
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@9acf4203b6f51a1d1733ce700fc7c21071c123c3 -
Trigger Event:
push
-
Statement type: