ccproxy-api 0.1.6__py3-none-any.whl → 0.2.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- ccproxy/api/__init__.py +1 -15
- ccproxy/api/app.py +439 -212
- ccproxy/api/bootstrap.py +30 -0
- ccproxy/api/decorators.py +85 -0
- ccproxy/api/dependencies.py +145 -176
- ccproxy/api/format_validation.py +54 -0
- ccproxy/api/middleware/cors.py +6 -3
- ccproxy/api/middleware/errors.py +402 -530
- ccproxy/api/middleware/hooks.py +563 -0
- ccproxy/api/middleware/normalize_headers.py +59 -0
- ccproxy/api/middleware/request_id.py +35 -16
- ccproxy/api/middleware/streaming_hooks.py +292 -0
- ccproxy/api/routes/__init__.py +5 -14
- ccproxy/api/routes/health.py +39 -672
- ccproxy/api/routes/plugins.py +277 -0
- ccproxy/auth/__init__.py +2 -19
- ccproxy/auth/bearer.py +25 -15
- ccproxy/auth/dependencies.py +123 -157
- ccproxy/auth/exceptions.py +0 -12
- ccproxy/auth/manager.py +35 -49
- ccproxy/auth/managers/__init__.py +10 -0
- ccproxy/auth/managers/base.py +523 -0
- ccproxy/auth/managers/base_enhanced.py +63 -0
- ccproxy/auth/managers/token_snapshot.py +77 -0
- ccproxy/auth/models/base.py +65 -0
- ccproxy/auth/models/credentials.py +40 -0
- ccproxy/auth/oauth/__init__.py +4 -18
- ccproxy/auth/oauth/base.py +533 -0
- ccproxy/auth/oauth/cli_errors.py +37 -0
- ccproxy/auth/oauth/flows.py +430 -0
- ccproxy/auth/oauth/protocol.py +366 -0
- ccproxy/auth/oauth/registry.py +408 -0
- ccproxy/auth/oauth/router.py +396 -0
- ccproxy/auth/oauth/routes.py +186 -113
- ccproxy/auth/oauth/session.py +151 -0
- ccproxy/auth/oauth/templates.py +342 -0
- ccproxy/auth/storage/__init__.py +2 -5
- ccproxy/auth/storage/base.py +279 -5
- ccproxy/auth/storage/generic.py +134 -0
- ccproxy/cli/__init__.py +1 -2
- ccproxy/cli/_settings_help.py +351 -0
- ccproxy/cli/commands/auth.py +1519 -793
- ccproxy/cli/commands/config/commands.py +209 -276
- ccproxy/cli/commands/plugins.py +669 -0
- ccproxy/cli/commands/serve.py +75 -810
- ccproxy/cli/commands/status.py +254 -0
- ccproxy/cli/decorators.py +83 -0
- ccproxy/cli/helpers.py +22 -60
- ccproxy/cli/main.py +359 -10
- ccproxy/cli/options/claude_options.py +0 -25
- ccproxy/config/__init__.py +7 -11
- ccproxy/config/core.py +227 -0
- ccproxy/config/env_generator.py +232 -0
- ccproxy/config/runtime.py +67 -0
- ccproxy/config/security.py +36 -3
- ccproxy/config/settings.py +382 -441
- ccproxy/config/toml_generator.py +299 -0
- ccproxy/config/utils.py +452 -0
- ccproxy/core/__init__.py +7 -271
- ccproxy/{_version.py → core/_version.py} +16 -3
- ccproxy/core/async_task_manager.py +516 -0
- ccproxy/core/async_utils.py +47 -14
- ccproxy/core/auth/__init__.py +6 -0
- ccproxy/core/constants.py +16 -50
- ccproxy/core/errors.py +53 -0
- ccproxy/core/id_utils.py +20 -0
- ccproxy/core/interfaces.py +16 -123
- ccproxy/core/logging.py +473 -18
- ccproxy/core/plugins/__init__.py +77 -0
- ccproxy/core/plugins/cli_discovery.py +211 -0
- ccproxy/core/plugins/declaration.py +455 -0
- ccproxy/core/plugins/discovery.py +604 -0
- ccproxy/core/plugins/factories.py +967 -0
- ccproxy/core/plugins/hooks/__init__.py +30 -0
- ccproxy/core/plugins/hooks/base.py +58 -0
- ccproxy/core/plugins/hooks/events.py +46 -0
- ccproxy/core/plugins/hooks/implementations/__init__.py +16 -0
- ccproxy/core/plugins/hooks/implementations/formatters/__init__.py +11 -0
- ccproxy/core/plugins/hooks/implementations/formatters/json.py +552 -0
- ccproxy/core/plugins/hooks/implementations/formatters/raw.py +370 -0
- ccproxy/core/plugins/hooks/implementations/http_tracer.py +431 -0
- ccproxy/core/plugins/hooks/layers.py +44 -0
- ccproxy/core/plugins/hooks/manager.py +186 -0
- ccproxy/core/plugins/hooks/registry.py +139 -0
- ccproxy/core/plugins/hooks/thread_manager.py +203 -0
- ccproxy/core/plugins/hooks/types.py +22 -0
- ccproxy/core/plugins/interfaces.py +416 -0
- ccproxy/core/plugins/loader.py +166 -0
- ccproxy/core/plugins/middleware.py +233 -0
- ccproxy/core/plugins/models.py +59 -0
- ccproxy/core/plugins/protocol.py +180 -0
- ccproxy/core/plugins/runtime.py +519 -0
- ccproxy/{observability/context.py → core/request_context.py} +137 -94
- ccproxy/core/status_report.py +211 -0
- ccproxy/core/transformers.py +13 -8
- ccproxy/data/claude_headers_fallback.json +558 -0
- ccproxy/data/codex_headers_fallback.json +121 -0
- ccproxy/http/__init__.py +30 -0
- ccproxy/http/base.py +95 -0
- ccproxy/http/client.py +323 -0
- ccproxy/http/hooks.py +642 -0
- ccproxy/http/pool.py +279 -0
- ccproxy/llms/formatters/__init__.py +7 -0
- ccproxy/llms/formatters/anthropic_to_openai/__init__.py +55 -0
- ccproxy/llms/formatters/anthropic_to_openai/errors.py +65 -0
- ccproxy/llms/formatters/anthropic_to_openai/requests.py +356 -0
- ccproxy/llms/formatters/anthropic_to_openai/responses.py +153 -0
- ccproxy/llms/formatters/anthropic_to_openai/streams.py +1546 -0
- ccproxy/llms/formatters/base.py +140 -0
- ccproxy/llms/formatters/base_model.py +33 -0
- ccproxy/llms/formatters/common/__init__.py +51 -0
- ccproxy/llms/formatters/common/identifiers.py +48 -0
- ccproxy/llms/formatters/common/streams.py +254 -0
- ccproxy/llms/formatters/common/thinking.py +74 -0
- ccproxy/llms/formatters/common/usage.py +135 -0
- ccproxy/llms/formatters/constants.py +55 -0
- ccproxy/llms/formatters/context.py +116 -0
- ccproxy/llms/formatters/mapping.py +33 -0
- ccproxy/llms/formatters/openai_to_anthropic/__init__.py +55 -0
- ccproxy/llms/formatters/openai_to_anthropic/_helpers.py +141 -0
- ccproxy/llms/formatters/openai_to_anthropic/errors.py +53 -0
- ccproxy/llms/formatters/openai_to_anthropic/requests.py +674 -0
- ccproxy/llms/formatters/openai_to_anthropic/responses.py +285 -0
- ccproxy/llms/formatters/openai_to_anthropic/streams.py +530 -0
- ccproxy/llms/formatters/openai_to_openai/__init__.py +53 -0
- ccproxy/llms/formatters/openai_to_openai/_helpers.py +325 -0
- ccproxy/llms/formatters/openai_to_openai/errors.py +6 -0
- ccproxy/llms/formatters/openai_to_openai/requests.py +388 -0
- ccproxy/llms/formatters/openai_to_openai/responses.py +594 -0
- ccproxy/llms/formatters/openai_to_openai/streams.py +1832 -0
- ccproxy/llms/formatters/utils.py +306 -0
- ccproxy/llms/models/__init__.py +9 -0
- ccproxy/llms/models/anthropic.py +619 -0
- ccproxy/llms/models/openai.py +844 -0
- ccproxy/llms/streaming/__init__.py +26 -0
- ccproxy/llms/streaming/accumulators.py +1074 -0
- ccproxy/llms/streaming/formatters.py +251 -0
- ccproxy/{adapters/openai/streaming.py → llms/streaming/processors.py} +193 -240
- ccproxy/models/__init__.py +8 -159
- ccproxy/models/detection.py +92 -193
- ccproxy/models/provider.py +75 -0
- ccproxy/plugins/access_log/README.md +32 -0
- ccproxy/plugins/access_log/__init__.py +20 -0
- ccproxy/plugins/access_log/config.py +33 -0
- ccproxy/plugins/access_log/formatter.py +126 -0
- ccproxy/plugins/access_log/hook.py +763 -0
- ccproxy/plugins/access_log/logger.py +254 -0
- ccproxy/plugins/access_log/plugin.py +137 -0
- ccproxy/plugins/access_log/writer.py +109 -0
- ccproxy/plugins/analytics/README.md +24 -0
- ccproxy/plugins/analytics/__init__.py +1 -0
- ccproxy/plugins/analytics/config.py +5 -0
- ccproxy/plugins/analytics/ingest.py +85 -0
- ccproxy/plugins/analytics/models.py +97 -0
- ccproxy/plugins/analytics/plugin.py +121 -0
- ccproxy/plugins/analytics/routes.py +163 -0
- ccproxy/plugins/analytics/service.py +284 -0
- ccproxy/plugins/claude_api/README.md +29 -0
- ccproxy/plugins/claude_api/__init__.py +10 -0
- ccproxy/plugins/claude_api/adapter.py +829 -0
- ccproxy/plugins/claude_api/config.py +52 -0
- ccproxy/plugins/claude_api/detection_service.py +461 -0
- ccproxy/plugins/claude_api/health.py +175 -0
- ccproxy/plugins/claude_api/hooks.py +284 -0
- ccproxy/plugins/claude_api/models.py +256 -0
- ccproxy/plugins/claude_api/plugin.py +298 -0
- ccproxy/plugins/claude_api/routes.py +118 -0
- ccproxy/plugins/claude_api/streaming_metrics.py +68 -0
- ccproxy/plugins/claude_api/tasks.py +84 -0
- ccproxy/plugins/claude_sdk/README.md +35 -0
- ccproxy/plugins/claude_sdk/__init__.py +80 -0
- ccproxy/plugins/claude_sdk/adapter.py +749 -0
- ccproxy/plugins/claude_sdk/auth.py +57 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/client.py +63 -39
- ccproxy/plugins/claude_sdk/config.py +210 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/converter.py +6 -6
- ccproxy/plugins/claude_sdk/detection_service.py +163 -0
- ccproxy/{services/claude_sdk_service.py → plugins/claude_sdk/handler.py} +123 -304
- ccproxy/plugins/claude_sdk/health.py +113 -0
- ccproxy/plugins/claude_sdk/hooks.py +115 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/manager.py +42 -32
- ccproxy/{claude_sdk → plugins/claude_sdk}/message_queue.py +8 -8
- ccproxy/{models/claude_sdk.py → plugins/claude_sdk/models.py} +64 -16
- ccproxy/plugins/claude_sdk/options.py +154 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/parser.py +23 -5
- ccproxy/plugins/claude_sdk/plugin.py +269 -0
- ccproxy/plugins/claude_sdk/routes.py +104 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/session_client.py +124 -12
- ccproxy/plugins/claude_sdk/session_pool.py +700 -0
- ccproxy/{claude_sdk → plugins/claude_sdk}/stream_handle.py +48 -43
- ccproxy/{claude_sdk → plugins/claude_sdk}/stream_worker.py +22 -18
- ccproxy/{claude_sdk → plugins/claude_sdk}/streaming.py +50 -16
- ccproxy/plugins/claude_sdk/tasks.py +97 -0
- ccproxy/plugins/claude_shared/README.md +18 -0
- ccproxy/plugins/claude_shared/__init__.py +12 -0
- ccproxy/plugins/claude_shared/model_defaults.py +171 -0
- ccproxy/plugins/codex/README.md +35 -0
- ccproxy/plugins/codex/__init__.py +6 -0
- ccproxy/plugins/codex/adapter.py +635 -0
- ccproxy/{config/codex.py → plugins/codex/config.py} +78 -12
- ccproxy/plugins/codex/detection_service.py +544 -0
- ccproxy/plugins/codex/health.py +162 -0
- ccproxy/plugins/codex/hooks.py +263 -0
- ccproxy/plugins/codex/model_defaults.py +39 -0
- ccproxy/plugins/codex/models.py +263 -0
- ccproxy/plugins/codex/plugin.py +275 -0
- ccproxy/plugins/codex/routes.py +129 -0
- ccproxy/plugins/codex/streaming_metrics.py +324 -0
- ccproxy/plugins/codex/tasks.py +106 -0
- ccproxy/plugins/codex/utils/__init__.py +1 -0
- ccproxy/plugins/codex/utils/sse_parser.py +106 -0
- ccproxy/plugins/command_replay/README.md +34 -0
- ccproxy/plugins/command_replay/__init__.py +17 -0
- ccproxy/plugins/command_replay/config.py +133 -0
- ccproxy/plugins/command_replay/formatter.py +432 -0
- ccproxy/plugins/command_replay/hook.py +294 -0
- ccproxy/plugins/command_replay/plugin.py +161 -0
- ccproxy/plugins/copilot/README.md +39 -0
- ccproxy/plugins/copilot/__init__.py +11 -0
- ccproxy/plugins/copilot/adapter.py +465 -0
- ccproxy/plugins/copilot/config.py +155 -0
- ccproxy/plugins/copilot/data/copilot_fallback.json +41 -0
- ccproxy/plugins/copilot/detection_service.py +255 -0
- ccproxy/plugins/copilot/manager.py +275 -0
- ccproxy/plugins/copilot/model_defaults.py +284 -0
- ccproxy/plugins/copilot/models.py +148 -0
- ccproxy/plugins/copilot/oauth/__init__.py +16 -0
- ccproxy/plugins/copilot/oauth/client.py +494 -0
- ccproxy/plugins/copilot/oauth/models.py +385 -0
- ccproxy/plugins/copilot/oauth/provider.py +602 -0
- ccproxy/plugins/copilot/oauth/storage.py +170 -0
- ccproxy/plugins/copilot/plugin.py +360 -0
- ccproxy/plugins/copilot/routes.py +294 -0
- ccproxy/plugins/credential_balancer/README.md +124 -0
- ccproxy/plugins/credential_balancer/__init__.py +6 -0
- ccproxy/plugins/credential_balancer/config.py +270 -0
- ccproxy/plugins/credential_balancer/factory.py +415 -0
- ccproxy/plugins/credential_balancer/hook.py +51 -0
- ccproxy/plugins/credential_balancer/manager.py +587 -0
- ccproxy/plugins/credential_balancer/plugin.py +146 -0
- ccproxy/plugins/dashboard/README.md +25 -0
- ccproxy/plugins/dashboard/__init__.py +1 -0
- ccproxy/plugins/dashboard/config.py +8 -0
- ccproxy/plugins/dashboard/plugin.py +71 -0
- ccproxy/plugins/dashboard/routes.py +67 -0
- ccproxy/plugins/docker/README.md +32 -0
- ccproxy/{docker → plugins/docker}/__init__.py +3 -0
- ccproxy/{docker → plugins/docker}/adapter.py +108 -10
- ccproxy/plugins/docker/config.py +82 -0
- ccproxy/{docker → plugins/docker}/docker_path.py +4 -3
- ccproxy/{docker → plugins/docker}/middleware.py +2 -2
- ccproxy/plugins/docker/plugin.py +198 -0
- ccproxy/{docker → plugins/docker}/stream_process.py +3 -3
- ccproxy/plugins/duckdb_storage/README.md +26 -0
- ccproxy/plugins/duckdb_storage/__init__.py +1 -0
- ccproxy/plugins/duckdb_storage/config.py +22 -0
- ccproxy/plugins/duckdb_storage/plugin.py +128 -0
- ccproxy/plugins/duckdb_storage/routes.py +51 -0
- ccproxy/plugins/duckdb_storage/storage.py +633 -0
- ccproxy/plugins/max_tokens/README.md +38 -0
- ccproxy/plugins/max_tokens/__init__.py +12 -0
- ccproxy/plugins/max_tokens/adapter.py +235 -0
- ccproxy/plugins/max_tokens/config.py +86 -0
- ccproxy/plugins/max_tokens/models.py +53 -0
- ccproxy/plugins/max_tokens/plugin.py +200 -0
- ccproxy/plugins/max_tokens/service.py +271 -0
- ccproxy/plugins/max_tokens/token_limits.json +54 -0
- ccproxy/plugins/metrics/README.md +35 -0
- ccproxy/plugins/metrics/__init__.py +10 -0
- ccproxy/{observability/metrics.py → plugins/metrics/collector.py} +20 -153
- ccproxy/plugins/metrics/config.py +85 -0
- ccproxy/plugins/metrics/grafana/dashboards/ccproxy-dashboard.json +1720 -0
- ccproxy/plugins/metrics/hook.py +403 -0
- ccproxy/plugins/metrics/plugin.py +268 -0
- ccproxy/{observability → plugins/metrics}/pushgateway.py +57 -59
- ccproxy/plugins/metrics/routes.py +107 -0
- ccproxy/plugins/metrics/tasks.py +117 -0
- ccproxy/plugins/oauth_claude/README.md +35 -0
- ccproxy/plugins/oauth_claude/__init__.py +14 -0
- ccproxy/plugins/oauth_claude/client.py +270 -0
- ccproxy/plugins/oauth_claude/config.py +84 -0
- ccproxy/plugins/oauth_claude/manager.py +482 -0
- ccproxy/plugins/oauth_claude/models.py +266 -0
- ccproxy/plugins/oauth_claude/plugin.py +149 -0
- ccproxy/plugins/oauth_claude/provider.py +571 -0
- ccproxy/plugins/oauth_claude/storage.py +212 -0
- ccproxy/plugins/oauth_codex/README.md +38 -0
- ccproxy/plugins/oauth_codex/__init__.py +14 -0
- ccproxy/plugins/oauth_codex/client.py +224 -0
- ccproxy/plugins/oauth_codex/config.py +95 -0
- ccproxy/plugins/oauth_codex/manager.py +256 -0
- ccproxy/plugins/oauth_codex/models.py +239 -0
- ccproxy/plugins/oauth_codex/plugin.py +146 -0
- ccproxy/plugins/oauth_codex/provider.py +574 -0
- ccproxy/plugins/oauth_codex/storage.py +92 -0
- ccproxy/plugins/permissions/README.md +28 -0
- ccproxy/plugins/permissions/__init__.py +22 -0
- ccproxy/plugins/permissions/config.py +28 -0
- ccproxy/{cli/commands/permission_handler.py → plugins/permissions/handlers/cli.py} +49 -25
- ccproxy/plugins/permissions/handlers/protocol.py +33 -0
- ccproxy/plugins/permissions/handlers/terminal.py +675 -0
- ccproxy/{api/routes → plugins/permissions}/mcp.py +34 -7
- ccproxy/{models/permissions.py → plugins/permissions/models.py} +65 -1
- ccproxy/plugins/permissions/plugin.py +153 -0
- ccproxy/{api/routes/permissions.py → plugins/permissions/routes.py} +20 -16
- ccproxy/{api/services/permission_service.py → plugins/permissions/service.py} +65 -11
- ccproxy/{api → plugins/permissions}/ui/permission_handler_protocol.py +1 -1
- ccproxy/{api → plugins/permissions}/ui/terminal_permission_handler.py +66 -10
- ccproxy/plugins/pricing/README.md +34 -0
- ccproxy/plugins/pricing/__init__.py +6 -0
- ccproxy/{pricing → plugins/pricing}/cache.py +7 -6
- ccproxy/{config/pricing.py → plugins/pricing/config.py} +32 -6
- ccproxy/plugins/pricing/exceptions.py +35 -0
- ccproxy/plugins/pricing/loader.py +440 -0
- ccproxy/{pricing → plugins/pricing}/models.py +13 -23
- ccproxy/plugins/pricing/plugin.py +169 -0
- ccproxy/plugins/pricing/service.py +191 -0
- ccproxy/plugins/pricing/tasks.py +300 -0
- ccproxy/{pricing → plugins/pricing}/updater.py +86 -72
- ccproxy/plugins/pricing/utils.py +99 -0
- ccproxy/plugins/request_tracer/README.md +40 -0
- ccproxy/plugins/request_tracer/__init__.py +7 -0
- ccproxy/plugins/request_tracer/config.py +120 -0
- ccproxy/plugins/request_tracer/hook.py +415 -0
- ccproxy/plugins/request_tracer/plugin.py +255 -0
- ccproxy/scheduler/__init__.py +2 -14
- ccproxy/scheduler/core.py +26 -41
- ccproxy/scheduler/manager.py +63 -107
- ccproxy/scheduler/registry.py +6 -32
- ccproxy/scheduler/tasks.py +346 -314
- ccproxy/services/__init__.py +0 -1
- ccproxy/services/adapters/__init__.py +11 -0
- ccproxy/services/adapters/base.py +123 -0
- ccproxy/services/adapters/chain_composer.py +88 -0
- ccproxy/services/adapters/chain_validation.py +44 -0
- ccproxy/services/adapters/chat_accumulator.py +200 -0
- ccproxy/services/adapters/delta_utils.py +142 -0
- ccproxy/services/adapters/format_adapter.py +136 -0
- ccproxy/services/adapters/format_context.py +11 -0
- ccproxy/services/adapters/format_registry.py +158 -0
- ccproxy/services/adapters/http_adapter.py +1045 -0
- ccproxy/services/adapters/mock_adapter.py +118 -0
- ccproxy/services/adapters/protocols.py +35 -0
- ccproxy/services/adapters/simple_converters.py +571 -0
- ccproxy/services/auth_registry.py +180 -0
- ccproxy/services/cache/__init__.py +6 -0
- ccproxy/services/cache/response_cache.py +261 -0
- ccproxy/services/cli_detection.py +437 -0
- ccproxy/services/config/__init__.py +6 -0
- ccproxy/services/config/proxy_configuration.py +111 -0
- ccproxy/services/container.py +256 -0
- ccproxy/services/factories.py +380 -0
- ccproxy/services/handler_config.py +76 -0
- ccproxy/services/interfaces.py +298 -0
- ccproxy/services/mocking/__init__.py +6 -0
- ccproxy/services/mocking/mock_handler.py +291 -0
- ccproxy/services/tracing/__init__.py +7 -0
- ccproxy/services/tracing/interfaces.py +61 -0
- ccproxy/services/tracing/null_tracer.py +57 -0
- ccproxy/streaming/__init__.py +23 -0
- ccproxy/streaming/buffer.py +1056 -0
- ccproxy/streaming/deferred.py +897 -0
- ccproxy/streaming/handler.py +117 -0
- ccproxy/streaming/interfaces.py +77 -0
- ccproxy/streaming/simple_adapter.py +39 -0
- ccproxy/streaming/sse.py +109 -0
- ccproxy/streaming/sse_parser.py +127 -0
- ccproxy/templates/__init__.py +6 -0
- ccproxy/templates/plugin_scaffold.py +695 -0
- ccproxy/testing/endpoints/__init__.py +33 -0
- ccproxy/testing/endpoints/cli.py +215 -0
- ccproxy/testing/endpoints/config.py +874 -0
- ccproxy/testing/endpoints/console.py +57 -0
- ccproxy/testing/endpoints/models.py +100 -0
- ccproxy/testing/endpoints/runner.py +1903 -0
- ccproxy/testing/endpoints/tools.py +308 -0
- ccproxy/testing/mock_responses.py +70 -1
- ccproxy/testing/response_handlers.py +20 -0
- ccproxy/utils/__init__.py +0 -6
- ccproxy/utils/binary_resolver.py +476 -0
- ccproxy/utils/caching.py +327 -0
- ccproxy/utils/cli_logging.py +101 -0
- ccproxy/utils/command_line.py +251 -0
- ccproxy/utils/headers.py +228 -0
- ccproxy/utils/model_mapper.py +120 -0
- ccproxy/utils/startup_helpers.py +95 -342
- ccproxy/utils/version_checker.py +279 -6
- ccproxy_api-0.2.0.dist-info/METADATA +212 -0
- ccproxy_api-0.2.0.dist-info/RECORD +417 -0
- {ccproxy_api-0.1.6.dist-info → ccproxy_api-0.2.0.dist-info}/WHEEL +1 -1
- ccproxy_api-0.2.0.dist-info/entry_points.txt +24 -0
- ccproxy/__init__.py +0 -4
- ccproxy/adapters/__init__.py +0 -11
- ccproxy/adapters/base.py +0 -80
- ccproxy/adapters/codex/__init__.py +0 -11
- ccproxy/adapters/openai/__init__.py +0 -42
- ccproxy/adapters/openai/adapter.py +0 -953
- ccproxy/adapters/openai/models.py +0 -412
- ccproxy/adapters/openai/response_adapter.py +0 -355
- ccproxy/adapters/openai/response_models.py +0 -178
- ccproxy/api/middleware/headers.py +0 -49
- ccproxy/api/middleware/logging.py +0 -180
- ccproxy/api/middleware/request_content_logging.py +0 -297
- ccproxy/api/middleware/server_header.py +0 -58
- ccproxy/api/responses.py +0 -89
- ccproxy/api/routes/claude.py +0 -371
- ccproxy/api/routes/codex.py +0 -1231
- ccproxy/api/routes/metrics.py +0 -1029
- ccproxy/api/routes/proxy.py +0 -211
- ccproxy/api/services/__init__.py +0 -6
- ccproxy/auth/conditional.py +0 -84
- ccproxy/auth/credentials_adapter.py +0 -93
- ccproxy/auth/models.py +0 -118
- ccproxy/auth/oauth/models.py +0 -48
- ccproxy/auth/openai/__init__.py +0 -13
- ccproxy/auth/openai/credentials.py +0 -166
- ccproxy/auth/openai/oauth_client.py +0 -334
- ccproxy/auth/openai/storage.py +0 -184
- ccproxy/auth/storage/json_file.py +0 -158
- ccproxy/auth/storage/keyring.py +0 -189
- ccproxy/claude_sdk/__init__.py +0 -18
- ccproxy/claude_sdk/options.py +0 -194
- ccproxy/claude_sdk/session_pool.py +0 -550
- ccproxy/cli/docker/__init__.py +0 -34
- ccproxy/cli/docker/adapter_factory.py +0 -157
- ccproxy/cli/docker/params.py +0 -274
- ccproxy/config/auth.py +0 -153
- ccproxy/config/claude.py +0 -348
- ccproxy/config/cors.py +0 -79
- ccproxy/config/discovery.py +0 -95
- ccproxy/config/docker_settings.py +0 -264
- ccproxy/config/observability.py +0 -158
- ccproxy/config/reverse_proxy.py +0 -31
- ccproxy/config/scheduler.py +0 -108
- ccproxy/config/server.py +0 -86
- ccproxy/config/validators.py +0 -231
- ccproxy/core/codex_transformers.py +0 -389
- ccproxy/core/http.py +0 -328
- ccproxy/core/http_transformers.py +0 -812
- ccproxy/core/proxy.py +0 -143
- ccproxy/core/validators.py +0 -288
- ccproxy/models/errors.py +0 -42
- ccproxy/models/messages.py +0 -269
- ccproxy/models/requests.py +0 -107
- ccproxy/models/responses.py +0 -270
- ccproxy/models/types.py +0 -102
- ccproxy/observability/__init__.py +0 -51
- ccproxy/observability/access_logger.py +0 -457
- ccproxy/observability/sse_events.py +0 -303
- ccproxy/observability/stats_printer.py +0 -753
- ccproxy/observability/storage/__init__.py +0 -1
- ccproxy/observability/storage/duckdb_simple.py +0 -677
- ccproxy/observability/storage/models.py +0 -70
- ccproxy/observability/streaming_response.py +0 -107
- ccproxy/pricing/__init__.py +0 -19
- ccproxy/pricing/loader.py +0 -251
- ccproxy/services/claude_detection_service.py +0 -269
- ccproxy/services/codex_detection_service.py +0 -263
- ccproxy/services/credentials/__init__.py +0 -55
- ccproxy/services/credentials/config.py +0 -105
- ccproxy/services/credentials/manager.py +0 -561
- ccproxy/services/credentials/oauth_client.py +0 -481
- ccproxy/services/proxy_service.py +0 -1827
- ccproxy/static/.keep +0 -0
- ccproxy/utils/cost_calculator.py +0 -210
- ccproxy/utils/disconnection_monitor.py +0 -83
- ccproxy/utils/model_mapping.py +0 -199
- ccproxy/utils/models_provider.py +0 -150
- ccproxy/utils/simple_request_logger.py +0 -284
- ccproxy/utils/streaming_metrics.py +0 -199
- ccproxy_api-0.1.6.dist-info/METADATA +0 -615
- ccproxy_api-0.1.6.dist-info/RECORD +0 -189
- ccproxy_api-0.1.6.dist-info/entry_points.txt +0 -4
- /ccproxy/{api/middleware/auth.py → auth/models/__init__.py} +0 -0
- /ccproxy/{claude_sdk → plugins/claude_sdk}/exceptions.py +0 -0
- /ccproxy/{docker → plugins/docker}/models.py +0 -0
- /ccproxy/{docker → plugins/docker}/protocol.py +0 -0
- /ccproxy/{docker → plugins/docker}/validators.py +0 -0
- /ccproxy/{auth/oauth/storage.py → plugins/permissions/handlers/__init__.py} +0 -0
- /ccproxy/{api → plugins/permissions}/ui/__init__.py +0 -0
- {ccproxy_api-0.1.6.dist-info → ccproxy_api-0.2.0.dist-info}/licenses/LICENSE +0 -0
ccproxy/utils/version_checker.py
CHANGED
|
@@ -3,6 +3,8 @@
|
|
|
3
3
|
from __future__ import annotations
|
|
4
4
|
|
|
5
5
|
import json
|
|
6
|
+
import os
|
|
7
|
+
import re
|
|
6
8
|
from datetime import datetime
|
|
7
9
|
from pathlib import Path
|
|
8
10
|
from typing import Any
|
|
@@ -13,18 +15,26 @@ import structlog
|
|
|
13
15
|
from packaging import version as pkg_version
|
|
14
16
|
from pydantic import BaseModel
|
|
15
17
|
|
|
16
|
-
from ccproxy.
|
|
17
|
-
from ccproxy.
|
|
18
|
+
from ccproxy.config.utils import get_ccproxy_config_dir
|
|
19
|
+
from ccproxy.core._version import __version__
|
|
18
20
|
|
|
19
21
|
|
|
20
22
|
logger = structlog.get_logger(__name__)
|
|
21
23
|
|
|
22
24
|
|
|
25
|
+
BRANCH_OVERRIDE_ENV_VAR = "CCPROXY_VERSION_BRANCH"
|
|
26
|
+
GITHUB_API_BASE = "https://api.github.com/repos/CaddyGlow/ccproxy-api"
|
|
27
|
+
|
|
28
|
+
|
|
23
29
|
class VersionCheckState(BaseModel):
|
|
24
30
|
"""State tracking for version checks."""
|
|
25
31
|
|
|
26
32
|
last_check_at: datetime
|
|
27
33
|
latest_version_found: str | None = None
|
|
34
|
+
latest_branch_name: str | None = None
|
|
35
|
+
latest_branch_commit: str | None = None
|
|
36
|
+
running_version: str | None = None
|
|
37
|
+
running_commit: str | None = None
|
|
28
38
|
|
|
29
39
|
|
|
30
40
|
async def fetch_latest_github_version() -> str | None:
|
|
@@ -61,9 +71,23 @@ async def fetch_latest_github_version() -> str | None:
|
|
|
61
71
|
except httpx.HTTPStatusError as e:
|
|
62
72
|
logger.warning("github_version_http_error", status_code=e.response.status_code)
|
|
63
73
|
return None
|
|
74
|
+
except httpx.RequestError as e:
|
|
75
|
+
logger.warning(
|
|
76
|
+
"github_version_fetch_http_error",
|
|
77
|
+
error=str(e),
|
|
78
|
+
error_type=type(e).__name__,
|
|
79
|
+
)
|
|
80
|
+
return None
|
|
81
|
+
except (json.JSONDecodeError, KeyError, TypeError) as e:
|
|
82
|
+
logger.warning(
|
|
83
|
+
"github_version_parse_error",
|
|
84
|
+
error=str(e),
|
|
85
|
+
error_type=type(e).__name__,
|
|
86
|
+
)
|
|
87
|
+
return None
|
|
64
88
|
except Exception as e:
|
|
65
89
|
logger.warning(
|
|
66
|
-
"
|
|
90
|
+
"github_version_fetch_unexpected_error",
|
|
67
91
|
error=str(e),
|
|
68
92
|
error_type=type(e).__name__,
|
|
69
93
|
)
|
|
@@ -80,6 +104,38 @@ def get_current_version() -> str:
|
|
|
80
104
|
return __version__
|
|
81
105
|
|
|
82
106
|
|
|
107
|
+
def extract_commit_from_version(version: str) -> str | None:
|
|
108
|
+
"""Extract a git commit SHA from a setuptools-scm formatted version."""
|
|
109
|
+
|
|
110
|
+
match = re.search(r"\+g(?P<sha>[0-9a-f]{7,40})", version)
|
|
111
|
+
if match:
|
|
112
|
+
return match.group("sha")
|
|
113
|
+
return None
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
def commit_refs_match(current: str | None, latest: str | None) -> bool:
|
|
117
|
+
"""Return True when two commit references identify the same commit."""
|
|
118
|
+
|
|
119
|
+
if not current or not latest:
|
|
120
|
+
return current == latest
|
|
121
|
+
|
|
122
|
+
current_lower = current.lower()
|
|
123
|
+
latest_lower = latest.lower()
|
|
124
|
+
|
|
125
|
+
# Normalize shorter/longer pair for prefix comparison
|
|
126
|
+
if len(current_lower) <= len(latest_lower):
|
|
127
|
+
return latest_lower.startswith(current_lower)
|
|
128
|
+
|
|
129
|
+
return current_lower.startswith(latest_lower)
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
def get_branch_override() -> str | None:
|
|
133
|
+
"""Return branch override from environment if provided."""
|
|
134
|
+
|
|
135
|
+
env_branch = os.getenv(BRANCH_OVERRIDE_ENV_VAR, "").strip()
|
|
136
|
+
return env_branch or None
|
|
137
|
+
|
|
138
|
+
|
|
83
139
|
def compare_versions(current: str, latest: str) -> bool:
|
|
84
140
|
"""
|
|
85
141
|
Compare version strings to determine if an update is available.
|
|
@@ -94,10 +150,25 @@ def compare_versions(current: str, latest: str) -> bool:
|
|
|
94
150
|
try:
|
|
95
151
|
current_parsed = pkg_version.parse(current)
|
|
96
152
|
latest_parsed = pkg_version.parse(latest)
|
|
153
|
+
|
|
154
|
+
# For dev versions, compare base version instead
|
|
155
|
+
if current_parsed.is_devrelease:
|
|
156
|
+
current_base = pkg_version.parse(current_parsed.base_version)
|
|
157
|
+
return latest_parsed > current_base
|
|
158
|
+
|
|
97
159
|
return latest_parsed > current_parsed
|
|
160
|
+
except (ValueError, TypeError, AttributeError) as e:
|
|
161
|
+
logger.error(
|
|
162
|
+
"version_comparison_parse_error",
|
|
163
|
+
current=current,
|
|
164
|
+
latest=latest,
|
|
165
|
+
error=str(e),
|
|
166
|
+
error_type=type(e).__name__,
|
|
167
|
+
)
|
|
168
|
+
return False
|
|
98
169
|
except Exception as e:
|
|
99
170
|
logger.error(
|
|
100
|
-
"
|
|
171
|
+
"version_comparison_unexpected_error",
|
|
101
172
|
current=current,
|
|
102
173
|
latest=latest,
|
|
103
174
|
error=str(e),
|
|
@@ -106,6 +177,172 @@ def compare_versions(current: str, latest: str) -> bool:
|
|
|
106
177
|
return False
|
|
107
178
|
|
|
108
179
|
|
|
180
|
+
async def fetch_latest_branch_commit(branch: str) -> str | None:
|
|
181
|
+
"""Fetch the latest commit SHA for a given branch from GitHub."""
|
|
182
|
+
|
|
183
|
+
url = f"{GITHUB_API_BASE}/branches/{branch}"
|
|
184
|
+
headers = {
|
|
185
|
+
"User-Agent": f"ccproxy-api/{__version__}",
|
|
186
|
+
"Accept": "application/vnd.github.v3+json",
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
try:
|
|
190
|
+
async with httpx.AsyncClient(timeout=15.0) as client:
|
|
191
|
+
response = await client.get(url, headers=headers)
|
|
192
|
+
response.raise_for_status()
|
|
193
|
+
|
|
194
|
+
data: dict[str, Any] = response.json()
|
|
195
|
+
commit_info = data.get("commit", {})
|
|
196
|
+
latest_sha = commit_info.get("sha")
|
|
197
|
+
|
|
198
|
+
if isinstance(latest_sha, str) and latest_sha:
|
|
199
|
+
logger.debug(
|
|
200
|
+
"github_branch_commit_fetched",
|
|
201
|
+
branch=branch,
|
|
202
|
+
latest_sha=latest_sha,
|
|
203
|
+
)
|
|
204
|
+
return latest_sha
|
|
205
|
+
|
|
206
|
+
logger.warning(
|
|
207
|
+
"github_branch_commit_missing_sha",
|
|
208
|
+
branch=branch,
|
|
209
|
+
)
|
|
210
|
+
return None
|
|
211
|
+
|
|
212
|
+
except httpx.TimeoutException:
|
|
213
|
+
logger.warning("github_branch_commit_timeout", branch=branch)
|
|
214
|
+
return None
|
|
215
|
+
except httpx.HTTPStatusError as e:
|
|
216
|
+
logger.warning(
|
|
217
|
+
"github_branch_commit_http_error",
|
|
218
|
+
branch=branch,
|
|
219
|
+
status_code=e.response.status_code,
|
|
220
|
+
)
|
|
221
|
+
return None
|
|
222
|
+
except httpx.RequestError as e:
|
|
223
|
+
logger.warning(
|
|
224
|
+
"github_branch_commit_http_request_error",
|
|
225
|
+
branch=branch,
|
|
226
|
+
error=str(e),
|
|
227
|
+
error_type=type(e).__name__,
|
|
228
|
+
)
|
|
229
|
+
return None
|
|
230
|
+
except (json.JSONDecodeError, KeyError, TypeError) as e:
|
|
231
|
+
logger.warning(
|
|
232
|
+
"github_branch_commit_parse_error",
|
|
233
|
+
branch=branch,
|
|
234
|
+
error=str(e),
|
|
235
|
+
error_type=type(e).__name__,
|
|
236
|
+
)
|
|
237
|
+
return None
|
|
238
|
+
except Exception as e:
|
|
239
|
+
logger.warning(
|
|
240
|
+
"github_branch_commit_unexpected_error",
|
|
241
|
+
branch=branch,
|
|
242
|
+
error=str(e),
|
|
243
|
+
error_type=type(e).__name__,
|
|
244
|
+
)
|
|
245
|
+
return None
|
|
246
|
+
|
|
247
|
+
|
|
248
|
+
async def fetch_branch_names_for_commit(commit: str) -> list[str]:
|
|
249
|
+
"""Fetch branch names for which the given commit is the HEAD."""
|
|
250
|
+
|
|
251
|
+
url = f"{GITHUB_API_BASE}/commits/{commit}/branches-where-head"
|
|
252
|
+
headers = {
|
|
253
|
+
"User-Agent": f"ccproxy-api/{__version__}",
|
|
254
|
+
"Accept": "application/vnd.github.v3+json",
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
try:
|
|
258
|
+
async with httpx.AsyncClient(timeout=15.0) as client:
|
|
259
|
+
response = await client.get(url, headers=headers)
|
|
260
|
+
response.raise_for_status()
|
|
261
|
+
|
|
262
|
+
data = response.json()
|
|
263
|
+
if not isinstance(data, list):
|
|
264
|
+
logger.warning(
|
|
265
|
+
"github_commit_branches_unexpected_payload",
|
|
266
|
+
commit=commit,
|
|
267
|
+
payload_type=type(data).__name__,
|
|
268
|
+
)
|
|
269
|
+
return []
|
|
270
|
+
|
|
271
|
+
branch_names: list[str] = []
|
|
272
|
+
for entry in data:
|
|
273
|
+
if not isinstance(entry, dict):
|
|
274
|
+
continue
|
|
275
|
+
name = entry.get("name")
|
|
276
|
+
if isinstance(name, str) and name:
|
|
277
|
+
branch_names.append(name)
|
|
278
|
+
|
|
279
|
+
logger.debug(
|
|
280
|
+
"github_commit_branches_fetched",
|
|
281
|
+
commit=commit,
|
|
282
|
+
branch_count=len(branch_names),
|
|
283
|
+
)
|
|
284
|
+
return branch_names
|
|
285
|
+
|
|
286
|
+
except httpx.TimeoutException:
|
|
287
|
+
logger.warning("github_commit_branches_timeout", commit=commit)
|
|
288
|
+
return []
|
|
289
|
+
except httpx.HTTPStatusError as e:
|
|
290
|
+
logger.warning(
|
|
291
|
+
"github_commit_branches_http_error",
|
|
292
|
+
commit=commit,
|
|
293
|
+
status_code=e.response.status_code,
|
|
294
|
+
)
|
|
295
|
+
return []
|
|
296
|
+
except httpx.RequestError as e:
|
|
297
|
+
logger.warning(
|
|
298
|
+
"github_commit_branches_http_request_error",
|
|
299
|
+
commit=commit,
|
|
300
|
+
error=str(e),
|
|
301
|
+
error_type=type(e).__name__,
|
|
302
|
+
)
|
|
303
|
+
return []
|
|
304
|
+
except (json.JSONDecodeError, ValueError, TypeError) as e:
|
|
305
|
+
logger.warning(
|
|
306
|
+
"github_commit_branches_parse_error",
|
|
307
|
+
commit=commit,
|
|
308
|
+
error=str(e),
|
|
309
|
+
error_type=type(e).__name__,
|
|
310
|
+
)
|
|
311
|
+
return []
|
|
312
|
+
except Exception as e:
|
|
313
|
+
logger.warning(
|
|
314
|
+
"github_commit_branches_unexpected_error",
|
|
315
|
+
commit=commit,
|
|
316
|
+
error=str(e),
|
|
317
|
+
error_type=type(e).__name__,
|
|
318
|
+
)
|
|
319
|
+
return []
|
|
320
|
+
|
|
321
|
+
|
|
322
|
+
async def resolve_branch_for_commit(commit: str) -> str | None:
|
|
323
|
+
"""Resolve the branch name associated with the provided commit hash."""
|
|
324
|
+
|
|
325
|
+
override = get_branch_override()
|
|
326
|
+
if override:
|
|
327
|
+
return override
|
|
328
|
+
|
|
329
|
+
if not commit:
|
|
330
|
+
return None
|
|
331
|
+
|
|
332
|
+
branch_candidates = await fetch_branch_names_for_commit(commit)
|
|
333
|
+
if not branch_candidates:
|
|
334
|
+
return None
|
|
335
|
+
|
|
336
|
+
# Prefer mainline branches if available
|
|
337
|
+
preferred_order = ("main", "master", "develop", "dev")
|
|
338
|
+
for preferred in preferred_order:
|
|
339
|
+
if preferred in branch_candidates:
|
|
340
|
+
return preferred
|
|
341
|
+
|
|
342
|
+
# Otherwise return the first branch reported by GitHub
|
|
343
|
+
return branch_candidates[0]
|
|
344
|
+
|
|
345
|
+
|
|
109
346
|
async def load_check_state(path: Path) -> VersionCheckState | None:
|
|
110
347
|
"""
|
|
111
348
|
Load version check state from file.
|
|
@@ -124,9 +361,25 @@ async def load_check_state(path: Path) -> VersionCheckState | None:
|
|
|
124
361
|
content = await f.read()
|
|
125
362
|
data = json.loads(content)
|
|
126
363
|
return VersionCheckState(**data)
|
|
364
|
+
except (OSError, FileNotFoundError, PermissionError) as e:
|
|
365
|
+
logger.warning(
|
|
366
|
+
"version_check_state_load_file_error",
|
|
367
|
+
path=str(path),
|
|
368
|
+
error=str(e),
|
|
369
|
+
error_type=type(e).__name__,
|
|
370
|
+
)
|
|
371
|
+
return None
|
|
372
|
+
except (json.JSONDecodeError, ValueError, TypeError) as e:
|
|
373
|
+
logger.warning(
|
|
374
|
+
"version_check_state_load_parse_error",
|
|
375
|
+
path=str(path),
|
|
376
|
+
error=str(e),
|
|
377
|
+
error_type=type(e).__name__,
|
|
378
|
+
)
|
|
379
|
+
return None
|
|
127
380
|
except Exception as e:
|
|
128
381
|
logger.warning(
|
|
129
|
-
"
|
|
382
|
+
"version_check_state_load_unexpected_error",
|
|
130
383
|
path=str(path),
|
|
131
384
|
error=str(e),
|
|
132
385
|
error_type=type(e).__name__,
|
|
@@ -154,9 +407,23 @@ async def save_check_state(path: Path, state: VersionCheckState) -> None:
|
|
|
154
407
|
await f.write(json.dumps(state_dict, indent=2))
|
|
155
408
|
|
|
156
409
|
logger.debug("version_check_state_saved", path=str(path))
|
|
410
|
+
except (OSError, FileNotFoundError, PermissionError) as e:
|
|
411
|
+
logger.warning(
|
|
412
|
+
"version_check_state_save_file_error",
|
|
413
|
+
path=str(path),
|
|
414
|
+
error=str(e),
|
|
415
|
+
error_type=type(e).__name__,
|
|
416
|
+
)
|
|
417
|
+
except (TypeError, ValueError) as e:
|
|
418
|
+
logger.warning(
|
|
419
|
+
"version_check_state_save_serialize_error",
|
|
420
|
+
path=str(path),
|
|
421
|
+
error=str(e),
|
|
422
|
+
error_type=type(e).__name__,
|
|
423
|
+
)
|
|
157
424
|
except Exception as e:
|
|
158
425
|
logger.warning(
|
|
159
|
-
"
|
|
426
|
+
"version_check_state_save_unexpected_error",
|
|
160
427
|
path=str(path),
|
|
161
428
|
error=str(e),
|
|
162
429
|
error_type=type(e).__name__,
|
|
@@ -176,7 +443,13 @@ def get_version_check_state_path() -> Path:
|
|
|
176
443
|
__all__ = [
|
|
177
444
|
"VersionCheckState",
|
|
178
445
|
"fetch_latest_github_version",
|
|
446
|
+
"fetch_latest_branch_commit",
|
|
447
|
+
"fetch_branch_names_for_commit",
|
|
448
|
+
"resolve_branch_for_commit",
|
|
179
449
|
"get_current_version",
|
|
450
|
+
"extract_commit_from_version",
|
|
451
|
+
"commit_refs_match",
|
|
452
|
+
"get_branch_override",
|
|
180
453
|
"compare_versions",
|
|
181
454
|
"load_check_state",
|
|
182
455
|
"save_check_state",
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: ccproxy-api
|
|
3
|
+
Version: 0.2.0
|
|
4
|
+
Summary: API server that provides an Anthropic and OpenAI compatible interface over Claude Code, allowing to use your Claude OAuth account or over the API.
|
|
5
|
+
License-File: LICENSE
|
|
6
|
+
Requires-Python: >=3.11
|
|
7
|
+
Requires-Dist: aiofiles>=24.1.0
|
|
8
|
+
Requires-Dist: fastapi[standard]>=0.115.14
|
|
9
|
+
Requires-Dist: httpx[http2]>=0.28.1
|
|
10
|
+
Requires-Dist: packaging>=25.0
|
|
11
|
+
Requires-Dist: pydantic-settings>=2.4.0
|
|
12
|
+
Requires-Dist: pydantic>=2.8.0
|
|
13
|
+
Requires-Dist: pyjwt>=2.10.1
|
|
14
|
+
Requires-Dist: rich-toolkit>=0.14.8
|
|
15
|
+
Requires-Dist: rich>=13.0.0
|
|
16
|
+
Requires-Dist: sortedcontainers>=2.4.0
|
|
17
|
+
Requires-Dist: structlog>=25.4.0
|
|
18
|
+
Requires-Dist: typer>=0.16.0
|
|
19
|
+
Requires-Dist: typing-extensions>=4.0.0
|
|
20
|
+
Requires-Dist: uvicorn>=0.34.0
|
|
21
|
+
Provides-Extra: plugins-claude
|
|
22
|
+
Requires-Dist: claude-agent-sdk>=0.1.0; extra == 'plugins-claude'
|
|
23
|
+
Requires-Dist: qrcode>=8.2; extra == 'plugins-claude'
|
|
24
|
+
Provides-Extra: plugins-codex
|
|
25
|
+
Requires-Dist: pyjwt>=2.10.1; extra == 'plugins-codex'
|
|
26
|
+
Requires-Dist: qrcode>=8.2; extra == 'plugins-codex'
|
|
27
|
+
Provides-Extra: plugins-mcp
|
|
28
|
+
Requires-Dist: fastapi-mcp>=0.3.7; extra == 'plugins-mcp'
|
|
29
|
+
Provides-Extra: plugins-metrics
|
|
30
|
+
Requires-Dist: prometheus-client>=0.22.1; extra == 'plugins-metrics'
|
|
31
|
+
Provides-Extra: plugins-storage
|
|
32
|
+
Requires-Dist: duckdb-engine>=0.17.0; extra == 'plugins-storage'
|
|
33
|
+
Requires-Dist: duckdb<1.4.0,>=1.1.0; extra == 'plugins-storage'
|
|
34
|
+
Requires-Dist: sqlalchemy>=2.0.0; extra == 'plugins-storage'
|
|
35
|
+
Requires-Dist: sqlmodel>=0.0.24; extra == 'plugins-storage'
|
|
36
|
+
Provides-Extra: plugins-tui
|
|
37
|
+
Requires-Dist: aioconsole>=0.8.1; extra == 'plugins-tui'
|
|
38
|
+
Requires-Dist: textual>=3.7.1; extra == 'plugins-tui'
|
|
39
|
+
Description-Content-Type: text/markdown
|
|
40
|
+
|
|
41
|
+
# CCProxy API Server
|
|
42
|
+
|
|
43
|
+
CCProxy is a local, plugin-based reverse proxy that unifies access to
|
|
44
|
+
multiple AI providers (e.g., Claude SDK/API and OpenAI Codex) behind a
|
|
45
|
+
consistent API. It ships with bundled plugins for providers, logging,
|
|
46
|
+
tracing, metrics, analytics, and more.
|
|
47
|
+
|
|
48
|
+
## Supported Providers
|
|
49
|
+
|
|
50
|
+
- Anthropic Claude API/SDK (OAuth2 flow or Claude CLI/SDK token files)
|
|
51
|
+
- OpenAI Codex (ChatGPT backend Responses API using OAuth for paid/pro accounts)
|
|
52
|
+
- GitHub Copilot (chat and completions for free, paid, or business accounts)
|
|
53
|
+
|
|
54
|
+
Each provider adapter exposes the same surface area: OpenAI Chat
|
|
55
|
+
Completions, OpenAI Responses, and Anthropic Messages. The proxy maintains a
|
|
56
|
+
shared model-mapping layer so you can reuse the same `model` identifier
|
|
57
|
+
across providers without rewriting client code.
|
|
58
|
+
|
|
59
|
+
Authentication can reuse existing provider files (e.g., Claude CLI SDK
|
|
60
|
+
tokens and the Codex CLI credential store), or you can run
|
|
61
|
+
`ccproxy auth login <provider>` to complete the OAuth flow from the CLI;
|
|
62
|
+
stored secrets are picked up automatically by the proxy.
|
|
63
|
+
|
|
64
|
+
## Extensibility
|
|
65
|
+
|
|
66
|
+
CCProxy's plugin system lets you add instrumentation and storage layers
|
|
67
|
+
without patching the core server. Bundled plugins currently include:
|
|
68
|
+
|
|
69
|
+
- [`access_log`](ccproxy/plugins/access_log/README.md): structured access
|
|
70
|
+
logging for client and provider traffic
|
|
71
|
+
- [`analytics`](ccproxy/plugins/analytics/README.md): DuckDB-backed analytics
|
|
72
|
+
APIs for captured request logs
|
|
73
|
+
- [`claude_api`](ccproxy/plugins/claude_api/README.md): Anthropic Claude HTTP
|
|
74
|
+
API adapter with health and metrics
|
|
75
|
+
- [`claude_sdk`](ccproxy/plugins/claude_sdk/README.md): local Claude CLI/SDK
|
|
76
|
+
adapter with session pooling
|
|
77
|
+
- [`codex`](ccproxy/plugins/codex/README.md): OpenAI Codex provider adapter
|
|
78
|
+
with OAuth support
|
|
79
|
+
- [`command_replay`](ccproxy/plugins/command_replay/README.md): generates
|
|
80
|
+
`curl`/`xh` commands for captured requests
|
|
81
|
+
- [`copilot`](ccproxy/plugins/copilot/README.md): GitHub Copilot provider
|
|
82
|
+
adapter with OAuth token management
|
|
83
|
+
- [`credential_balancer`](ccproxy/plugins/credential_balancer/README.md):
|
|
84
|
+
rotates upstream credentials based on health
|
|
85
|
+
- [`dashboard`](ccproxy/plugins/dashboard/README.md): serves the CCProxy
|
|
86
|
+
dashboard SPA and APIs
|
|
87
|
+
- [`docker`](ccproxy/plugins/docker/README.md): runs providers inside Docker via
|
|
88
|
+
CLI extensions
|
|
89
|
+
- [`duckdb_storage`](ccproxy/plugins/duckdb_storage/README.md): exposes
|
|
90
|
+
DuckDB-backed storage for logs and analytics
|
|
91
|
+
- [`max_tokens`](ccproxy/plugins/max_tokens/README.md): normalizes
|
|
92
|
+
`max_tokens` fields to provider limits
|
|
93
|
+
- [`metrics`](ccproxy/plugins/metrics/README.md): Prometheus-compatible metrics
|
|
94
|
+
with optional Pushgateway
|
|
95
|
+
- [`oauth_claude`](ccproxy/plugins/oauth_claude/README.md): standalone OAuth
|
|
96
|
+
provider for Claude integrations
|
|
97
|
+
- [`oauth_codex`](ccproxy/plugins/oauth_codex/README.md): standalone OAuth
|
|
98
|
+
provider for Codex integrations
|
|
99
|
+
- [`permissions`](ccproxy/plugins/permissions/README.md): interactive approval
|
|
100
|
+
flow for privileged tool actions
|
|
101
|
+
- [`pricing`](ccproxy/plugins/pricing/README.md): caches model pricing data for
|
|
102
|
+
cost-aware features
|
|
103
|
+
- [`request_tracer`](ccproxy/plugins/request_tracer/README.md): detailed
|
|
104
|
+
request/response tracing for debugging
|
|
105
|
+
|
|
106
|
+
Shared helpers such as
|
|
107
|
+
[`claude_shared`](ccproxy/plugins/claude_shared/README.md) provide metadata
|
|
108
|
+
consumed by the Claude plugins. Each plugin directory contains its own README
|
|
109
|
+
with configuration examples.
|
|
110
|
+
|
|
111
|
+
## Quick Links
|
|
112
|
+
|
|
113
|
+
- Docs site entry: `docs/index.md`
|
|
114
|
+
- Getting started: `docs/getting-started/quickstart.md`
|
|
115
|
+
- Configuration reference: `docs/getting-started/configuration.md`
|
|
116
|
+
- Examples: `docs/examples.md`
|
|
117
|
+
- Migration (0.2): `docs/migration/0.2-plugin-first.md`
|
|
118
|
+
|
|
119
|
+
## Plugin Config Quickstart
|
|
120
|
+
|
|
121
|
+
The plugin system is enabled by default (`enable_plugins = true`), and all
|
|
122
|
+
discovered plugins load automatically when no additional filters are set. Use
|
|
123
|
+
these knobs to adjust what runs:
|
|
124
|
+
|
|
125
|
+
- `enabled_plugins`: optional allow list; when set, only the listed plugins run.
|
|
126
|
+
- `disabled_plugins`: optional block list applied when `enabled_plugins` is not
|
|
127
|
+
set.
|
|
128
|
+
- `plugins.<name>.enabled`: per-plugin flag (defaults to `true`) that you can
|
|
129
|
+
override in TOML or environment variables. Any plugin set to `false` is added
|
|
130
|
+
to the deny list alongside `disabled_plugins` during startup.
|
|
131
|
+
|
|
132
|
+
During startup we merge `disabled_plugins` and any `plugins.<name>.enabled = false`
|
|
133
|
+
entries into a single deny list. At runtime the loader checks the allow list
|
|
134
|
+
first and then confirms the plugin is not deny listed. Configure plugins under
|
|
135
|
+
`plugins.<name>` in TOML or via nested environment variables.
|
|
136
|
+
|
|
137
|
+
Use `ccproxy plugins list` to inspect discovered plugins and
|
|
138
|
+
`ccproxy plugins settings <name>` to review configuration fields.
|
|
139
|
+
|
|
140
|
+
### TOML example (`.ccproxy.toml`)
|
|
141
|
+
|
|
142
|
+
```toml
|
|
143
|
+
enable_plugins = true
|
|
144
|
+
# enabled_plugins = ["metrics", "analytics"] # Optional allow list
|
|
145
|
+
disabled_plugins = ["duckdb_storage"] # Optional block list
|
|
146
|
+
|
|
147
|
+
[plugins.access_log]
|
|
148
|
+
client_enabled = true
|
|
149
|
+
client_format = "structured"
|
|
150
|
+
client_log_file = "/tmp/ccproxy/access.log"
|
|
151
|
+
|
|
152
|
+
[plugins.request_tracer]
|
|
153
|
+
json_logs_enabled = true
|
|
154
|
+
raw_http_enabled = true
|
|
155
|
+
log_dir = "/tmp/ccproxy/traces"
|
|
156
|
+
|
|
157
|
+
[plugins.duckdb_storage]
|
|
158
|
+
enabled = false
|
|
159
|
+
|
|
160
|
+
[plugins.analytics]
|
|
161
|
+
enabled = true
|
|
162
|
+
|
|
163
|
+
# Metrics plugin
|
|
164
|
+
[plugins.metrics]
|
|
165
|
+
enabled = true
|
|
166
|
+
# pushgateway_enabled = true
|
|
167
|
+
# pushgateway_url = "http://localhost:9091"
|
|
168
|
+
# pushgateway_job = "ccproxy"
|
|
169
|
+
# pushgateway_push_interval = 60
|
|
170
|
+
```
|
|
171
|
+
|
|
172
|
+
### Environment variables (nested with `__`)
|
|
173
|
+
|
|
174
|
+
```bash
|
|
175
|
+
export DISABLED_PLUGINS="duckdb_storage" # Optional block list
|
|
176
|
+
export PLUGINS__ACCESS_LOG__ENABLED=true
|
|
177
|
+
export PLUGINS__ACCESS_LOG__CLIENT_ENABLED=true
|
|
178
|
+
export PLUGINS__ACCESS_LOG__CLIENT_FORMAT=structured
|
|
179
|
+
export PLUGINS__ACCESS_LOG__CLIENT_LOG_FILE=/tmp/ccproxy/access.log
|
|
180
|
+
|
|
181
|
+
export PLUGINS__REQUEST_TRACER__ENABLED=true
|
|
182
|
+
export PLUGINS__REQUEST_TRACER__JSON_LOGS_ENABLED=true
|
|
183
|
+
export PLUGINS__REQUEST_TRACER__RAW_HTTP_ENABLED=true
|
|
184
|
+
export PLUGINS__REQUEST_TRACER__LOG_DIR=/tmp/ccproxy/traces
|
|
185
|
+
|
|
186
|
+
export PLUGINS__DUCKDB_STORAGE__ENABLED=true
|
|
187
|
+
export PLUGINS__ANALYTICS__ENABLED=true
|
|
188
|
+
export PLUGINS__METRICS__ENABLED=true
|
|
189
|
+
# export PLUGINS__METRICS__PUSHGATEWAY_ENABLED=true
|
|
190
|
+
# export PLUGINS__METRICS__PUSHGATEWAY_URL=http://localhost:9091
|
|
191
|
+
```
|
|
192
|
+
|
|
193
|
+
## Running
|
|
194
|
+
|
|
195
|
+
To install the latest stable release without cloning the repository, use `uvx`
|
|
196
|
+
to grab the published wheel and launch the CLI:
|
|
197
|
+
|
|
198
|
+
```bash
|
|
199
|
+
uvx --with "ccproxy-api[all]" ccproxy serve --port 8000
|
|
200
|
+
```
|
|
201
|
+
|
|
202
|
+
If you prefer `pipx`, install the package (optionally with extras) and use the
|
|
203
|
+
local shim:
|
|
204
|
+
|
|
205
|
+
```bash
|
|
206
|
+
pipx install "ccproxy-api[all]"
|
|
207
|
+
ccproxy serve # default on localhost:8000
|
|
208
|
+
```
|
|
209
|
+
|
|
210
|
+
## License
|
|
211
|
+
|
|
212
|
+
See `LICENSE`.
|