Multi-profile Codex CLI quota monitor with CLI and TUI
Project description
codex-quota
A utility for managing isolated Codex CLI profiles and monitoring quota usage across multiple Codex environments.
Includes JSON output for automation and a live terminal dashboard.
Note: This is an independent community tool. It is not affiliated with or endorsed by OpenAI.
Requirements
- Python 3.11+
- Official Codex CLI available as
codex
Install Codex CLI:
npm install -g @openai/codex
or
brew install codex
Tested with codex-cli 0.136.0.
Installation
Install from PyPI:
pip install codex-quota
Install from source:
git clone https://github.com/ssh-den/codex-quota.git
cd codex-quota
python -m venv .venv
source .venv/bin/activate # Windows: .venv\Scripts\activate
pip install -e .
Quick Start
# Initialize (creates config and profiles directory)
codex-quota init
# Add a profile and log in
codex-quota profile add personal --login
# Check quota status
codex-quota status
# Launch the live terminal dashboard
codex-quota tui
Security
Authentication is fully delegated to the official Codex CLI. codex-quota never reads, stores, or transmits credentials. It sets CODEX_HOME and calls codex on your behalf.
Profile and config directories are created with 0700 permissions on POSIX systems so auth.json and related state are not exposed by a weak umask on multi-user machines.
Script filesystem layout
~/.codex-quota/profiles/
├── personal/
├── work/
└── testing/
~/.config/codex-quota/
└── config.json
The filesystem is the source of truth. Profiles are discovered dynamically. The configuration file stores application settings only and does not maintain a profile registry.
Configuration
Default configuration:
{
"profiles_dir": "~/.codex-quota/profiles",
"codex_bin": "codex",
"default_model": "gpt-5.4-mini",
"reasoning_effort": "low",
"refresh_seconds": 30.0
}
Edit ~/.config/codex-quota/config.json to change model defaults or reasoning effort.
Initialization
codex-quota init
Behavior:
- Create config directory if missing
- Create profiles directory if missing
- Create
config.jsonif missing - Preserve existing configuration
- Verify Codex CLI installation
- Print detected Codex version
- Discover existing profiles by scanning
~/.codex-quota/profiles/*
Overwrite configuration:
codex-quota init --overwrite-config
Run bootstrap plus quota verification (--check is an alias):
codex-quota init --verify
Profile management
List profiles:
codex-quota profile list
Add profile:
codex-quota profile add personal
Create and login immediately:
codex-quota profile add personal --login
Create and force device-code auth on headless or remote machines:
codex-quota profile add personal --login --device-auth
Print profile path:
codex-quota profile path personal
Remove profile:
codex-quota profile remove personal
Force remove (non-interactive):
codex-quota profile remove personal --yes
Profile names are validated. Path traversal, path separators, absolute paths, whitespace, and suspicious characters are rejected.
Login
codex-quota login personal
Equivalent to:
CODEX_HOME="$HOME/.codex-quota/profiles/personal" codex login
Use device authorization instead of the local browser callback flow:
codex-quota login personal --device-auth
Create the profile automatically if missing:
codex-quota login personal --create
Combine both when bootstrapping a remote profile in one step:
codex-quota login personal --create --device-auth
Status
All profiles:
codex-quota status
Single profile:
codex-quota status personal
JSON output:
codex-quota status --json
Example human-readable output:
Profile Status 5h Left Week Left
personal OK 99% 69%
work OK 74% 52%
Default JSON is safe for CI logs and automation:
- No absolute profile paths
- No raw
rateLimitspayload - No plan or credits metadata unless explicitly requested
Opt in to the extra fields only when you need them:
codex-quota status --json --json-paths --json-raw
Quota lockout detection uses rateLimitReachedType != null. The tool does not infer lockout solely from usedPercent == 100.
Check command
Automation-oriented health check:
codex-quota check
Single profile:
codex-quota check personal
Exit codes:
| Code | Meaning |
|---|---|
| 0 | Selected profiles are authenticated, readable, and not quota-blocked |
| 2 | Codex missing, auth missing/expired, app-server failure, quota block, or no selected profiles found |
Exec
Run Codex under a selected profile:
codex-quota exec personal "Reply with exactly: ok"
Disable forced JSON output:
codex-quota exec personal "Say hello" --raw
exec now respects the user's existing Codex config and rules by default instead of disabling them.
TUI
Launch:
codex-quota tui
Custom refresh interval:
codex-quota tui --refresh 15
The TUI uses Textual and dynamically discovers profiles from the filesystem.
Available commands:
refresh
login <profile> [--device-auth]
exec <profile> <prompt>
quit
Internal quota API
Status collection launches:
CODEX_HOME=<profile-home> codex app-server --stdio
Then performs the following JSON-RPC sequence:
1. initialize
2. initialized
3. account/rateLimits/read
Responses are normalized for display and safe JSON output. Raw API values are available only through explicit --json-raw.
License
MIT License
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 codex_quota-1.0.0.tar.gz.
File metadata
- Download URL: codex_quota-1.0.0.tar.gz
- Upload date:
- Size: 22.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
46eafb6fde966efa8054e61a7139c7967b8110992dda5c9e2a5971dcfd986373
|
|
| MD5 |
23b87986c32e88f9399452de7f097e90
|
|
| BLAKE2b-256 |
cbc60c40c6a9372b7f8937041a54f22c342ba14e3e94c2490bdcfffbbf88aeca
|
Provenance
The following attestation bundles were made for codex_quota-1.0.0.tar.gz:
Publisher:
publish-pypi.yml on ssh-den/codex-quota
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
codex_quota-1.0.0.tar.gz -
Subject digest:
46eafb6fde966efa8054e61a7139c7967b8110992dda5c9e2a5971dcfd986373 - Sigstore transparency entry: 1719614898
- Sigstore integration time:
-
Permalink:
ssh-den/codex-quota@ad6e1a586ad45826e504a15f6615e8276f3bf609 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ssh-den
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@ad6e1a586ad45826e504a15f6615e8276f3bf609 -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file codex_quota-1.0.0-py3-none-any.whl.
File metadata
- Download URL: codex_quota-1.0.0-py3-none-any.whl
- Upload date:
- Size: 19.9 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 |
4bfbf4a0bc1f0936d5b5b522d73fba7db6f00743d9d9515832062574c8a0f8c8
|
|
| MD5 |
8b83dd3070f168ce2ea8acabeaf834be
|
|
| BLAKE2b-256 |
733746257e8c195c7ee570f5e494f9cd48989471b105c816fdac31e673078739
|
Provenance
The following attestation bundles were made for codex_quota-1.0.0-py3-none-any.whl:
Publisher:
publish-pypi.yml on ssh-den/codex-quota
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
codex_quota-1.0.0-py3-none-any.whl -
Subject digest:
4bfbf4a0bc1f0936d5b5b522d73fba7db6f00743d9d9515832062574c8a0f8c8 - Sigstore transparency entry: 1719614997
- Sigstore integration time:
-
Permalink:
ssh-den/codex-quota@ad6e1a586ad45826e504a15f6615e8276f3bf609 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ssh-den
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@ad6e1a586ad45826e504a15f6615e8276f3bf609 -
Trigger Event:
workflow_dispatch
-
Statement type: