CLI to manage Docker services under /srv/docker (ownership, permissions, ACLs)
Project description
coxyz
CLI to manage Docker services under /srv/docker following coxyz rules
(ownership, permissions, POSIX ACLs).
Replaces check_fix_permission.zsh + services.zsh with a single typed Python
tool driven by a YAML configuration.
Install
coxyz is published on PyPI as the coxyz-cli
package — the installed command stays coxyz. It needs root for most operations
(chown / setfacl), so install it system-wide. Commands that need root
re-exec themselves through sudo automatically — you no longer have to prefix
them yourself (set COXYZ_NO_SUDO=1 to opt out, e.g. in containers running as
root).
sudo apt install -y pipx
# Install into an isolated venv under /opt, with the binary on the system PATH.
sudo env PIPX_HOME=/opt/pipx PIPX_BIN_DIR=/usr/local/bin pipx install coxyz-cli
With pipx ≥ 1.5 you can use the shorter
sudo pipx install --global coxyz-cliinstead. Debian 12 ships pipx 1.4.3, which needs theenvform above.
Then run:
coxyz check
Optionally enable shell completion for your user (no sudo):
coxyz --install-completion
Once installed, completion suggests service names for commands that take a
service argument (check, apply, dev add/remove, meta scaffold/validate),
categories for -C/--category, and existing images for image remove.
Update
sudo env PIPX_HOME=/opt/pipx PIPX_BIN_DIR=/usr/local/bin pipx upgrade coxyz-cli
Migrating from a manual install
Earlier setups used hand-written coxyz / coxyz-update wrapper scripts and a
venv in /usr/local/libexec/coxyz. Remove them before installing from PyPI:
sudo rm -f /usr/local/bin/coxyz /usr/local/bin/coxyz-update
sudo rm -rf /usr/local/libexec/coxyz
rm -f ~/.zsh/completions/_coxyz ~/.zcompdump* # stale completion artefacts
(/etc/coxyz/config.yaml is kept — it is your configuration, not part of the
install.)
Configuration
coxyz reads, in order: --config FILE, /etc/coxyz/config.yaml,
~/.config/coxyz/config.yaml, then the bundled defaults.
coxyz show-config # inspect the resolved config
coxyz edit # create/edit /etc/coxyz/config.yaml (seeded from defaults)
Example excludes in config.yaml:
exclude:
- "*.bak"
- "*/do_not_touch/"
Commands
coxyz list # list services with image, ports, status
coxyz list -C apps # filter by category
coxyz check # validate config + audit all services (exit 1 on drift)
coxyz check bitwarden # audit one service
coxyz check apps/bitwarden -v # verbose (show OK findings too)
coxyz apply # preview planned fixes, confirm, then apply
coxyz apply bitwarden -y
coxyz create # interactive prompts, confirm, then create
coxyz create -C apps -n myapp -y
coxyz manifest # aggregate every service.yaml → API manifest
coxyz manifest --dry-run # validate + preview without writing
coxyz meta scaffold apps/nginx # add a service.yaml template to an existing service
coxyz meta validate # validate all service.yaml descriptors
coxyz dev add apps/nginx # make a service editable via code-server
coxyz dev remove apps/nginx # revoke it
coxyz dev list # show dev-enabled services
coxyz image add api # scaffold a self-built image context in /opt/images
coxyz image remove api # delete an image build context
coxyz image list # list image build contexts
coxyz show-config # print resolved config
coxyz edit # edit /etc/coxyz/config.yaml
Most operations require root (chown / setfacl). coxyz elevates itself with
sudo automatically when needed, so the examples above work without a prefix
(use COXYZ_NO_SUDO=1 to disable auto-elevation).
How it works
-
Config (
/etc/coxyz/config.yamlor bundled default) defines:- root dir, ACL principals, authorized categories
excludeglob patterns to ignore paths during audit/apply- per-path rules: mode, ACL perms, optional owner override, audit-only flag
-
check: read-only. First validates the config's structure (missing keys, bad values, sections nested in the wrong place), then audits permissions/ACL — reporting drift and warn-only (data/,.env). -
apply: shows planned changes, asks for confirmation, then applies fixes.- Touches: category/service dirs,
compose.yaml, theconfig/directory. - Never touches:
data/contents,.envfiles (audit-only). - Creates required missing directories before applying path fixes.
- Touches: category/service dirs,
-
Dev-mode awareness: both
checkandapplyfirst read the code-server compose to learn which services are dev-enabled. For those services the dev principal's recursive ACL and the default ACL onconfig/anddata/are treated as expected — not drift — and any fix is non-destructive (it never usessetfacl --set/-b, which would wipe the dev grant). A leftover dev ACL on a service that is not dev-enabled is still correctly flagged for removal. -
create: scaffolds<category>/<service>/{config/,data/}plus emptycompose.yamland.env, aservice.yamltemplate, with correct owners + perms + ACL. It does not templatecompose.yaml— you fill it in. Then it refreshes the dashboard manifest. -
service.yaml(dashboard descriptor): a per-service file describing how the service appears on the coxyz dashboard —name,icon,description,public(true ⇒ exposed by the API, false ⇒ hidden entirely), optionalurl/kind/container/tags, and adetails:block (summary, features, internalports,depends_on,tech). Put only non-sensitive info here. Its permissions are governed by theservice_filerule (default640). -
manifest: reads everyservice.yaml, validates it, and aggregates the public ones into the JSON file atapi.manifest(default/srv/docker/apps/api/data/manifest.json, mode644), which the coxyz-api container mounts read-only and serves at/api/services. Private descriptors never reach the manifest. -
meta scaffold <service>: drops aservice.yamltemplate into an existing service (won't overwrite).meta validate: validates descriptors only. -
checkalso validates everyservice.yaml(a missing one is a warning; a malformed one is an error that fails the check). -
Self-built images live outside the service tree, in their own build context under
images.dir(default/opt/images/<name>/— Dockerfile + sources). The matching service under/srv/dockerstays empty: it just consumes the built image, exactly like a third-party image. Same convention for source repos underrepos.dir(default/opt/repos).image add/remove/list:addscaffolds<images.dir>/<name>/with the configuredowner/mode(defaultboxyz_dev:boxyz_dev,775) plus a Dockerfile template;removedeletes the whole context;listshows each context with its Dockerfile/compliance.- The
775mode means the dev principal's group can edit while others (the root Komodo Periphery process) can read — so Komodo builds the context with no per-service ACL and without touching/srv/dockerisolation. - Configure the locations under the
images:andrepos:sections;check/applythen enforce the owner/mode of every<dir>/<name>directory.
# Build with Komodo (or the CLI): context = the image's own directory. docker build -t api-coxyz:latest /opt/images/api
-
list: parses eachcompose.yamlfor image/ports and runs an audit to show a compliance status. -
dev add/remove/list: makes a service editable through code-server.addgrants thedev.principalgroup (defaultboxyz_dev) a recursive read/write ACL on the service'sconfig/anddata/(existing files and a default ACL so new files inherit it), and mounts both dirs into the code-server compose under/workspace/services/<category>/<service>/.removerevokes only that group's ACL entry and unmounts. The managed mounts live in a marker-delimited block (# >>> coxyz dev ... >>>) that is the single source of truth —listreads it; everything else in the compose is left untouched. Configured under thedev:key inconfig.yaml.
ACL handling
A path governed by an ACL rule is brought to compliance with a single
setfacl --set call that writes the base entries (u::/g::/o::, i.e. the
octal mode) and the named entries together. setfacl then recomputes the ACL
mask as the union of the owning group and every named entry, so each entry
stays fully effective — getfacl never shows an #effective: restriction.
coxyz deliberately never runs chmod on an ACL-managed path: a chmod after
a setfacl would rewrite the mask instead of the group bits and silently shrink
the effective rights of every named entry.
One consequence: when a named entry grants more than the owning group (e.g. a
principal with rw on a 750 directory), the mask widens and ls -l shows the
wider group digit (770). That is correct POSIX behaviour — the audit compares
ACL entries, not the displayed mode.
File layout (enforced)
/srv/docker/<category>/<service>/
├── compose.yaml 660 svc_<cat>:svc_<cat> + ACL principals
├── config/ 750 svc_<cat>:svc_<cat> + ACL principals
│ └── ... (contents not audited)
└── data/ 750 svc_<cat>:svc_<cat> no ACL (audit only)
Development
make test # run the test suite
make build # build sdist + wheel into dist/
make release # tag the current version and push (CI publishes to PyPI)
Releasing: bump __version__ in src/coxyz/__init__.py, commit, then
make release. The tag vX.Y.Z triggers .github/workflows/publish.yml,
which publishes to PyPI via Trusted Publishing.
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 coxyz_cli-1.1.0.tar.gz.
File metadata
- Download URL: coxyz_cli-1.1.0.tar.gz
- Upload date:
- Size: 51.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9941a21551eb449047c2b53ef509b811c76199edb33b4aac3c16e347fb6a9c8b
|
|
| MD5 |
114884809ef82e2be3e33eebdac4d24e
|
|
| BLAKE2b-256 |
e17ed809b3eeb08bef2df9b8c3fa0c93ed06a898db2c23ba38d699c3c0e65997
|
Provenance
The following attestation bundles were made for coxyz_cli-1.1.0.tar.gz:
Publisher:
publish.yml on Coxyz/Checker
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
coxyz_cli-1.1.0.tar.gz -
Subject digest:
9941a21551eb449047c2b53ef509b811c76199edb33b4aac3c16e347fb6a9c8b - Sigstore transparency entry: 1883406501
- Sigstore integration time:
-
Permalink:
Coxyz/Checker@df5c30e5fc2fc0259d9445a2d941317f440bc49a -
Branch / Tag:
refs/tags/v1.1.0 - Owner: https://github.com/Coxyz
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@df5c30e5fc2fc0259d9445a2d941317f440bc49a -
Trigger Event:
push
-
Statement type:
File details
Details for the file coxyz_cli-1.1.0-py3-none-any.whl.
File metadata
- Download URL: coxyz_cli-1.1.0-py3-none-any.whl
- Upload date:
- Size: 43.6 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 |
aa0606f9850eafc868856ae849cd209766eaf0c531414f29e5015f56fa471fe7
|
|
| MD5 |
137fb22417944b83464aba72366cc06c
|
|
| BLAKE2b-256 |
f9870ed45b09653424c190eeea279f4452573ca2f435501a807fbd384c353711
|
Provenance
The following attestation bundles were made for coxyz_cli-1.1.0-py3-none-any.whl:
Publisher:
publish.yml on Coxyz/Checker
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
coxyz_cli-1.1.0-py3-none-any.whl -
Subject digest:
aa0606f9850eafc868856ae849cd209766eaf0c531414f29e5015f56fa471fe7 - Sigstore transparency entry: 1883406631
- Sigstore integration time:
-
Permalink:
Coxyz/Checker@df5c30e5fc2fc0259d9445a2d941317f440bc49a -
Branch / Tag:
refs/tags/v1.1.0 - Owner: https://github.com/Coxyz
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@df5c30e5fc2fc0259d9445a2d941317f440bc49a -
Trigger Event:
push
-
Statement type: