Unified server-side Python utility for the PocketShell Android client.
Project description
pocketshell
Unified server-side Python utility for the PocketShell
Android client. Replaces the separately-installed quse and tmuxctl
utilities the app currently probes for on every remote host.
This first release ships the skeleton plus the pocketshell usage
subcommand only. Follow-up rounds will add jobs, agent-log,
sessions, repos, and an optional daemon mode. See
issue #170 for
the design spike and phased roll-out plan.
Install
The recommended path is uv tool install, which lands the binary on PATH
under ~/.local/bin/:
uv tool install pocketshell
For local development from a clone:
cd tools/pocketshell
uv venv
uv pip install -e .
pocketshell --help
pipx install pocketshell works the same way for users who prefer
pipx. Both install paths produce a pocketshell binary that the
PocketShell app's bootstrap probe detects.
Optional extras
pocketshell qr-share requires the qrcode[pil] package (Pillow) to
render QR images. Because Pillow is heavy and not needed by any other
subcommand, it ships behind an optional qr extra:
uv tool install pocketshell --with qrcode[pil]
# or
pip install pocketshell[qr]
Without the extra, every other subcommand keeps working; only
pocketshell qr-share exits 127 with a friendly install hint.
Usage
pocketshell usage # human-readable lines, one per provider
pocketshell usage --json # machine-readable JSON (consumed by the app)
pocketshell usage codex # filter to a single provider
The output shape is byte-identical to quse [provider] [--json] so any
consumer that already parses quse output keeps working when the app
routes through pocketshell usage instead. Under the hood the first
release delegates to the quse CLI via subprocess; later rounds will
fold the provider-detection logic in directly and drop the subprocess
hop.
If quse is not installed, pocketshell usage exits with code 127 and
prints an install hint to stderr.
pocketshell repos list
Enumerate git repositories — either cloned on this host (--local) or
owned by the authenticated GitHub user (--remote). The two modes
share one unified JSON schema so a future merged view can interleave
them transparently.
pocketshell repos list --local # scan ~/git for clones (human)
pocketshell repos list --local --json # same, JSON output
pocketshell repos list --remote --json # via owner-only `gh api user/repos`
pocketshell repos list --remote --limit 20
Schema (every entry):
{
"owner": "alexeygrigorev", // null when remote URL is non-GitHub
"name": "pocketshell", // local dir basename, or GH repo name
"full_name": "alexeygrigorev/pocketshell", // null when owner unknown
"local": { // populated by --local scans
"path": "/home/alexey/git/pocketshell",
"head": "main"
},
"remote": { // populated by --remote scans
"default_branch": "main",
"html_url": "https://github.com/alexeygrigorev/pocketshell",
"ssh_url": "git@github.com:alexeygrigorev/pocketshell.git",
"updated_at": "2026-05-27T12:00:00Z"
}
}
--local scans ~/git by default (override with one or more --root
flags or the colon-separated POCKETSHELL_REPOS_ROOTS env var) and
populates local for every entry. owner and full_name are
best-effort from the parsed remote.origin.url; non-GitHub remotes
leave them null.
--remote delegates to gh api 'user/repos?affiliation=owner&sort=updated' --paginate --slurp.
Requires gh on PATH (apt install gh on Debian/Ubuntu,
brew install gh on macOS) authenticated via
gh auth login -s repo:read. Sorted by updated_at descending so the
picker shows the most-recently-touched repos first. Missing gh exits
127 with an install hint; a non-zero gh exit (auth missing,
rate-limit, etc.) propagates the exit code and stderr verbatim.
With neither flag, defaults to --local and prints a one-line
discoverability hint mentioning --remote.
Daemon mode caches repos.list_local for 10 s and repos.list_remote
for 5 min. --no-daemon forces the in-process path; --no-cache
forces the daemon to re-run upstream on the next call.
pocketshell qr-share
Builds a pocketshell.ssh-import.v1 payload from an ~/.ssh/config
alias (resolved via ssh -G) or from explicit flags, wraps it in one or
more pocketshell.qr.v1 chunked envelopes (matching the Kotlin
QrChunkCodec byte-for-byte), and emits QR codes for the phone-side
scanner to consume (issue #129).
pocketshell qr-share prod # ssh-config alias
pocketshell qr-share --host h --user u --key ~/.ssh/id_ed25519 --name h
pocketshell qr-share prod --png --out-dir /tmp/qr # write PNGs
pocketshell qr-share prod --print-only --id deadbeef # debug envelopes
When stdout is a TTY the QRs are drawn inline as Unicode blocks; between
multi-part transmissions the command pauses on "Press Enter for next
QR" so the user can scan each in turn. When stdout is not a TTY (or
--png is passed) a numbered PNG sequence (qr-share-01.png,
qr-share-02.png, ...) is written to --out-dir.
Requires the optional qr extra (see Optional extras).
Without it, the command exits 127 with the install hint and every other
subcommand keeps working.
Development
cd tools/pocketshell
uv venv
uv pip install -e ".[dev]"
uv run pytest
Or via the dependency-group:
uv sync --group dev
uv run pytest
The tests stub quse.usage.collect_usage so they run in seconds without
hitting any provider API.
Release flow
pocketshell ships in lockstep with the Android app. Every time the
maintainer cuts an Android release tag (vX.Y.Z), the
Build workflow assembles the APK
and also builds the Python sdist + wheel and publishes them to PyPI.
Version coupling
Two files must agree on the release version:
app/build.gradle.kts->versionName = "X.Y.Z"tools/pocketshell/pyproject.toml->version = "X.Y.Z"
scripts/check-pypi-version.sh
enforces this. The release workflow runs it with --check-tag vX.Y.Z
before publishing, so a tag pushed with mismatched versions fails the
job loudly before anything reaches PyPI.
Run it locally before tagging:
scripts/check-pypi-version.sh # local match check
scripts/check-pypi-version.sh --check-tag v0.3.0
Bumping a release
- Pick the next semantic version after the latest GitHub Release/tag.
- Update both version sources in the same commit:
app/build.gradle.kts-> bumpversionName(andversionCode).tools/pocketshell/pyproject.toml-> bumpversionto the same value asversionName.
- Run
scripts/check-pypi-version.shto confirm they match. - Commit the bump on
main, push, and run the emulator release validation gate (scripts/release-emulator-validation.sh) as described inprocess.md-> "Release Builds". - Push the tag with
scripts/push-release-tag.sh. The tag-triggeredBuildworkflow then:- builds and uploads the APK + creates the GitHub Release
- runs
scripts/check-pypi-version.sh --check-tag vX.Y.Z - builds the Python sdist + wheel
- publishes them to PyPI via OIDC trusted publishing
The PyPI publish job depends on the APK build job, so a broken APK
build also aborts the PyPI publish. If only the PyPI publish fails the
maintainer can re-trigger the workflow at the same tag from the
Actions tab; the APK build is idempotent against an existing release
(softprops/action-gh-release updates the existing release rather
than failing).
PyPI trusted publishing setup (one-time)
The publish-pypi job uses GitHub's OIDC token instead of a long-lived
API token. This avoids storing a PYPI_API_TOKEN secret in the repo
and means there is nothing to rotate. The trade-off is that the
project owner must complete one configuration step on pypi.org before
the first automated tag publish:
- Sign in to https://pypi.org/ with the project owner account.
- Open the
pocketshellproject page -> Manage -> Publishing. - Under Trusted publishers, click Add a new pending publisher
(if the project is empty) or Add a new publisher, then fill in:
- PyPI Project Name:
pocketshell - Owner:
alexeygrigorev - Repository name:
pocketshell - Workflow name:
build.yml - Environment name:
pypi
- PyPI Project Name:
- Save the publisher.
- In this repository on GitHub, open
Settings -> Environments -> New environment -> name it
pypi. No secrets or reviewers are required; the environment exists purely to scope the OIDC token. (If the environment already exists, confirm it has no protection rules that would block the workflow from running.) - Push the next release tag. The
Publish to PyPI via trusted publishingstep should succeed without any token configuration.
Why trusted publishing (and not PYPI_API_TOKEN)?
- No long-lived secret to rotate, leak, or accidentally print in logs.
- The OIDC subject is scoped to
repo=alexeygrigorev/pocketshell,workflow=build.yml,environment=pypi, so a compromised fork or a different workflow file in this repo cannot reuse it. - D22 (no backwards-compat): we do not also maintain a token-fallback path. If trusted publishing breaks, fix it; do not add a token branch alongside.
If trusted publishing is ever unavailable for a tag (e.g. PyPI outage on the OIDC verifier), the recommended manual escape hatch is:
cd tools/pocketshell
python -m build
python -m twine upload dist/*
with the maintainer's account. Do not re-add a PYPI_API_TOKEN secret
as a permanent fallback.
Why a unified CLI?
The PocketShell app previously probed for two binaries (quse,
tmuxctl) on every host. That meant two installs to keep up to date,
two probes to surface failures from, and two PATH-discovery edge cases
(see issue #41).
A single pocketshell binary collapses those into one install, one
probe, one bootstrap row. The app keeps detecting quse and tmuxctl
as a parallel path while pocketshell ramps up to feature parity; once
parity is reached, the legacy probes are removed in a hard-cut follow-up
(no compat shim — see decision D22 in docs/decisions.md).
Project details
Release history Release notifications | RSS feed
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 pocketshell-0.3.1.tar.gz.
File metadata
- Download URL: pocketshell-0.3.1.tar.gz
- Upload date:
- Size: 96.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6d99956828b85ebdf9cd00695b61c97d770c4295dbb8d8a04ae43adbb621eb72
|
|
| MD5 |
1b38226ae3f8b5db62521e2b07240b4a
|
|
| BLAKE2b-256 |
1c1aee3eb03cdcefcb4c5d23aeaf104c202f63192c7c3aaa6db0593feac2ca40
|
File details
Details for the file pocketshell-0.3.1-py3-none-any.whl.
File metadata
- Download URL: pocketshell-0.3.1-py3-none-any.whl
- Upload date:
- Size: 60.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5a68910bc3c79b280fb84d41769a926335da128b1369d2e84fd5baf726397633
|
|
| MD5 |
b7109688ce9d852496f3704476fbf891
|
|
| BLAKE2b-256 |
414e29f11b217702a1a9b769c0dc7bcf56d31e9c92470df68aefb74c5c2a3ee8
|