Use Codex or Grok OAuth from tools that expect OpenAI- or Anthropic-compatible APIs.
rotom is a local Rust gateway for Claude Code, OpenAI SDKs, Anthropic SDKs, and other API-compatible clients. Log in once with an OAuth provider, then point your tools at the local server.
Claude Code running through Grok:
Claude Code running through GPT:
cargo install rotom
rotom login
rotom serve --bind 127.0.0.1:14550 --api-key local-secretSet Claude Code or any Anthropic-compatible client to use the local gateway:
export ANTHROPIC_BASE_URL=http://127.0.0.1:14550
export ANTHROPIC_AUTH_TOKEN=local-secret
export ANTHROPIC_MODEL="gpt-5.5"
claudelogin lists the available OAuth providers and starts the selected flow. Use
rotom login --provider openai or rotom login --provider grok to skip the
prompt. Complete the login in a browser, then paste the full redirected URL
from the browser address bar, for example
http://localhost:1455/auth/callback?code=...&state=.... This matches
OpenClaw's remote/headless fallback and does not require your gateway host to
be reachable from the public internet.
ANTHROPIC_BASE_URL should point at the rotom server root, not /v1, because
Anthropic clients append /v1/messages themselves.
For non-interactive validation:
ANTHROPIC_BASE_URL=http://127.0.0.1:14550 \
ANTHROPIC_AUTH_TOKEN=local-secret \
ANTHROPIC_MODEL="gpt-5.5" \
claude -p "Reply with the single word OK"Grok OAuth uses the same local credential flow with xAI's OAuth endpoints:
rotom login --provider grok
rotom serve --bind 127.0.0.1:14550 --api-key local-secretCredentials are stored per provider, so logging in to Grok does not replace
Codex credentials. When both are present, serve exposes both Codex and Grok
models through the same local OpenAI-compatible and Anthropic-compatible routes.
Use --provider grok only when you intentionally want to serve one provider.
If rotom daemon is already running when you add a new provider, restart it so
the running service loads the new provider:
rotom daemon restartxAI may still restrict OAuth API access by account tier even when browser login succeeds.
List the model registry grouped by provider:
rotom models
rotom models --provider openai
rotom models --provider grokOpenAI-compatible chat request:
curl http://127.0.0.1:14550/v1/chat/completions \
-H 'content-type: application/json' \
-d '{
"model": "gpt-5.5",
"messages": [{"role": "user", "content": "hello"}]
}'Anthropic-compatible Messages request:
curl http://127.0.0.1:14550/v1/messages \
-H 'content-type: application/json' \
-H 'x-api-key: local-secret' \
-H 'anthropic-version: 2023-06-01' \
-d '{
"model": "gpt-5.5",
"max_tokens": 1024,
"messages": [{"role": "user", "content": "hello"}]
}'Optional local API key protection:
ROTOM_API_KEY=local-secret rotom serve
curl http://127.0.0.1:14550/v1/models -H 'authorization: Bearer local-secret'Interactive runtime configuration:
rotom config
rotom config show
rotom config resetThe config file is stored at ~/.rotom/config.json by default and is used as
the fallback source for rotom serve and rotom daemon install.
Update to the latest published release:
rotom updaterotom defaults unsupported Anthropic-native model ids such as
claude-sonnet-* to gpt-5.5. To override that fallback explicitly:
ROTOM_MODEL_FALLBACK=gpt-5.5 rotom serve --api-key local-secretrotom rewrites known unsupported Anthropic model ids to the effective
fallback before calling Codex. When you do not configure one explicitly, the
default fallback is gpt-5.5.
Common pitfalls:
- Do not set
ANTHROPIC_BASE_URLtohttp://127.0.0.1:14550/v1; Claude Code appends/v1/messagesitself. - Use a model that
/v1/modelsactually returns, such asgpt-5.5. If Claude Code defaults toclaude-sonnet-*, the request will fail because rotom proxies Codex models, not Anthropic-hosted model IDs. ANTHROPIC_AUTH_TOKENis only the local gateway key configured with--api-key; it is not your upstream OpenAI/Codex OAuth token.- If you prefer a background service, install the daemon first and then point
ANTHROPIC_BASE_URLat the daemon address instead of runningrotom servemanually.
Refresh stored OAuth tokens while the server is running:
curl -X POST http://127.0.0.1:14550/v1/auth/refresh \
-H 'authorization: Bearer local-secret'From the CLI, rotom refresh refreshes all saved providers. Use
rotom refresh --provider grok to refresh only one provider.
Check token expiry, account metadata, and rate-limit windows:
rotom statusFetch the same status data over HTTP:
curl http://127.0.0.1:14550/v1/status \
-H 'authorization: Bearer local-secret'Example fields:
{
"account_id": "acc_123",
"token": {
"expires_at_local": "2026-05-05 12:15:07 +08:00",
"remaining_seconds": 813427
},
"account": {
"email": "user@example.com",
"plan": "chatgptpro",
"has_active_subscription": true
}
}Install rotom as a per-user background daemon:
rotom daemon install
rotom daemon reinstall
rotom daemon start
rotom daemon status
rotom daemon restart
rotom daemon stop
rotom daemon uninstallOn macOS, rotom installs a LaunchAgent at
~/Library/LaunchAgents/com.rotom.daemon.plist. On Linux, it installs a
systemd user unit at ~/.config/systemd/user/rotom.service.
Windows does not currently implement native daemon/service management; use WSL
and run the Linux build there if you need rotom daemon commands.
On Linux, inspect the per-user service with:
rotom daemon status
systemctl --user status rotom.serviceThe daemon runs rotom serve with the options passed at install time:
rotom daemon install \
--bind 127.0.0.1:14550 \
--api-key local-secretModels returned by /v1/models include the OpenAI/Codex registry:
gpt-5.1
gpt-5.1-codex-max
gpt-5.1-codex-mini
gpt-5.2
gpt-5.2-codex
gpt-5.3-codex
gpt-5.3-codex-spark
gpt-5.4
gpt-5.4-mini
gpt-5.5
and the Grok registry:
grok-4.3
grok-4.3-fast
grok-4
Credentials are stored at ~/.rotom/auth.json by default. Override with
--auth-file, ROTOM_AUTH_FILE, or ROTOM_HOME.
Runtime config supports model_fallback, and the CLI accepts
--model-fallback / ROTOM_MODEL_FALLBACK. When unset, rotom defaults the
fallback to gpt-5.5.
OpenAI compatibility currently covers:
GET /v1/modelsPOST /v1/chat/completionsPOST /v1/responsesPOST /v1/images/generationsPOST /v1/responses/compactPOST /v1/responses/input_tokens
On POST /v1/chat/completions, rotom accepts common OpenAI compatibility
fields such as temperature, max_tokens, max_completion_tokens, and
max_output_tokens, but the current Codex upstream rejects those parameters.
rotom therefore accepts them without error and omits them from the upstream
Codex request, so they should be treated as compatibility no-ops rather than
effective sampling or output-length controls.
/v1/responses supports previous_response_id only within the same running
rotom process. rotom intentionally keeps this local state in memory.
Image generation is exposed in two compatibility shapes:
- OpenAI-style
POST /v1/images/generations - OpenAI Responses hosted tool
{"type":"image_generation"}
Current image-generation caveats:
- OpenAI
POST /v1/responsessupports streaming image-generation events POST /v1/images/generationsremains non-streaming- Anthropic
POST /v1/messagesimage generation streaming is exposed as a rotom extension that emitsimagecontent blocks only once the upstream response completes - generated images are returned as base64 payloads
- Anthropic compatibility uses a rotom extension that returns
content: [{"type":"image","source":{"type":"base64",...}}]onPOST /v1/messageswhen the request includes a tool namedimage_generation
Anthropic compatibility currently covers:
GET /v1/modelswith ananthropic-versionheaderPOST /v1/messagesPOST /v1/messages/count_tokensPOST /v1/messages/batchesGET /v1/messages/batchesGET /v1/messages/batches/{batch_id}POST /v1/messages/batches/{batch_id}/cancelDELETE /v1/messages/batches/{batch_id}GET /v1/messages/batches/{batch_id}/resultsx-api-keyorauthorization: Bearer ...local auth- Anthropic-style SSE events for streaming text and tool use
Message batches execute asynchronously in a background task. Cancellation is
best-effort at request boundaries inside the batch worker: requests that have
already started are allowed to finish, while not-yet-started requests are
marked as canceled.
The implementation intentionally follows Ollama's compatibility strategy where possible: Anthropic headers are accepted, locally configured auth is enforced, and unsupported advanced Anthropic-only features are ignored rather than rejected when possible.
The default Codex OAuth flow follows OpenClaw/pi-ai's Codex flow: PKCE, manual
paste of the http://localhost:1455/auth/callback?... redirect URL, token
exchange at https://auth.openai.com/oauth/token, and Codex requests to
https://chatgpt.com/backend-api/codex/responses. Grok OAuth uses xAI OIDC
discovery, PKCE, manual callback paste, and xAI Responses requests under
https://api.x.ai/v1.
rotom is an unofficial compatibility tool. It is not affiliated with, endorsed by, or supported by OpenAI, Anthropic, or xAI.
You are responsible for making sure your usage complies with the terms, policies, account restrictions, and data-handling obligations that apply to your upstream account and deployment environment. In particular, do not assume that personal OAuth-backed access can be shared, resold, or safely exposed as a multi-user hosted service. The LGPLv3 license for this repository does not change those upstream restrictions.
Copyright (c) 2026 rotom contributors.
Licensed under the GNU Lesser General Public License v3.0 only. See LICENSE.


