pi-multi-account
v1.7.0
Published
Automatic multi-account failover & rotation for Pi Agent across Anthropic (Claude), OpenAI/ChatGPT Codex, and Qwen/Alibaba. Auto-discovers authenticated accounts, grows the rotation on login, and drops accounts on logout, expiry, or quota/rate-limit error
Downloads
787
Maintainers
Readme
pi-multi-account
Automatic multi-account failover & rotation for Pi Agent, across Anthropic (Claude), OpenAI / ChatGPT Codex, and Qwen / Alibaba.
When the account you are using hits a quota or rate limit, pi-multi-account transparently switches to the next authenticated account/model and (optionally) resumes the interrupted task — so a long agent run does not die just because one account ran out of budget.
What it does
- Auto-discovers every authenticated account from
~/.pi/agent/auth.json(Anthropic Claude Pro/Max, OpenAI/ChatGPT Codex, and Qwen/Alibaba) and builds the failover rotation dynamically — no manual config editing. - Grows the rotation on login. Run
/login, choose Use a subscription, then select a numbered slot such asanthropic-account-3oropenai-codex-account-5. The next discovery sweep adds it to the rotation automatically. - Handles auth failures without poisoning healthy OAuth accounts. A generic final 401 briefly cools down a refreshable account and moves the current task forward. Explicit provider verdicts such as
authentication token has been invalidatedforce an early refresh; if the refresh token is dead too, the slot is removed and Pi prints the interactive/loginrecovery steps. - Fails over on quota / rate-limit (429 / 402 / 403 and friends): the exhausted account goes on cooldown (parsed from the provider's own reset metadata when available) and Pi switches to the next available account/model.
- Optional auto-continue: queues a safe continuation prompt after a switch so the agent keeps going from the last safe point.
- Session-bound overnight resume: if every account is cooling down, the live Pi session waits for the earliest recovery and continues automatically. A new user message,
/multi-account stop, session exit, or Esc during a running turn cancels the chain. - Deduplicates provably identical accounts so duplicate Codex
accountIdvalues and identical credentials do not consume multiple rotation slots or get separate cooldowns. New provable duplicate logins are rejected before the redundant slot is saved. - Keeps your thinking level stable across switches instead of letting it drift downward.
- Shows live limits for the active account in Pi's footer: remaining 5-hour and 7-day allowance plus reset countdowns for Codex and Anthropic OAuth accounts.
Install
pi install npm:pi-multi-accountRestart Pi or run /reload after installation.
Anthropic (Claude Pro/Max) works out of the box. OAuth login and request shaping for the base
anthropicprovider and everyanthropic-account-*alias are built in — no separatepi-anthropic-authinstall is required. If you already havepi-anthropic-auth, the two coexist safely (the shaping is idempotent). OpenAI Codex / ChatGPT and Qwen accounts work as well.
Recommended setting
Set Pi provider-level retries to zero so the SDK does not keep retrying an exhausted account before failover kicks in. In ~/.pi/agent/settings.json:
{ "retry": { "provider": { "maxRetries": 0 } } }Usage
Add accounts by opening the login picker:
/login
Use a subscription
ChatGPT Plus/Pro (Codex openai-codex-account-2)
/multi-account rediscoverPi 0.79.3 does not accept a provider argument after /login; select the account
slot from the interactive provider picker instead.
Check what's in the rotation at any time:
/multi-account statusForce-refresh and display detailed limits for the active account:
/multi-account limits refreshExample status output:
pi-multi-account: enabled · auto-discover ON
Current: anthropic/claude-opus-4-8
Current limits: Claude | 5h 0% left/2h14m | 7d 92% left/1d18h
Rotation (3): anthropic → openai-codex → openai-codex-account-2
Registered login slots: anthropic-account-2, openai-codex-account-2
Cooldowns: none
Invalidated (need re-login): none
Pending auto-resume: noneCommands
All three names are aliases for the same command: /multi-account, /provider-failover, /failover.
| Subcommand | Description |
|---|---|
| status (default) | Show enabled state, current model, rotation, login slots, cooldowns, invalidations, pending resume. |
| limits [refresh] | Show active-account 5h/7d limits; refresh bypasses the cache. Aliases: usage, quota. |
| rediscover | Force a re-scan of auth.json and rebuild the rotation now. |
| add [anthropic\|codex] | Print the next free account slot to select from the interactive /login picker. |
| next | Manually switch to the next fallback, deliberately overriding recorded cooldowns. |
| stop | Abort and cancel automatic failover/resume for the current task. |
| reset | Clear all cooldowns, invalidations and any pending auto-resume. |
| reload | Reload config from disk and re-discover accounts. |
| enable / disable | Turn failover on/off for the current Pi process. |
How rotation membership works
- Joins the rotation when an account has a present, non-expired credential in
auth.json(after/login). - Leaves the rotation when the credential is logged out / removed, its access token is expired with no refresh token, an API key is rejected, or a refreshable OAuth credential produces three distinct final auth failures without a success in between.
- Quota / rate-limit does not invalidate an account — it puts it on a temporary cooldown and the account returns once the cooldown expires.
- Duplicate identities share one rotation position and one cooldown, and status/startup identifies the redundant slot. Codex/ChatGPT is matched by stored
accountId; identical API keys or literal identical tokens are also matched. Separate Anthropic OAuth logins cannot be proven identical because Anthropic's stored credential exposes no stable account identifier.
Rotation refresh is triggered by changes to auth.json (detected on session/turn start) or on demand with /multi-account rediscover.
After re-authenticating an invalidated slot, restart any older Pi processes that
were already running. Pi keeps a still-unexpired access token in each process's
memory, so an old process can continue using the invalidated token even after a
new /login updates auth.json.
Configuration
A default config is created at ~/.pi/agent/provider-failover.json on first run. Useful keys:
| Key | Default | Description |
|---|---|---|
| enabled | true | Master switch. |
| autoContinue | true | Queue a continuation prompt after a switch. |
| autoDiscover | true | Auto-discover accounts from auth.json. |
| includeQwen | true | Include Qwen / Alibaba accounts. |
| providerOrder | ["anthropic","openai-codex","qwen"] | Preferred family order in the rotation. |
| cooldownMs | 6 h | Default cooldown when no reset metadata is provided. |
| showUsage | true | Show active Codex/Claude limits in Pi's footer. |
| usageRefreshMs | 5 min | Provider usage cache TTL; Anthropic is clamped to at least 10 min to avoid endpoint throttling. |
| usageStatusRefreshMs | 1 min | Re-render the footer countdown periodically; stale caches are refreshed according to usageRefreshMs. |
| maxAutoContinuesPerPrompt | 8 | Cap on auto-resume hops per task. |
| continuationPrompt | (built-in) | Template; supports {from}, {to}, {reason}. |
State (cooldowns, invalidations, recent switches, and an in-session pending resume marker) is persisted to ~/.pi/agent/provider-failover-state.json. Pending work is deliberately discarded when the session closes or a different session starts.
Privacy & security
pi-multi-account reads auth.json but never writes credentials itself and never stores credentials in its state. Account/token values are reduced to a short irreversible SHA-256 fingerprint for re-login detection and deduplication. To display limits, the active OAuth access token is sent only to its own provider's usage endpoint (chatgpt.com/backend-api/wham/usage or api.anthropic.com/api/oauth/usage); cached state contains percentages, reset times, plan/credit metadata, and the fingerprint, never the token. Config and state files are written with 0600 permissions.
