CCProxy is a local, plugin-based reverse proxy that unifies access to multiple AI providers (e.g., Claude SDK/API and OpenAI Codex) behind a consistent API. It ships with bundled plugins for providers, logging, tracing, metrics, analytics, and more.
- Anthropic Claude API/SDK (OAuth2 flow or Claude CLI/SDK token files)
- OpenAI Codex (ChatGPT backend Responses API using OAuth for paid/pro accounts)
- GitHub Copilot (chat and completions for free, paid, or business accounts)
Each provider adapter exposes the same surface area: OpenAI Chat
Completions, OpenAI Responses, and Anthropic Messages. The proxy maintains a
shared model-mapping layer so you can reuse the same model identifier
across providers without rewriting client code.
Authentication can reuse existing provider files (e.g., Claude CLI SDK
tokens and the Codex CLI credential store), or you can run
ccproxy auth login <provider> to complete the OAuth flow from the CLI;
stored secrets are picked up automatically by the proxy.
CCProxy's plugin system lets you add instrumentation and storage layers without patching the core server. Bundled plugins currently include:
access_log: structured access logging for client and provider trafficanalytics: DuckDB-backed analytics APIs for captured request logsclaude_api: Anthropic Claude HTTP API adapter with health and metricsclaude_sdk: local Claude CLI/SDK adapter with session poolingcodex: OpenAI Codex provider adapter with OAuth supportcommand_replay: generatescurl/xhcommands for captured requestscopilot: GitHub Copilot provider adapter with OAuth token managementcredential_balancer: rotates upstream credentials based on healthdashboard: serves the CCProxy dashboard SPA and APIsdocker: runs providers inside Docker via CLI extensionsduckdb_storage: exposes DuckDB-backed storage for logs and analyticsmax_tokens: normalizesmax_tokensfields to provider limitsmetrics: Prometheus-compatible metrics with optional Pushgatewayoauth_claude: standalone OAuth provider for Claude integrationsoauth_codex: standalone OAuth provider for Codex integrationspermissions: interactive approval flow for privileged tool actionspricing: caches model pricing data for cost-aware featuresrequest_tracer: detailed request/response tracing for debugging
Shared helpers such as
claude_shared provide metadata
consumed by the Claude plugins. Each plugin directory contains its own README
with configuration examples.
- Docs site entry:
docs/index.md - Getting started:
docs/getting-started/quickstart.md - Configuration reference:
docs/getting-started/configuration.md - Examples:
docs/examples.md - Migration (0.2):
docs/migration/0.2-plugin-first.md
The plugin system is enabled by default (enable_plugins = true), and all
discovered plugins load automatically when no additional filters are set. Use
these knobs to adjust what runs:
enabled_plugins: optional allow list; when set, only the listed plugins run.disabled_plugins: optional block list applied whenenabled_pluginsis not set.plugins.<name>.enabled: per-plugin flag (defaults totrue) that you can override in TOML or environment variables. Any plugin set tofalseis added to the deny list alongsidedisabled_pluginsduring startup.
During startup we merge disabled_plugins and any plugins.<name>.enabled = false
entries into a single deny list. At runtime the loader checks the allow list
first and then confirms the plugin is not deny listed. Configure plugins under
plugins.<name> in TOML or via nested environment variables.
Use ccproxy plugins list to inspect discovered plugins and
ccproxy plugins settings <name> to review configuration fields.
enable_plugins = true
# enabled_plugins = ["metrics", "analytics"] # Optional allow list
disabled_plugins = ["duckdb_storage"] # Optional block list
[plugins.access_log]
client_enabled = true
client_format = "structured"
client_log_file = "/tmp/ccproxy/access.log"
[plugins.request_tracer]
json_logs_enabled = true
raw_http_enabled = true
log_dir = "/tmp/ccproxy/traces"
[plugins.duckdb_storage]
enabled = false
[plugins.analytics]
enabled = true
# Metrics plugin
[plugins.metrics]
enabled = true
# pushgateway_enabled = true
# pushgateway_url = "http://localhost:9091"
# pushgateway_job = "ccproxy"
# pushgateway_push_interval = 60export DISABLED_PLUGINS="duckdb_storage" # Optional block list
export PLUGINS__ACCESS_LOG__ENABLED=true
export PLUGINS__ACCESS_LOG__CLIENT_ENABLED=true
export PLUGINS__ACCESS_LOG__CLIENT_FORMAT=structured
export PLUGINS__ACCESS_LOG__CLIENT_LOG_FILE=/tmp/ccproxy/access.log
export PLUGINS__REQUEST_TRACER__ENABLED=true
export PLUGINS__REQUEST_TRACER__JSON_LOGS_ENABLED=true
export PLUGINS__REQUEST_TRACER__RAW_HTTP_ENABLED=true
export PLUGINS__REQUEST_TRACER__LOG_DIR=/tmp/ccproxy/traces
export PLUGINS__DUCKDB_STORAGE__ENABLED=true
export PLUGINS__ANALYTICS__ENABLED=true
export PLUGINS__METRICS__ENABLED=true
# export PLUGINS__METRICS__PUSHGATEWAY_ENABLED=true
# export PLUGINS__METRICS__PUSHGATEWAY_URL=http://localhost:9091To install the latest stable release without cloning the repository, use uvx
to grab the published wheel and launch the CLI:
uvx --with "ccproxy-api[all]" ccproxy serve --port 8000If you prefer pipx, install the package (optionally with extras) and use the
local shim:
pipx install "ccproxy-api[all]"
ccproxy serve # default on localhost:8000See LICENSE.