From fccfb162b402aa22638f339f68b4f08527040d30 Mon Sep 17 00:00:00 2001 From: daniel <15257433+kslamph@users.noreply.github.com> Date: Tue, 24 Mar 2026 15:23:09 +0800 Subject: [PATCH 01/51] fix(gemini-cli): use backend project ID from onboarding response - Simplify project ID selection to always use the backend project ID returned by Gemini onboarding - Update Gemini CLI version from 0.31.0 to 0.34.0 - Add 'terminal' to User-Agent string for better client identification Co-Authored-By: Claude Sonnet 4.6 --- .gitignore | 1 + .../api/handlers/management/auth_files.go | 17 ++------- internal/cmd/login.go | 36 ++----------------- internal/misc/header_utils.go | 4 +-- 4 files changed, 9 insertions(+), 49 deletions(-) diff --git a/.gitignore b/.gitignore index 90ff3a94..80f4b2eb 100644 --- a/.gitignore +++ b/.gitignore @@ -49,3 +49,4 @@ _bmad-output/* # macOS .DS_Store ._* +.gocache/ diff --git a/internal/api/handlers/management/auth_files.go b/internal/api/handlers/management/auth_files.go index 2e1f02bf..a7916e79 100644 --- a/internal/api/handlers/management/auth_files.go +++ b/internal/api/handlers/management/auth_files.go @@ -2566,20 +2566,9 @@ func performGeminiCLISetup(ctx context.Context, httpClient *http.Client, storage finalProjectID := projectID if responseProjectID != "" { if explicitProject && !strings.EqualFold(responseProjectID, projectID) { - // Check if this is a free user (gen-lang-client projects or free/legacy tier) - isFreeUser := strings.HasPrefix(projectID, "gen-lang-client-") || - strings.EqualFold(tierID, "FREE") || - strings.EqualFold(tierID, "LEGACY") - - if isFreeUser { - // For free users, use backend project ID for preview model access - log.Infof("Gemini onboarding: frontend project %s maps to backend project %s", projectID, responseProjectID) - log.Infof("Using backend project ID: %s (recommended for preview model access)", responseProjectID) - finalProjectID = responseProjectID - } else { - // Pro users: keep requested project ID (original behavior) - log.Warnf("Gemini onboarding returned project %s instead of requested %s; keeping requested project ID.", responseProjectID, projectID) - } + log.Infof("Gemini onboarding: requested project %s maps to backend project %s", projectID, responseProjectID) + log.Infof("Using backend project ID: %s", responseProjectID) + finalProjectID = responseProjectID } else { finalProjectID = responseProjectID } diff --git a/internal/cmd/login.go b/internal/cmd/login.go index 16af718e..298e9546 100644 --- a/internal/cmd/login.go +++ b/internal/cmd/login.go @@ -333,39 +333,9 @@ func performGeminiCLISetup(ctx context.Context, httpClient *http.Client, storage finalProjectID := projectID if responseProjectID != "" { if explicitProject && !strings.EqualFold(responseProjectID, projectID) { - // Check if this is a free user (gen-lang-client projects or free/legacy tier) - isFreeUser := strings.HasPrefix(projectID, "gen-lang-client-") || - strings.EqualFold(tierID, "FREE") || - strings.EqualFold(tierID, "LEGACY") - - if isFreeUser { - // Interactive prompt for free users - fmt.Printf("\nGoogle returned a different project ID:\n") - fmt.Printf(" Requested (frontend): %s\n", projectID) - fmt.Printf(" Returned (backend): %s\n\n", responseProjectID) - fmt.Printf(" Backend project IDs have access to preview models (gemini-3-*).\n") - fmt.Printf(" This is normal for free tier users.\n\n") - fmt.Printf("Which project ID would you like to use?\n") - fmt.Printf(" [1] Backend (recommended): %s\n", responseProjectID) - fmt.Printf(" [2] Frontend: %s\n\n", projectID) - fmt.Printf("Enter choice [1]: ") - - reader := bufio.NewReader(os.Stdin) - choice, _ := reader.ReadString('\n') - choice = strings.TrimSpace(choice) - - if choice == "2" { - log.Infof("Using frontend project ID: %s", projectID) - fmt.Println(". Warning: Frontend project IDs may not have access to preview models.") - finalProjectID = projectID - } else { - log.Infof("Using backend project ID: %s (recommended)", responseProjectID) - finalProjectID = responseProjectID - } - } else { - // Pro users: keep requested project ID (original behavior) - log.Warnf("Gemini onboarding returned project %s instead of requested %s; keeping requested project ID.", responseProjectID, projectID) - } + log.Infof("Gemini onboarding: requested project %s maps to backend project %s", projectID, responseProjectID) + log.Infof("Using backend project ID: %s", responseProjectID) + finalProjectID = responseProjectID } else { finalProjectID = responseProjectID } diff --git a/internal/misc/header_utils.go b/internal/misc/header_utils.go index 5752a269..ac022a96 100644 --- a/internal/misc/header_utils.go +++ b/internal/misc/header_utils.go @@ -12,7 +12,7 @@ import ( const ( // GeminiCLIVersion is the version string reported in the User-Agent for upstream requests. - GeminiCLIVersion = "0.31.0" + GeminiCLIVersion = "0.34.0" // GeminiCLIApiClientHeader is the value for the X-Goog-Api-Client header sent to the Gemini CLI upstream. GeminiCLIApiClientHeader = "google-genai-sdk/1.41.0 gl-node/v22.19.0" @@ -46,7 +46,7 @@ func GeminiCLIUserAgent(model string) string { if model == "" { model = "unknown" } - return fmt.Sprintf("GeminiCLI/%s/%s (%s; %s)", GeminiCLIVersion, model, geminiCLIOS(), geminiCLIArch()) + return fmt.Sprintf("GeminiCLI/%s/%s (%s; %s; terminal)", GeminiCLIVersion, model, geminiCLIOS(), geminiCLIArch()) } // ScrubProxyAndFingerprintHeaders removes all headers that could reveal From 91387ca2472aac07440b542b80fb1f070f63a624 Mon Sep 17 00:00:00 2001 From: daniel <15257433+kslamph@users.noreply.github.com> Date: Mon, 30 Mar 2026 20:07:02 +0800 Subject: [PATCH 02/51] refactor(gemini-cli): simplify redundant if/else in project ID assignment Both branches assign finalProjectID = responseProjectID, so move the assignment outside the conditional and keep only the logging inside. --- internal/api/handlers/management/auth_files.go | 4 +--- internal/cmd/login.go | 4 +--- 2 files changed, 2 insertions(+), 6 deletions(-) diff --git a/internal/api/handlers/management/auth_files.go b/internal/api/handlers/management/auth_files.go index a7916e79..63b1d62d 100644 --- a/internal/api/handlers/management/auth_files.go +++ b/internal/api/handlers/management/auth_files.go @@ -2568,10 +2568,8 @@ func performGeminiCLISetup(ctx context.Context, httpClient *http.Client, storage if explicitProject && !strings.EqualFold(responseProjectID, projectID) { log.Infof("Gemini onboarding: requested project %s maps to backend project %s", projectID, responseProjectID) log.Infof("Using backend project ID: %s", responseProjectID) - finalProjectID = responseProjectID - } else { - finalProjectID = responseProjectID } + finalProjectID = responseProjectID } storage.ProjectID = strings.TrimSpace(finalProjectID) diff --git a/internal/cmd/login.go b/internal/cmd/login.go index 298e9546..22404dac 100644 --- a/internal/cmd/login.go +++ b/internal/cmd/login.go @@ -335,10 +335,8 @@ func performGeminiCLISetup(ctx context.Context, httpClient *http.Client, storage if explicitProject && !strings.EqualFold(responseProjectID, projectID) { log.Infof("Gemini onboarding: requested project %s maps to backend project %s", projectID, responseProjectID) log.Infof("Using backend project ID: %s", responseProjectID) - finalProjectID = responseProjectID - } else { - finalProjectID = responseProjectID } + finalProjectID = responseProjectID } storage.ProjectID = strings.TrimSpace(finalProjectID) From 6431cec7d3c12d14a5eac272a8555d82753b4dad Mon Sep 17 00:00:00 2001 From: Code_G <12405078+codeg-dev@users.noreply.github.com> Date: Mon, 6 Apr 2026 17:16:15 +0900 Subject: [PATCH 03/51] fix(claude-auth): dedupe OAuth refresh and honor 429 backoff Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus --- internal/auth/claude/anthropic_auth.go | 135 +++++++++++++++++++- internal/auth/claude/anthropic_auth_test.go | 123 ++++++++++++++++++ 2 files changed, 257 insertions(+), 1 deletion(-) create mode 100644 internal/auth/claude/anthropic_auth_test.go diff --git a/internal/auth/claude/anthropic_auth.go b/internal/auth/claude/anthropic_auth.go index 12bb53ac..b7f997ef 100644 --- a/internal/auth/claude/anthropic_auth.go +++ b/internal/auth/claude/anthropic_auth.go @@ -6,15 +6,18 @@ package claude import ( "context" "encoding/json" + "errors" "fmt" "io" "net/http" "net/url" "strings" + "sync" "time" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" log "github.com/sirupsen/logrus" + "golang.org/x/sync/singleflight" ) // OAuth configuration constants for Claude/Anthropic @@ -23,8 +26,94 @@ const ( TokenURL = "https://api.anthropic.com/v1/oauth/token" ClientID = "9d1c250a-e61b-44d9-88ed-5944d1962f5e" RedirectURI = "http://localhost:54545/callback" + + claudeRefreshMinBackoff = 5 * time.Second + claudeRefreshMaxBackoff = 5 * time.Minute ) +var ( + claudeRefreshGroup singleflight.Group + claudeRefreshMu sync.Mutex + claudeRefreshBlock = make(map[string]time.Time) +) + +type refreshHTTPError struct { + status int + message string + retryable bool +} + +func (e *refreshHTTPError) Error() string { + return fmt.Sprintf("token refresh failed with status %d: %s", e.status, e.message) +} + +func (e *refreshHTTPError) Retryable() bool { + return e != nil && e.retryable +} + +func resetClaudeRefreshState() { + claudeRefreshMu.Lock() + defer claudeRefreshMu.Unlock() + claudeRefreshBlock = make(map[string]time.Time) + claudeRefreshGroup = singleflight.Group{} +} + +func claudeRefreshBlockedUntil(refreshToken string) time.Time { + claudeRefreshMu.Lock() + defer claudeRefreshMu.Unlock() + return claudeRefreshBlock[refreshToken] +} + +func setClaudeRefreshBlockedUntil(refreshToken string, until time.Time) { + claudeRefreshMu.Lock() + defer claudeRefreshMu.Unlock() + claudeRefreshBlock[refreshToken] = until +} + +func clearClaudeRefreshBlockedUntil(refreshToken string) { + claudeRefreshMu.Lock() + defer claudeRefreshMu.Unlock() + delete(claudeRefreshBlock, refreshToken) +} + +func clampClaudeRefreshBackoff(d time.Duration) time.Duration { + if d < claudeRefreshMinBackoff { + return claudeRefreshMinBackoff + } + if d > claudeRefreshMaxBackoff { + return claudeRefreshMaxBackoff + } + return d +} + +func parseClaudeRetryAfter(resp *http.Response) time.Duration { + if resp == nil { + return claudeRefreshMinBackoff + } + if raw := strings.TrimSpace(resp.Header.Get("Retry-After")); raw != "" { + if seconds, err := time.ParseDuration(raw + "s"); err == nil { + return clampClaudeRefreshBackoff(seconds) + } + if when, err := http.ParseTime(raw); err == nil { + return clampClaudeRefreshBackoff(time.Until(when)) + } + } + if raw := strings.TrimSpace(resp.Header.Get("Retry-After-Ms")); raw != "" { + if ms, err := time.ParseDuration(raw + "ms"); err == nil { + return clampClaudeRefreshBackoff(ms) + } + } + return claudeRefreshMinBackoff +} + +func isClaudeRefreshRetryable(err error) bool { + var httpErr *refreshHTTPError + if errors.As(err, &httpErr) { + return httpErr.Retryable() + } + return true +} + // tokenResponse represents the response structure from Anthropic's OAuth token endpoint. // It contains access token, refresh token, and associated user/organization information. type tokenResponse struct { @@ -222,6 +311,35 @@ func (o *ClaudeAuth) RefreshTokens(ctx context.Context, refreshToken string) (*C if refreshToken == "" { return nil, fmt.Errorf("refresh token is required") } + if blockedUntil := claudeRefreshBlockedUntil(refreshToken); blockedUntil.After(time.Now()) { + return nil, &refreshHTTPError{ + status: http.StatusTooManyRequests, + message: fmt.Sprintf("refresh temporarily blocked until %s", blockedUntil.Format(time.RFC3339)), + retryable: false, + } + } + + result, err, _ := claudeRefreshGroup.Do(refreshToken, func() (interface{}, error) { + return o.refreshTokensSingleFlight(context.WithoutCancel(ctx), refreshToken) + }) + if err != nil { + return nil, err + } + tokenData, ok := result.(*ClaudeTokenData) + if !ok || tokenData == nil { + return nil, fmt.Errorf("token refresh failed: invalid single-flight result") + } + return tokenData, nil +} + +func (o *ClaudeAuth) refreshTokensSingleFlight(ctx context.Context, refreshToken string) (*ClaudeTokenData, error) { + if blockedUntil := claudeRefreshBlockedUntil(refreshToken); blockedUntil.After(time.Now()) { + return nil, &refreshHTTPError{ + status: http.StatusTooManyRequests, + message: fmt.Sprintf("refresh temporarily blocked until %s", blockedUntil.Format(time.RFC3339)), + retryable: false, + } + } reqBody := map[string]interface{}{ "client_id": ClientID, @@ -256,7 +374,17 @@ func (o *ClaudeAuth) RefreshTokens(ctx context.Context, refreshToken string) (*C } if resp.StatusCode != http.StatusOK { - return nil, fmt.Errorf("token refresh failed with status %d: %s", resp.StatusCode, string(body)) + message := string(body) + if resp.StatusCode == http.StatusTooManyRequests { + retryAfter := parseClaudeRetryAfter(resp) + setClaudeRefreshBlockedUntil(refreshToken, time.Now().Add(retryAfter)) + return nil, &refreshHTTPError{status: resp.StatusCode, message: message, retryable: false} + } + return nil, &refreshHTTPError{ + status: resp.StatusCode, + message: message, + retryable: resp.StatusCode >= http.StatusInternalServerError, + } } // log.Debugf("Token response: %s", string(body)) @@ -267,6 +395,8 @@ func (o *ClaudeAuth) RefreshTokens(ctx context.Context, refreshToken string) (*C } // Create token data + clearClaudeRefreshBlockedUntil(refreshToken) + return &ClaudeTokenData{ AccessToken: tokenResp.AccessToken, RefreshToken: tokenResp.RefreshToken, @@ -328,6 +458,9 @@ func (o *ClaudeAuth) RefreshTokensWithRetry(ctx context.Context, refreshToken st lastErr = err log.Warnf("Token refresh attempt %d failed: %v", attempt+1, err) + if !isClaudeRefreshRetryable(err) { + break + } } return nil, fmt.Errorf("token refresh failed after %d attempts: %w", maxRetries, lastErr) diff --git a/internal/auth/claude/anthropic_auth_test.go b/internal/auth/claude/anthropic_auth_test.go new file mode 100644 index 00000000..0b14d083 --- /dev/null +++ b/internal/auth/claude/anthropic_auth_test.go @@ -0,0 +1,123 @@ +package claude + +import ( + "context" + "io" + "net/http" + "strings" + "sync" + "sync/atomic" + "testing" + "time" +) + +type roundTripFunc func(*http.Request) (*http.Response, error) + +func (f roundTripFunc) RoundTrip(req *http.Request) (*http.Response, error) { + return f(req) +} + +func TestRefreshTokensWithRetry_429BlocksImmediateReplay(t *testing.T) { + resetClaudeRefreshState() + defer resetClaudeRefreshState() + + var calls int32 + auth := &ClaudeAuth{ + httpClient: &http.Client{ + Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) { + atomic.AddInt32(&calls, 1) + return &http.Response{ + StatusCode: http.StatusTooManyRequests, + Body: io.NopCloser(strings.NewReader(`{"error":"rate_limited"}`)), + Header: http.Header{"Retry-After": []string{"60"}}, + Request: req, + }, nil + }), + }, + } + + _, err := auth.RefreshTokensWithRetry(context.Background(), "dummy_refresh_token", 3) + if err == nil { + t.Fatalf("expected 429 refresh error") + } + if !strings.Contains(err.Error(), "status 429") { + t.Fatalf("expected status 429 in error, got %v", err) + } + if got := atomic.LoadInt32(&calls); got != 1 { + t.Fatalf("expected 1 refresh attempt after 429, got %d", got) + } + + _, err = auth.RefreshTokensWithRetry(context.Background(), "dummy_refresh_token", 3) + if err == nil { + t.Fatalf("expected immediate blocked refresh error") + } + if got := atomic.LoadInt32(&calls); got != 1 { + t.Fatalf("expected blocked retry to avoid a second refresh call, got %d attempts", got) + } + if blockedUntil := claudeRefreshBlockedUntil("dummy_refresh_token"); !blockedUntil.After(time.Now()) { + t.Fatalf("expected blocked-until timestamp to be set, got %v", blockedUntil) + } +} + +func TestRefreshTokens_DeduplicatesConcurrentRefresh(t *testing.T) { + resetClaudeRefreshState() + defer resetClaudeRefreshState() + + var calls int32 + started := make(chan struct{}) + release := make(chan struct{}) + var once sync.Once + + auth := &ClaudeAuth{ + httpClient: &http.Client{ + Transport: roundTripFunc(func(req *http.Request) (*http.Response, error) { + atomic.AddInt32(&calls, 1) + once.Do(func() { close(started) }) + <-release + return &http.Response{ + StatusCode: http.StatusOK, + Body: io.NopCloser(strings.NewReader(`{ + "access_token":"new-access", + "refresh_token":"new-refresh", + "token_type":"Bearer", + "expires_in":3600, + "account":{"email_address":"shared@example.com"} + }`)), + Header: make(http.Header), + Request: req, + }, nil + }), + }, + } + + results := make(chan *ClaudeTokenData, 2) + errs := make(chan error, 2) + runRefresh := func() { + td, err := auth.RefreshTokens(context.Background(), "shared-refresh-token") + results <- td + errs <- err + } + + go runRefresh() + go runRefresh() + + <-started + time.Sleep(20 * time.Millisecond) + if got := atomic.LoadInt32(&calls); got != 1 { + t.Fatalf("expected concurrent refresh to share a single upstream call, got %d", got) + } + close(release) + + for i := 0; i < 2; i++ { + if err := <-errs; err != nil { + t.Fatalf("expected refresh to succeed, got %v", err) + } + td := <-results + if td == nil || td.AccessToken != "new-access" { + t.Fatalf("expected refreshed access token, got %#v", td) + } + } + if got := atomic.LoadInt32(&calls); got != 1 { + t.Fatalf("expected exactly 1 upstream refresh call, got %d", got) + } +} From 29e32aaab940f0681b9f9a9b2da94b81d2e5d098 Mon Sep 17 00:00:00 2001 From: Code_G <12405078+codeg-dev@users.noreply.github.com> Date: Mon, 6 Apr 2026 17:16:42 +0900 Subject: [PATCH 04/51] fix(executor): route Claude refresh through retry-aware auth Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus --- internal/runtime/executor/claude_executor.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 7b2e5d8d..93487311 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -594,7 +594,7 @@ func (e *ClaudeExecutor) Refresh(ctx context.Context, auth *cliproxyauth.Auth) ( return auth, nil } svc := claudeauth.NewClaudeAuth(e.cfg) - td, err := svc.RefreshTokens(ctx, refreshToken) + td, err := svc.RefreshTokensWithRetry(ctx, refreshToken, 3) if err != nil { return nil, err } From a0fe273081eaa3a4b89ec21fd718a4585b84fda2 Mon Sep 17 00:00:00 2001 From: DragonFSKY Date: Sun, 22 Mar 2026 09:49:34 +0800 Subject: [PATCH 05/51] fix(websocket): skip stale state merge after client-side compact After a Codex CLI compact, the client sends a full conversation transcript (with compaction items or assistant messages) as input. Previously, normalizeResponseSubsequentRequest() unconditionally merged this with stale lastRequest/lastResponseOutput, breaking function_call/function_call_output pairings and causing 400 errors ("No tool output found for function call"). Add inputContainsFullTranscript() heuristic that detects compaction items (type=compaction/compaction_summary) or assistant messages in the input array, and bypasses the merge when a full transcript is present. Fixes #2207 --- .../openai/openai_responses_websocket.go | 66 +++++++++--- .../openai/openai_responses_websocket_test.go | 101 ++++++++++++++++++ 2 files changed, 155 insertions(+), 12 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_websocket.go b/sdk/api/handlers/openai/openai_responses_websocket.go index 2f6b14a7..6457cd3e 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket.go +++ b/sdk/api/handlers/openai/openai_responses_websocket.go @@ -315,20 +315,32 @@ func normalizeResponseSubsequentRequest(rawJSON []byte, lastRequest []byte, last } } - existingInput := gjson.GetBytes(lastRequest, "input") - mergedInput, errMerge := mergeJSONArrayRaw(existingInput.Raw, normalizeJSONArrayRaw(lastResponseOutput)) - if errMerge != nil { - return nil, lastRequest, &interfaces.ErrorMessage{ - StatusCode: http.StatusBadRequest, - Error: fmt.Errorf("invalid previous response output: %w", errMerge), + // When the client sends a full conversation transcript (e.g. after compact), + // the input already contains the complete history including assistant messages. + // In that case, skip merging with stale lastRequest/lastResponseOutput to avoid + // breaking function_call / function_call_output pairings. + // See: https://github.com/router-for-me/CLIProxyAPI/issues/2207 + var mergedInput string + if inputContainsFullTranscript(nextInput) { + log.Infof("responses websocket: full transcript detected, skipping stale merge (input items=%d)", len(nextInput.Array())) + mergedInput = nextInput.Raw + } else { + existingInput := gjson.GetBytes(lastRequest, "input") + var errMerge error + mergedInput, errMerge = mergeJSONArrayRaw(existingInput.Raw, normalizeJSONArrayRaw(lastResponseOutput)) + if errMerge != nil { + return nil, lastRequest, &interfaces.ErrorMessage{ + StatusCode: http.StatusBadRequest, + Error: fmt.Errorf("invalid previous response output: %w", errMerge), + } } - } - mergedInput, errMerge = mergeJSONArrayRaw(mergedInput, nextInput.Raw) - if errMerge != nil { - return nil, lastRequest, &interfaces.ErrorMessage{ - StatusCode: http.StatusBadRequest, - Error: fmt.Errorf("invalid request input: %w", errMerge), + mergedInput, errMerge = mergeJSONArrayRaw(mergedInput, nextInput.Raw) + if errMerge != nil { + return nil, lastRequest, &interfaces.ErrorMessage{ + StatusCode: http.StatusBadRequest, + Error: fmt.Errorf("invalid request input: %w", errMerge), + } } } dedupedInput, errDedupeFunctionCalls := dedupeFunctionCallsByCallID(mergedInput) @@ -691,6 +703,36 @@ func mergeJSONArrayRaw(existingRaw, appendRaw string) (string, error) { return string(out), nil } +// inputContainsFullTranscript returns true when the input array looks like a +// complete conversation history rather than an incremental append. After a +// client-side compact the input already carries the full (compacted) transcript +// which may include assistant messages or compaction items. Merging that with +// the stale lastRequest / lastResponseOutput would duplicate or break +// function_call / function_call_output pairings, so the caller should use the +// input as-is. +// +// Heuristic: the array is a full transcript when it contains either +// - a message with role="assistant", or +// - a compaction item (type="compaction" or "compaction_summary"). +// +// Normal incremental turns only contain user messages or function_call_output +// items and never carry either of these signals. +func inputContainsFullTranscript(input gjson.Result) bool { + if !input.IsArray() { + return false + } + for _, item := range input.Array() { + t := item.Get("type").String() + if t == "message" && item.Get("role").String() == "assistant" { + return true + } + if t == "compaction" || t == "compaction_summary" { + return true + } + } + return false +} + func normalizeJSONArrayRaw(raw []byte) string { trimmed := strings.TrimSpace(string(raw)) if trimmed == "" { diff --git a/sdk/api/handlers/openai/openai_responses_websocket_test.go b/sdk/api/handlers/openai/openai_responses_websocket_test.go index ecfc90b3..2c5ef579 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket_test.go +++ b/sdk/api/handlers/openai/openai_responses_websocket_test.go @@ -1400,3 +1400,104 @@ func TestResponsesWebsocketCompactionResetsTurnStateOnTranscriptReplacement(t *t t.Fatalf("post-compact function call id = %s, want call-1", items[0].Get("call_id").String()) } } + +func TestInputContainsFullTranscriptDetectsAssistantMessage(t *testing.T) { + input := gjson.Parse(`[ + {"type":"message","role":"user","content":"hello"}, + {"type":"message","role":"assistant","content":"hi there"} + ]`) + if !inputContainsFullTranscript(input) { + t.Fatal("expected full transcript when assistant message is present") + } +} + +func TestInputContainsFullTranscriptDetectsCompactionItem(t *testing.T) { + for _, typ := range []string{"compaction", "compaction_summary"} { + input := gjson.Parse(`[{"type":"message","role":"user","content":"hello"},{"type":"` + typ + `","encrypted_content":"summary"}]`) + if !inputContainsFullTranscript(input) { + t.Fatalf("expected full transcript for type=%s", typ) + } + } +} + +func TestInputContainsFullTranscriptFalseForIncremental(t *testing.T) { + // Normal incremental turns: user messages or function_call_output only. + for _, raw := range []string{ + `[{"type":"function_call_output","call_id":"call-1","output":"result"}]`, + `[{"type":"message","role":"user","content":"next question"}]`, + `[]`, + } { + if inputContainsFullTranscript(gjson.Parse(raw)) { + t.Fatalf("incremental input must not be detected as full transcript: %s", raw) + } + } +} + +func TestNormalizeSubsequentRequestCompactSkipsMerge(t *testing.T) { + lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ + {"type":"message","role":"user","id":"msg-1","content":"original long prompt"}, + {"type":"message","role":"assistant","id":"msg-2","content":"original long response"}, + {"type":"function_call","id":"fc-1","call_id":"call-old","name":"bash","arguments":"{}"}, + {"type":"function_call_output","id":"fco-1","call_id":"call-old","output":"old result"} + ]}`) + lastResponseOutput := []byte(`[ + {"type":"message","role":"assistant","id":"msg-3","content":"another assistant reply"}, + {"type":"function_call","id":"fc-2","call_id":"call-stale","name":"read","arguments":"{}"} + ]`) + + // Remote compact response: user messages + compaction item, NO assistant message. + // This is the primary compact scenario from Codex CLI. + raw := []byte(`{"type":"response.create","input":[ + {"type":"message","role":"user","id":"msg-1c","content":"compacted user msg"}, + {"type":"compaction","encrypted_content":"conversation summary"} + ]}`) + + normalized, _, errMsg := normalizeResponsesWebsocketRequest(raw, lastRequest, lastResponseOutput) + if errMsg != nil { + t.Fatalf("unexpected error: %v", errMsg.Error) + } + + input := gjson.GetBytes(normalized, "input").Array() + if len(input) != 2 { + t.Fatalf("input len = %d, want 2 (compacted only); stale state was not skipped", len(input)) + } + if input[0].Get("id").String() != "msg-1c" { + t.Fatalf("input[0].id = %q, want %q", input[0].Get("id").String(), "msg-1c") + } + if input[1].Get("type").String() != "compaction" { + t.Fatalf("input[1].type = %q, want %q", input[1].Get("type").String(), "compaction") + } +} + +func TestNormalizeSubsequentRequestIncrementalInputStillMerges(t *testing.T) { + // Normal incremental flow: user sends function_call_output (no assistant message). + lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ + {"type":"message","role":"user","id":"msg-1","content":"hello"} + ]}`) + lastResponseOutput := []byte(`[ + {"type":"message","role":"assistant","id":"msg-2","content":"let me check"}, + {"type":"function_call","id":"fc-1","call_id":"call-1","name":"bash","arguments":"{}"} + ]`) + raw := []byte(`{"type":"response.create","input":[ + {"type":"function_call_output","call_id":"call-1","id":"fco-1","output":"done"} + ]}`) + + normalized, _, errMsg := normalizeResponsesWebsocketRequest(raw, lastRequest, lastResponseOutput) + if errMsg != nil { + t.Fatalf("unexpected error: %v", errMsg.Error) + } + + input := gjson.GetBytes(normalized, "input").Array() + + // Should be merged: msg-1 + msg-2 + fc-1 + fco-1 = 4 items + if len(input) != 4 { + t.Fatalf("input len = %d, want 4 (merged)", len(input)) + } + wantIDs := []string{"msg-1", "msg-2", "fc-1", "fco-1"} + for i, want := range wantIDs { + got := input[i].Get("id").String() + if got != want { + t.Fatalf("input[%d].id = %q, want %q", i, got, want) + } + } +} From d2d0e6f6a1c2e4126151cd1ad78e7809598620ad Mon Sep 17 00:00:00 2001 From: DragonFSKY Date: Mon, 23 Mar 2026 23:27:20 +0800 Subject: [PATCH 06/51] fix(websocket): narrow compact replay detection --- .../openai/openai_responses_websocket.go | 23 ++++-------- .../openai/openai_responses_websocket_test.go | 36 +++++++++++++++++-- 2 files changed, 40 insertions(+), 19 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_websocket.go b/sdk/api/handlers/openai/openai_responses_websocket.go index 6457cd3e..273547d8 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket.go +++ b/sdk/api/handlers/openai/openai_responses_websocket.go @@ -703,29 +703,20 @@ func mergeJSONArrayRaw(existingRaw, appendRaw string) (string, error) { return string(out), nil } -// inputContainsFullTranscript returns true when the input array looks like a -// complete conversation history rather than an incremental append. After a -// client-side compact the input already carries the full (compacted) transcript -// which may include assistant messages or compaction items. Merging that with -// the stale lastRequest / lastResponseOutput would duplicate or break -// function_call / function_call_output pairings, so the caller should use the -// input as-is. +// inputContainsFullTranscript returns true when the input array carries compact +// replay markers that indicate the client already sent the full conversation +// transcript. Merging that input with stale lastRequest/lastResponseOutput +// would duplicate or break function_call/function_call_output pairings, so the +// caller should use the input as-is. // -// Heuristic: the array is a full transcript when it contains either -// - a message with role="assistant", or -// - a compaction item (type="compaction" or "compaction_summary"). -// -// Normal incremental turns only contain user messages or function_call_output -// items and never carry either of these signals. +// Assistant messages alone are not enough to classify the payload as a replay: +// incremental websocket requests may legitimately append assistant items. func inputContainsFullTranscript(input gjson.Result) bool { if !input.IsArray() { return false } for _, item := range input.Array() { t := item.Get("type").String() - if t == "message" && item.Get("role").String() == "assistant" { - return true - } if t == "compaction" || t == "compaction_summary" { return true } diff --git a/sdk/api/handlers/openai/openai_responses_websocket_test.go b/sdk/api/handlers/openai/openai_responses_websocket_test.go index 2c5ef579..82b96f14 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket_test.go +++ b/sdk/api/handlers/openai/openai_responses_websocket_test.go @@ -1401,13 +1401,13 @@ func TestResponsesWebsocketCompactionResetsTurnStateOnTranscriptReplacement(t *t } } -func TestInputContainsFullTranscriptDetectsAssistantMessage(t *testing.T) { +func TestInputContainsFullTranscriptFalseForAssistantMessageOnly(t *testing.T) { input := gjson.Parse(`[ {"type":"message","role":"user","content":"hello"}, {"type":"message","role":"assistant","content":"hi there"} ]`) - if !inputContainsFullTranscript(input) { - t.Fatal("expected full transcript when assistant message is present") + if inputContainsFullTranscript(input) { + t.Fatal("assistant message alone must not be treated as full transcript") } } @@ -1501,3 +1501,33 @@ func TestNormalizeSubsequentRequestIncrementalInputStillMerges(t *testing.T) { } } } + +func TestNormalizeSubsequentRequestAssistantIncrementalInputStillMerges(t *testing.T) { + lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ + {"type":"message","role":"user","id":"msg-1","content":"hello"} + ]}`) + lastResponseOutput := []byte(`[ + {"type":"message","role":"assistant","id":"msg-2","content":"prior assistant"}, + {"type":"function_call","id":"fc-1","call_id":"call-1","name":"bash","arguments":"{}"} + ]`) + raw := []byte(`{"type":"response.append","input":[ + {"type":"message","role":"assistant","id":"msg-3","content":"patched assistant turn"} + ]}`) + + normalized, _, errMsg := normalizeResponsesWebsocketRequest(raw, lastRequest, lastResponseOutput) + if errMsg != nil { + t.Fatalf("unexpected error: %v", errMsg.Error) + } + + input := gjson.GetBytes(normalized, "input").Array() + if len(input) != 4 { + t.Fatalf("input len = %d, want 4 (merged)", len(input)) + } + wantIDs := []string{"msg-1", "msg-2", "fc-1", "msg-3"} + for i, want := range wantIDs { + got := input[i].Get("id").String() + if got != want { + t.Fatalf("input[%d].id = %q, want %q", i, got, want) + } + } +} From 4ca00f79832a2111d23d1d80b490e5a9c3026aab Mon Sep 17 00:00:00 2001 From: DragonFSKY Date: Tue, 24 Mar 2026 19:48:32 +0800 Subject: [PATCH 07/51] fix(websocket): gate compact replay by downstream support --- .../openai/openai_responses_websocket.go | 166 ++++++++++++------ .../openai/openai_responses_websocket_test.go | 106 +++++++++-- 2 files changed, 211 insertions(+), 61 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_websocket.go b/sdk/api/handlers/openai/openai_responses_websocket.go index 273547d8..caf26f13 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket.go +++ b/sdk/api/handlers/openai/openai_responses_websocket.go @@ -116,6 +116,19 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { allowIncrementalInputWithPreviousResponseID = h.websocketUpstreamSupportsIncrementalInputForModel(requestModelName) } + allowCompactionReplayBypass := false + if pinnedAuthID != "" && h != nil && h.AuthManager != nil { + if pinnedAuth, ok := h.AuthManager.GetByID(pinnedAuthID); ok && pinnedAuth != nil { + allowCompactionReplayBypass = responsesWebsocketAuthSupportsCompactionReplay(pinnedAuth) + } + } else { + requestModelName := strings.TrimSpace(gjson.GetBytes(payload, "model").String()) + if requestModelName == "" { + requestModelName = strings.TrimSpace(gjson.GetBytes(lastRequest, "model").String()) + } + allowCompactionReplayBypass = h.websocketUpstreamSupportsCompactionReplayForModel(requestModelName) + } + var requestJSON []byte var updatedLastRequest []byte var errMsg *interfaces.ErrorMessage @@ -124,6 +137,7 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { lastRequest, lastResponseOutput, allowIncrementalInputWithPreviousResponseID, + allowCompactionReplayBypass, ) if errMsg != nil { h.LoggingAPIResponseError(context.WithValue(context.Background(), "gin", c), errMsg) @@ -222,10 +236,10 @@ func websocketUpgradeHeaders(req *http.Request) http.Header { } func normalizeResponsesWebsocketRequest(rawJSON []byte, lastRequest []byte, lastResponseOutput []byte) ([]byte, []byte, *interfaces.ErrorMessage) { - return normalizeResponsesWebsocketRequestWithMode(rawJSON, lastRequest, lastResponseOutput, true) + return normalizeResponsesWebsocketRequestWithMode(rawJSON, lastRequest, lastResponseOutput, true, true) } -func normalizeResponsesWebsocketRequestWithMode(rawJSON []byte, lastRequest []byte, lastResponseOutput []byte, allowIncrementalInputWithPreviousResponseID bool) ([]byte, []byte, *interfaces.ErrorMessage) { +func normalizeResponsesWebsocketRequestWithMode(rawJSON []byte, lastRequest []byte, lastResponseOutput []byte, allowIncrementalInputWithPreviousResponseID bool, allowCompactionReplayBypass bool) ([]byte, []byte, *interfaces.ErrorMessage) { requestType := strings.TrimSpace(gjson.GetBytes(rawJSON, "type").String()) switch requestType { case wsRequestTypeCreate: @@ -233,10 +247,10 @@ func normalizeResponsesWebsocketRequestWithMode(rawJSON []byte, lastRequest []by if len(lastRequest) == 0 { return normalizeResponseCreateRequest(rawJSON) } - return normalizeResponseSubsequentRequest(rawJSON, lastRequest, lastResponseOutput, allowIncrementalInputWithPreviousResponseID) + return normalizeResponseSubsequentRequest(rawJSON, lastRequest, lastResponseOutput, allowIncrementalInputWithPreviousResponseID, allowCompactionReplayBypass) case wsRequestTypeAppend: // log.Infof("responses websocket: response.append request") - return normalizeResponseSubsequentRequest(rawJSON, lastRequest, lastResponseOutput, allowIncrementalInputWithPreviousResponseID) + return normalizeResponseSubsequentRequest(rawJSON, lastRequest, lastResponseOutput, allowIncrementalInputWithPreviousResponseID, allowCompactionReplayBypass) default: return nil, lastRequest, &interfaces.ErrorMessage{ StatusCode: http.StatusBadRequest, @@ -265,7 +279,7 @@ func normalizeResponseCreateRequest(rawJSON []byte) ([]byte, []byte, *interfaces return normalized, bytes.Clone(normalized), nil } -func normalizeResponseSubsequentRequest(rawJSON []byte, lastRequest []byte, lastResponseOutput []byte, allowIncrementalInputWithPreviousResponseID bool) ([]byte, []byte, *interfaces.ErrorMessage) { +func normalizeResponseSubsequentRequest(rawJSON []byte, lastRequest []byte, lastResponseOutput []byte, allowIncrementalInputWithPreviousResponseID bool, allowCompactionReplayBypass bool) ([]byte, []byte, *interfaces.ErrorMessage) { if len(lastRequest) == 0 { return nil, lastRequest, &interfaces.ErrorMessage{ StatusCode: http.StatusBadRequest, @@ -315,16 +329,21 @@ func normalizeResponseSubsequentRequest(rawJSON []byte, lastRequest []byte, last } } - // When the client sends a full conversation transcript (e.g. after compact), - // the input already contains the complete history including assistant messages. - // In that case, skip merging with stale lastRequest/lastResponseOutput to avoid - // breaking function_call / function_call_output pairings. + // When the client sends a compact replay for a downstream that can consume it + // directly, the input already carries the canonical history. In that case, + // skip merging with stale lastRequest/lastResponseOutput to avoid breaking + // function_call / function_call_output pairings. // See: https://github.com/router-for-me/CLIProxyAPI/issues/2207 var mergedInput string - if inputContainsFullTranscript(nextInput) { + if allowCompactionReplayBypass && inputContainsFullTranscript(nextInput) { log.Infof("responses websocket: full transcript detected, skipping stale merge (input items=%d)", len(nextInput.Array())) mergedInput = nextInput.Raw } else { + appendInputRaw := nextInput.Raw + if inputContainsFullTranscript(nextInput) { + appendInputRaw = inputWithoutCompactionItems(nextInput) + } + existingInput := gjson.GetBytes(lastRequest, "input") var errMerge error mergedInput, errMerge = mergeJSONArrayRaw(existingInput.Raw, normalizeJSONArrayRaw(lastResponseOutput)) @@ -335,7 +354,7 @@ func normalizeResponseSubsequentRequest(rawJSON []byte, lastRequest []byte, last } } - mergedInput, errMerge = mergeJSONArrayRaw(mergedInput, nextInput.Raw) + mergedInput, errMerge = mergeJSONArrayRaw(mergedInput, appendInputRaw) if errMerge != nil { return nil, lastRequest, &interfaces.ErrorMessage{ StatusCode: http.StatusBadRequest, @@ -492,72 +511,104 @@ func websocketUpstreamSupportsIncrementalInput(attributes map[string]string, met } func (h *OpenAIResponsesAPIHandler) websocketUpstreamSupportsIncrementalInputForModel(modelName string) bool { - if h == nil || h.AuthManager == nil { + auths, _ := h.responsesWebsocketAvailableAuthsForModel(modelName) + for _, auth := range auths { + if websocketUpstreamSupportsIncrementalInput(auth.Attributes, auth.Metadata) { + return true + } + } + return false +} + +func (h *OpenAIResponsesAPIHandler) websocketUpstreamSupportsCompactionReplayForModel(modelName string) bool { + auths, _ := h.responsesWebsocketAvailableAuthsForModel(modelName) + if len(auths) == 0 { return false } + for _, auth := range auths { + if !responsesWebsocketAuthSupportsCompactionReplay(auth) { + return false + } + } + return true +} - resolvedModelName := modelName +func (h *OpenAIResponsesAPIHandler) responsesWebsocketAvailableAuthsForModel(modelName string) ([]*coreauth.Auth, string) { + if h == nil || h.AuthManager == nil { + return nil, "" + } + resolvedModelName := responsesWebsocketResolvedModelName(modelName) + providerSet, modelKey := responsesWebsocketProviderSetForModel(resolvedModelName) + if len(providerSet) == 0 { + return nil, modelKey + } + + registryRef := registry.GetGlobalRegistry() + now := time.Now() + auths := h.AuthManager.List() + available := make([]*coreauth.Auth, 0, len(auths)) + for _, auth := range auths { + if !responsesWebsocketAuthMatchesModel(auth, providerSet, modelKey, registryRef, now) { + continue + } + available = append(available, auth) + } + return available, modelKey +} + +func responsesWebsocketResolvedModelName(modelName string) string { initialSuffix := thinking.ParseSuffix(modelName) if initialSuffix.ModelName == "auto" { resolvedBase := util.ResolveAutoModel(initialSuffix.ModelName) if initialSuffix.HasSuffix { - resolvedModelName = fmt.Sprintf("%s(%s)", resolvedBase, initialSuffix.RawSuffix) - } else { - resolvedModelName = resolvedBase + return fmt.Sprintf("%s(%s)", resolvedBase, initialSuffix.RawSuffix) } - } else { - resolvedModelName = util.ResolveAutoModel(modelName) + return resolvedBase } + return util.ResolveAutoModel(modelName) +} +func responsesWebsocketProviderSetForModel(resolvedModelName string) (map[string]struct{}, string) { parsed := thinking.ParseSuffix(resolvedModelName) baseModel := strings.TrimSpace(parsed.ModelName) providers := util.GetProviderName(baseModel) if len(providers) == 0 && baseModel != resolvedModelName { providers = util.GetProviderName(resolvedModelName) } - if len(providers) == 0 { - return false - } - providerSet := make(map[string]struct{}, len(providers)) - for i := 0; i < len(providers); i++ { - providerKey := strings.TrimSpace(strings.ToLower(providers[i])) + for _, provider := range providers { + providerKey := strings.TrimSpace(strings.ToLower(provider)) if providerKey == "" { continue } providerSet[providerKey] = struct{}{} } - if len(providerSet) == 0 { - return false - } - modelKey := baseModel if modelKey == "" { modelKey = strings.TrimSpace(resolvedModelName) } - registryRef := registry.GetGlobalRegistry() - now := time.Now() - auths := h.AuthManager.List() - for i := 0; i < len(auths); i++ { - auth := auths[i] - if auth == nil { - continue - } - providerKey := strings.TrimSpace(strings.ToLower(auth.Provider)) - if _, ok := providerSet[providerKey]; !ok { - continue - } - if modelKey != "" && registryRef != nil && !registryRef.ClientSupportsModel(auth.ID, modelKey) { - continue - } - if !responsesWebsocketAuthAvailableForModel(auth, modelKey, now) { - continue - } - if websocketUpstreamSupportsIncrementalInput(auth.Attributes, auth.Metadata) { - return true - } + return providerSet, modelKey +} + +func responsesWebsocketAuthMatchesModel(auth *coreauth.Auth, providerSet map[string]struct{}, modelKey string, registryRef *registry.ModelRegistry, now time.Time) bool { + if auth == nil { + return false } - return false + providerKey := strings.TrimSpace(strings.ToLower(auth.Provider)) + if _, ok := providerSet[providerKey]; !ok { + return false + } + if modelKey != "" && registryRef != nil && !registryRef.ClientSupportsModel(auth.ID, modelKey) { + return false + } + return responsesWebsocketAuthAvailableForModel(auth, modelKey, now) +} + +func responsesWebsocketAuthSupportsCompactionReplay(auth *coreauth.Auth) bool { + if auth == nil { + return false + } + return strings.EqualFold(strings.TrimSpace(auth.Provider), "codex") } func responsesWebsocketAuthAvailableForModel(auth *coreauth.Auth, modelName string, now time.Time) bool { @@ -724,6 +775,21 @@ func inputContainsFullTranscript(input gjson.Result) bool { return false } +func inputWithoutCompactionItems(input gjson.Result) string { + if !input.IsArray() { + return normalizeJSONArrayRaw([]byte(input.Raw)) + } + filtered := make([]string, 0, len(input.Array())) + for _, item := range input.Array() { + t := item.Get("type").String() + if t == "compaction" || t == "compaction_summary" { + continue + } + filtered = append(filtered, item.Raw) + } + return "[" + strings.Join(filtered, ",") + "]" +} + func normalizeJSONArrayRaw(raw []byte) string { trimmed := strings.TrimSpace(string(raw)) if trimmed == "" { diff --git a/sdk/api/handlers/openai/openai_responses_websocket_test.go b/sdk/api/handlers/openai/openai_responses_websocket_test.go index 82b96f14..f2c4319e 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket_test.go +++ b/sdk/api/handlers/openai/openai_responses_websocket_test.go @@ -242,7 +242,7 @@ func TestNormalizeResponsesWebsocketRequestWithPreviousResponseIDIncremental(t * ]`) raw := []byte(`{"type":"response.create","previous_response_id":"resp-1","input":[{"type":"function_call_output","call_id":"call-1","id":"tool-out-1"}]}`) - normalized, next, errMsg := normalizeResponsesWebsocketRequestWithMode(raw, lastRequest, lastResponseOutput, true) + normalized, next, errMsg := normalizeResponsesWebsocketRequestWithMode(raw, lastRequest, lastResponseOutput, true, false) if errMsg != nil { t.Fatalf("unexpected error: %v", errMsg.Error) } @@ -278,7 +278,7 @@ func TestNormalizeResponsesWebsocketRequestWithPreviousResponseIDMergedWhenIncre ]`) raw := []byte(`{"type":"response.create","previous_response_id":"resp-1","input":[{"type":"function_call_output","call_id":"call-1","id":"tool-out-1"}]}`) - normalized, next, errMsg := normalizeResponsesWebsocketRequestWithMode(raw, lastRequest, lastResponseOutput, false) + normalized, next, errMsg := normalizeResponsesWebsocketRequestWithMode(raw, lastRequest, lastResponseOutput, false, false) if errMsg != nil { t.Fatalf("unexpected error: %v", errMsg.Error) } @@ -867,6 +867,53 @@ func TestWebsocketUpstreamSupportsIncrementalInputForModel(t *testing.T) { } } +func TestWebsocketUpstreamSupportsCompactionReplayForModel(t *testing.T) { + manager := coreauth.NewManager(nil, nil, nil) + auth := &coreauth.Auth{ + ID: "auth-codex", + Provider: "codex", + Status: coreauth.StatusActive, + } + if _, err := manager.Register(context.Background(), auth); err != nil { + t.Fatalf("Register auth: %v", err) + } + registry.GetGlobalRegistry().RegisterClient(auth.ID, auth.Provider, []*registry.ModelInfo{{ID: "test-model"}}) + t.Cleanup(func() { + registry.GetGlobalRegistry().UnregisterClient(auth.ID) + }) + + base := handlers.NewBaseAPIHandlers(&sdkconfig.SDKConfig{}, manager) + h := NewOpenAIResponsesAPIHandler(base) + if !h.websocketUpstreamSupportsCompactionReplayForModel("test-model") { + t.Fatalf("expected codex upstream to support compaction replay") + } +} + +func TestWebsocketUpstreamSupportsCompactionReplayForModelFalseWhenMixedBackends(t *testing.T) { + manager := coreauth.NewManager(nil, nil, nil) + auths := []*coreauth.Auth{ + {ID: "auth-codex", Provider: "codex", Status: coreauth.StatusActive}, + {ID: "auth-claude", Provider: "claude", Status: coreauth.StatusActive}, + } + for _, auth := range auths { + if _, err := manager.Register(context.Background(), auth); err != nil { + t.Fatalf("Register auth %s: %v", auth.ID, err) + } + registry.GetGlobalRegistry().RegisterClient(auth.ID, auth.Provider, []*registry.ModelInfo{{ID: "test-model"}}) + } + t.Cleanup(func() { + for _, auth := range auths { + registry.GetGlobalRegistry().UnregisterClient(auth.ID) + } + }) + + base := handlers.NewBaseAPIHandlers(&sdkconfig.SDKConfig{}, manager) + h := NewOpenAIResponsesAPIHandler(base) + if h.websocketUpstreamSupportsCompactionReplayForModel("test-model") { + t.Fatalf("expected mixed backend model to disable compaction replay bypass") + } +} + func TestResponsesWebsocketPrewarmHandledLocallyForSSEUpstream(t *testing.T) { gin.SetMode(gin.TestMode) @@ -1469,6 +1516,45 @@ func TestNormalizeSubsequentRequestCompactSkipsMerge(t *testing.T) { } } +func TestNormalizeSubsequentRequestCompactMergesWhenCompactionReplayUnsupported(t *testing.T) { + lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ + {"type":"message","role":"user","id":"msg-1","content":"original long prompt"}, + {"type":"message","role":"assistant","id":"msg-2","content":"original long response"}, + {"type":"function_call","id":"fc-1","call_id":"call-old","name":"bash","arguments":"{}"}, + {"type":"function_call_output","id":"fco-1","call_id":"call-old","output":"old result"} + ]}`) + lastResponseOutput := []byte(`[ + {"type":"message","role":"assistant","id":"msg-3","content":"another assistant reply"}, + {"type":"function_call","id":"fc-2","call_id":"call-stale","name":"read","arguments":"{}"} + ]`) + raw := []byte(`{"type":"response.create","input":[ + {"type":"message","role":"user","id":"msg-1c","content":"compacted user msg"}, + {"type":"compaction","encrypted_content":"conversation summary"} + ]}`) + + normalized, _, errMsg := normalizeResponsesWebsocketRequestWithMode(raw, lastRequest, lastResponseOutput, false, false) + if errMsg != nil { + t.Fatalf("unexpected error: %v", errMsg.Error) + } + + input := gjson.GetBytes(normalized, "input").Array() + if len(input) != 7 { + t.Fatalf("input len = %d, want 7 (merged fallback without compaction items)", len(input)) + } + wantIDs := []string{"msg-1", "msg-2", "fc-1", "fco-1", "msg-3", "fc-2", "msg-1c"} + for i, want := range wantIDs { + got := input[i].Get("id").String() + if got != want { + t.Fatalf("input[%d].id = %q, want %q", i, got, want) + } + } + for _, item := range input { + if item.Get("type").String() == "compaction" || item.Get("type").String() == "compaction_summary" { + t.Fatalf("compaction items must be stripped for unsupported downstream fallback: %s", item.Raw) + } + } +} + func TestNormalizeSubsequentRequestIncrementalInputStillMerges(t *testing.T) { // Normal incremental flow: user sends function_call_output (no assistant message). lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ @@ -1502,7 +1588,9 @@ func TestNormalizeSubsequentRequestIncrementalInputStillMerges(t *testing.T) { } } -func TestNormalizeSubsequentRequestAssistantIncrementalInputStillMerges(t *testing.T) { +func TestNormalizeSubsequentRequestAssistantInputTriggersTranscriptReplacement(t *testing.T) { + // After dev's shouldReplaceWebsocketTranscript, assistant messages in input + // trigger transcript replacement (no merge with prior state). lastRequest := []byte(`{"model":"gpt-5.4","stream":true,"input":[ {"type":"message","role":"user","id":"msg-1","content":"hello"} ]}`) @@ -1520,14 +1608,10 @@ func TestNormalizeSubsequentRequestAssistantIncrementalInputStillMerges(t *testi } input := gjson.GetBytes(normalized, "input").Array() - if len(input) != 4 { - t.Fatalf("input len = %d, want 4 (merged)", len(input)) + if len(input) != 1 { + t.Fatalf("input len = %d, want 1 (transcript replacement, not merge)", len(input)) } - wantIDs := []string{"msg-1", "msg-2", "fc-1", "msg-3"} - for i, want := range wantIDs { - got := input[i].Get("id").String() - if got != want { - t.Fatalf("input[%d].id = %q, want %q", i, got, want) - } + if input[0].Get("id").String() != "msg-3" { + t.Fatalf("input[0].id = %q, want %q", input[0].Get("id").String(), "msg-3") } } From e707cf7d462f0895f3eb3901aec8f337e68a980e Mon Sep 17 00:00:00 2001 From: Enzo Lucchesi Date: Sat, 18 Apr 2026 10:34:02 -0400 Subject: [PATCH 08/51] fix(claude): only reverse-remap OAuth tool names that were forward-renamed MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit remapOAuthToolNames renames lowercase client-sent tools (e.g. `glob` → `Glob`) to Claude Code equivalents on OAuth requests to avoid tool-name fingerprinting. The reverse pass previously ran against a *global* reverse map and rewrote every tool_use block whose name matched any value in oauthToolRenameMap — regardless of what the client actually sent. For clients that send mixed casing (notably Amp CLI — `Bash`, `Read`, `Grep`, `Task` alongside `glob`, `skill`, etc.) this corrupted the response. Any forward rename in the request set the "renamed" flag, which then unconditionally lowercased every `Bash` in the response to `bash`. Amp's tool registry has `Bash`, not `bash`, so it rejected the tool_use with `tool "bash" is not allowed for smart mode` and tool execution failed. Fix: `remapOAuthToolNames` now returns a per-request map keyed on the upstream (TitleCase) name valued with the original client-sent name. The reverse functions take this map and only touch entries in it. Names the client sent in TitleCase pass through untouched in both directions. - Change remapOAuthToolNames signature from `([]byte, bool)` to `([]byte, map[string]string)`; populate at every rename site (tools[], tool_choice.name, message tool_use, tool_reference, nested tool_reference inside tool_result). - Change reverseRemapOAuthToolNames and reverseRemapOAuthToolNamesFromStreamLine to accept and consume the per-request map; remove the global oauthToolRenameReverseMap. - Update all three executor call sites (Execute, ExecuteStream direct passthrough, ExecuteStream translated) + count_tokens. - Add regression tests for the mixed-case scenario in both the non-streaming and SSE code paths. --- internal/runtime/executor/claude_executor.go | 95 ++++++++++++------- .../runtime/executor/claude_executor_test.go | 91 +++++++++++++++--- 2 files changed, 136 insertions(+), 50 deletions(-) diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 235db1f3..7f00ac08 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -65,14 +65,13 @@ var oauthToolRenameMap = map[string]string{ "notebookedit": "NotebookEdit", } -// oauthToolRenameReverseMap is the inverse of oauthToolRenameMap for response decoding. -var oauthToolRenameReverseMap = func() map[string]string { - m := make(map[string]string, len(oauthToolRenameMap)) - for k, v := range oauthToolRenameMap { - m[v] = k - } - return m -}() +// The reverse map is now computed per-request in remapOAuthToolNames so that +// only names the client actually caused us to rewrite are restored on the +// response. A global reverse map — as used previously — corrupted responses +// for clients that sent mixed casing (e.g. Amp CLI sends `Bash` TitleCase +// alongside `glob` lowercase; the request flagged renames via `glob→Glob`, +// then the global reverse map incorrectly rewrote every `Bash` in the +// response to `bash`, causing Amp to reject the tool_use as unknown). // oauthToolsToRemove lists tool names that must be stripped from OAuth requests // even after remapping. Currently empty — all tools are mapped instead of removed. @@ -191,7 +190,7 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r bodyForTranslation := body bodyForUpstream := body oauthToken := isClaudeOAuthToken(apiKey) - oauthToolNamesRemapped := false + var oauthToolNamesReverseMap map[string]string if oauthToken && !auth.ToolPrefixDisabled() { bodyForUpstream = applyClaudeToolPrefix(body, claudeToolPrefix) } @@ -199,7 +198,7 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r // tools without official counterparts. This prevents Anthropic from // fingerprinting the request as third-party via tool naming patterns. if oauthToken { - bodyForUpstream, oauthToolNamesRemapped = remapOAuthToolNames(bodyForUpstream) + bodyForUpstream, oauthToolNamesReverseMap = remapOAuthToolNames(bodyForUpstream) } // Enable cch signing by default for OAuth tokens (not just experimental flag). // Claude Code always computes cch; missing or invalid cch is a detectable fingerprint. @@ -297,8 +296,8 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r data = stripClaudeToolPrefixFromResponse(data, claudeToolPrefix) } // Reverse the OAuth tool name remap so the downstream client sees original names. - if isClaudeOAuthToken(apiKey) && oauthToolNamesRemapped { - data = reverseRemapOAuthToolNames(data) + if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { + data = reverseRemapOAuthToolNames(data, oauthToolNamesReverseMap) } var param any out := sdktranslator.TranslateNonStream( @@ -373,7 +372,7 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A bodyForTranslation := body bodyForUpstream := body oauthToken := isClaudeOAuthToken(apiKey) - oauthToolNamesRemapped := false + var oauthToolNamesReverseMap map[string]string if oauthToken && !auth.ToolPrefixDisabled() { bodyForUpstream = applyClaudeToolPrefix(body, claudeToolPrefix) } @@ -381,7 +380,7 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A // tools without official counterparts. This prevents Anthropic from // fingerprinting the request as third-party via tool naming patterns. if oauthToken { - bodyForUpstream, oauthToolNamesRemapped = remapOAuthToolNames(bodyForUpstream) + bodyForUpstream, oauthToolNamesReverseMap = remapOAuthToolNames(bodyForUpstream) } // Enable cch signing by default for OAuth tokens (not just experimental flag). if oauthToken || experimentalCCHSigningEnabled(e.cfg, auth) { @@ -475,8 +474,8 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { line = stripClaudeToolPrefixFromStreamLine(line, claudeToolPrefix) } - if isClaudeOAuthToken(apiKey) && oauthToolNamesRemapped { - line = reverseRemapOAuthToolNamesFromStreamLine(line) + if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { + line = reverseRemapOAuthToolNamesFromStreamLine(line, oauthToolNamesReverseMap) } // Forward the line as-is to preserve SSE format cloned := make([]byte, len(line)+1) @@ -505,8 +504,8 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { line = stripClaudeToolPrefixFromStreamLine(line, claudeToolPrefix) } - if isClaudeOAuthToken(apiKey) && oauthToolNamesRemapped { - line = reverseRemapOAuthToolNamesFromStreamLine(line) + if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { + line = reverseRemapOAuthToolNamesFromStreamLine(line, oauthToolNamesReverseMap) } chunks := sdktranslator.TranslateStream( ctx, @@ -1009,8 +1008,25 @@ func isClaudeOAuthToken(apiKey string) bool { // It operates on: tools[].name, tool_choice.name, and all tool_use/tool_reference // references in messages. Removed tools' corresponding tool_result blocks are preserved // (they just become orphaned, which is safe for Claude). -func remapOAuthToolNames(body []byte) ([]byte, bool) { - renamed := false +// +// The returned map is keyed on the upstream (TitleCase) name and maps to the +// client-supplied original name. Callers MUST pass this map to the reverse +// functions so only names the client actually caused us to rewrite are restored +// on the response. A global reverse map (the previous implementation) incorrectly +// rewrote names the client originally sent in TitleCase (e.g. Amp CLI's `Bash`) +// when any OTHER tool in the same request triggered a forward rename (e.g. +// Amp's `glob`→`Glob`), because the global reverse map contained `Bash`→`bash` +// regardless of what the client originally sent. +func remapOAuthToolNames(body []byte) ([]byte, map[string]string) { + reverseMap := make(map[string]string) + recordRename := func(original, renamed string) { + // Preserve the first-seen original name if the same upstream name is + // produced from multiple call sites; they all map back identically. + if _, exists := reverseMap[renamed]; !exists { + reverseMap[renamed] = original + } + } + // 1. Rewrite tools array in a single pass (if present). // IMPORTANT: do not mutate names first and then rebuild from an older gjson // snapshot. gjson results are snapshots of the original bytes; rebuilding from a @@ -1043,7 +1059,7 @@ func remapOAuthToolNames(body []byte) ([]byte, bool) { updatedTool, err := sjson.Set(toolJSON, "name", newName) if err == nil { toolJSON = updatedTool - renamed = true + recordRename(name, newName) } } @@ -1068,7 +1084,7 @@ func remapOAuthToolNames(body []byte) ([]byte, bool) { body, _ = sjson.DeleteBytes(body, "tool_choice") } else if newName, ok := oauthToolRenameMap[tcName]; ok && newName != tcName { body, _ = sjson.SetBytes(body, "tool_choice.name", newName) - renamed = true + recordRename(tcName, newName) } } @@ -1088,14 +1104,14 @@ func remapOAuthToolNames(body []byte) ([]byte, bool) { if newName, ok := oauthToolRenameMap[name]; ok && newName != name { path := fmt.Sprintf("messages.%d.content.%d.name", msgIndex.Int(), contentIndex.Int()) body, _ = sjson.SetBytes(body, path, newName) - renamed = true + recordRename(name, newName) } case "tool_reference": toolName := part.Get("tool_name").String() if newName, ok := oauthToolRenameMap[toolName]; ok && newName != toolName { path := fmt.Sprintf("messages.%d.content.%d.tool_name", msgIndex.Int(), contentIndex.Int()) body, _ = sjson.SetBytes(body, path, newName) - renamed = true + recordRename(toolName, newName) } case "tool_result": // Handle nested tool_reference blocks inside tool_result.content[] @@ -1109,7 +1125,7 @@ func remapOAuthToolNames(body []byte) ([]byte, bool) { if newName, ok := oauthToolRenameMap[nestedToolName]; ok && newName != nestedToolName { nestedPath := fmt.Sprintf("messages.%d.content.%d.content.%d.tool_name", msgIndex.Int(), contentIndex.Int(), nestedIndex.Int()) body, _ = sjson.SetBytes(body, nestedPath, newName) - renamed = true + recordRename(nestedToolName, newName) } } return true @@ -1122,13 +1138,16 @@ func remapOAuthToolNames(body []byte) ([]byte, bool) { }) } - return body, renamed + return body, reverseMap } -// reverseRemapOAuthToolNames reverses the tool name mapping for non-stream responses. -// It maps Claude Code TitleCase names back to the original lowercase names so the -// downstream client receives tool names it recognizes. -func reverseRemapOAuthToolNames(body []byte) []byte { +// reverseRemapOAuthToolNames reverses the tool name mapping for non-stream responses +// using the per-request map produced by remapOAuthToolNames. Names the client sent +// that were NOT forward-renamed are passed through unchanged. +func reverseRemapOAuthToolNames(body []byte, reverseMap map[string]string) []byte { + if len(reverseMap) == 0 { + return body + } content := gjson.GetBytes(body, "content") if !content.Exists() || !content.IsArray() { return body @@ -1138,13 +1157,13 @@ func reverseRemapOAuthToolNames(body []byte) []byte { switch partType { case "tool_use": name := part.Get("name").String() - if origName, ok := oauthToolRenameReverseMap[name]; ok { + if origName, ok := reverseMap[name]; ok { path := fmt.Sprintf("content.%d.name", index.Int()) body, _ = sjson.SetBytes(body, path, origName) } case "tool_reference": toolName := part.Get("tool_name").String() - if origName, ok := oauthToolRenameReverseMap[toolName]; ok { + if origName, ok := reverseMap[toolName]; ok { path := fmt.Sprintf("content.%d.tool_name", index.Int()) body, _ = sjson.SetBytes(body, path, origName) } @@ -1154,8 +1173,12 @@ func reverseRemapOAuthToolNames(body []byte) []byte { return body } -// reverseRemapOAuthToolNamesFromStreamLine reverses the tool name mapping for SSE stream lines. -func reverseRemapOAuthToolNamesFromStreamLine(line []byte) []byte { +// reverseRemapOAuthToolNamesFromStreamLine reverses the tool name mapping for SSE +// stream lines, using the per-request reverseMap produced by remapOAuthToolNames. +func reverseRemapOAuthToolNamesFromStreamLine(line []byte, reverseMap map[string]string) []byte { + if len(reverseMap) == 0 { + return line + } payload := helps.JSONPayload(line) if len(payload) == 0 || !gjson.ValidBytes(payload) { return line @@ -1173,7 +1196,7 @@ func reverseRemapOAuthToolNamesFromStreamLine(line []byte) []byte { switch blockType { case "tool_use": name := contentBlock.Get("name").String() - if origName, ok := oauthToolRenameReverseMap[name]; ok { + if origName, ok := reverseMap[name]; ok { updated, err = sjson.SetBytes(payload, "content_block.name", origName) if err != nil { return line @@ -1183,7 +1206,7 @@ func reverseRemapOAuthToolNamesFromStreamLine(line []byte) []byte { } case "tool_reference": toolName := contentBlock.Get("tool_name").String() - if origName, ok := oauthToolRenameReverseMap[toolName]; ok { + if origName, ok := reverseMap[toolName]; ok { updated, err = sjson.SetBytes(payload, "content_block.tool_name", origName) if err != nil { return line diff --git a/internal/runtime/executor/claude_executor_test.go b/internal/runtime/executor/claude_executor_test.go index c1ce8fc0..0176340b 100644 --- a/internal/runtime/executor/claude_executor_test.go +++ b/internal/runtime/executor/claude_executor_test.go @@ -1989,19 +1989,16 @@ func TestNormalizeClaudeTemperatureForThinking_AfterForcedToolChoiceKeepsOrigina func TestRemapOAuthToolNames_TitleCase_NoReverseNeeded(t *testing.T) { body := []byte(`{"tools":[{"name":"Bash","description":"Run shell commands","input_schema":{"type":"object","properties":{"cmd":{"type":"string"}}}}],"messages":[{"role":"user","content":[{"type":"text","text":"hi"}]}]}`) - out, renamed := remapOAuthToolNames(body) - if renamed { - t.Fatalf("renamed = true, want false") + out, reverseMap := remapOAuthToolNames(body) + if len(reverseMap) != 0 { + t.Fatalf("reverseMap = %v, want empty", reverseMap) } if got := gjson.GetBytes(out, "tools.0.name").String(); got != "Bash" { t.Fatalf("tools.0.name = %q, want %q", got, "Bash") } resp := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"Bash","input":{"cmd":"ls"}}]}`) - reversed := resp - if renamed { - reversed = reverseRemapOAuthToolNames(resp) - } + reversed := reverseRemapOAuthToolNames(resp, reverseMap) if got := gjson.GetBytes(reversed, "content.0.name").String(); got != "Bash" { t.Fatalf("content.0.name = %q, want %q", got, "Bash") } @@ -2010,20 +2007,86 @@ func TestRemapOAuthToolNames_TitleCase_NoReverseNeeded(t *testing.T) { func TestRemapOAuthToolNames_Lowercase_ReverseApplied(t *testing.T) { body := []byte(`{"tools":[{"name":"bash","description":"Run shell commands","input_schema":{"type":"object","properties":{"cmd":{"type":"string"}}}}],"messages":[{"role":"user","content":[{"type":"text","text":"hi"}]}]}`) - out, renamed := remapOAuthToolNames(body) - if !renamed { - t.Fatalf("renamed = false, want true") + out, reverseMap := remapOAuthToolNames(body) + if reverseMap["Bash"] != "bash" { + t.Fatalf("reverseMap = %v, want entry Bash->bash", reverseMap) } if got := gjson.GetBytes(out, "tools.0.name").String(); got != "Bash" { t.Fatalf("tools.0.name = %q, want %q", got, "Bash") } resp := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"Bash","input":{"cmd":"ls"}}]}`) - reversed := resp - if renamed { - reversed = reverseRemapOAuthToolNames(resp) - } + reversed := reverseRemapOAuthToolNames(resp, reverseMap) if got := gjson.GetBytes(reversed, "content.0.name").String(); got != "bash" { t.Fatalf("content.0.name = %q, want %q", got, "bash") } } + +// TestRemapOAuthToolNames_MixedCase_OnlyRenamedToolsReversed is the regression +// test for a case where a single request contains both a TitleCase tool (which +// must pass through unchanged) and a lowercase tool that we forward-rename. +// Before the fix, triggering ANY forward rename caused the reverse pass to +// lowercase every TitleCase tool in the response using a global reverse map, +// corrupting tool names the client originally sent in TitleCase (notably Amp +// CLI's `Bash`, which its registry lookup cannot find as `bash`). +func TestRemapOAuthToolNames_MixedCase_OnlyRenamedToolsReversed(t *testing.T) { + body := []byte(`{"tools":[` + + `{"name":"Bash","input_schema":{"type":"object","properties":{"cmd":{"type":"string"}}}},` + + `{"name":"glob","input_schema":{"type":"object","properties":{"filePattern":{"type":"string"}}}}` + + `]}`) + + out, reverseMap := remapOAuthToolNames(body) + + // Forward: TitleCase `Bash` is not a forward-map key, must pass through. + if got := gjson.GetBytes(out, "tools.0.name").String(); got != "Bash" { + t.Fatalf("tools.0.name = %q, want %q (TitleCase tool must not be renamed)", got, "Bash") + } + // Forward: `glob` is a forward-map key, upstream sees `Glob`. + if got := gjson.GetBytes(out, "tools.1.name").String(); got != "Glob" { + t.Fatalf("tools.1.name = %q, want %q", got, "Glob") + } + + // Reverse map records ONLY the rename that happened. + if len(reverseMap) != 1 || reverseMap["Glob"] != "glob" { + t.Fatalf("reverseMap = %v, want {Glob:glob}", reverseMap) + } + + // Upstream responds with a `Bash` tool_use. Since we never renamed `Bash`, + // reverseRemap MUST leave it alone. + bashResp := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"Bash","input":{"cmd":"ls"}}]}`) + reversed := reverseRemapOAuthToolNames(bashResp, reverseMap) + if got := gjson.GetBytes(reversed, "content.0.name").String(); got != "Bash" { + t.Fatalf("content.0.name = %q, want %q (Bash must be preserved; was never forward-renamed)", got, "Bash") + } + + // Upstream responds with a `Glob` tool_use. Since we renamed `glob`→`Glob`, + // reverseRemap MUST restore the original `glob`. + globResp := []byte(`{"content":[{"type":"tool_use","id":"toolu_02","name":"Glob","input":{"filePattern":"**/*.go"}}]}`) + reversed = reverseRemapOAuthToolNames(globResp, reverseMap) + if got := gjson.GetBytes(reversed, "content.0.name").String(); got != "glob" { + t.Fatalf("content.0.name = %q, want %q (Glob must be restored to client's original `glob`)", got, "glob") + } +} + +// TestReverseRemapOAuthToolNamesFromStreamLine_HonorsPerRequestMap guards the +// SSE streaming code path against the same mixed-case bug. +func TestReverseRemapOAuthToolNamesFromStreamLine_HonorsPerRequestMap(t *testing.T) { + reverseMap := map[string]string{"Glob": "glob"} + + // Bash block was never renamed, must pass through as-is. + bashLine := []byte(`data: {"type":"content_block_start","index":0,"content_block":{"type":"tool_use","id":"toolu_01","name":"Bash","input":{}}}`) + out := reverseRemapOAuthToolNamesFromStreamLine(bashLine, reverseMap) + if !bytes.Contains(out, []byte(`"name":"Bash"`)) { + t.Fatalf("Bash should be preserved, got: %s", string(out)) + } + if bytes.Contains(out, []byte(`"name":"bash"`)) { + t.Fatalf("Bash must not be lowercased, got: %s", string(out)) + } + + // Glob block IS in the reverseMap, must be restored to `glob`. + globLine := []byte(`data: {"type":"content_block_start","index":0,"content_block":{"type":"tool_use","id":"toolu_02","name":"Glob","input":{}}}`) + out = reverseRemapOAuthToolNamesFromStreamLine(globLine, reverseMap) + if !bytes.Contains(out, []byte(`"name":"glob"`)) { + t.Fatalf("Glob should be restored to glob, got: %s", string(out)) + } +} From 03ea4e569fb1df9237d7032469a744737cd30d72 Mon Sep 17 00:00:00 2001 From: edlsh Date: Sat, 18 Apr 2026 12:49:02 -0400 Subject: [PATCH 09/51] perf(claude): pre-allocate reverseMap capacity Address Gemini code review suggestion: the reverseMap can contain at most len(oauthToolRenameMap) entries, so pre-allocating avoids reallocations as entries are added. --- internal/runtime/executor/claude_executor.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 7f00ac08..78fa3cd6 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -1018,7 +1018,7 @@ func isClaudeOAuthToken(apiKey string) bool { // Amp's `glob`→`Glob`), because the global reverse map contained `Bash`→`bash` // regardless of what the client originally sent. func remapOAuthToolNames(body []byte) ([]byte, map[string]string) { - reverseMap := make(map[string]string) + reverseMap := make(map[string]string, len(oauthToolRenameMap)) recordRename := func(original, renamed string) { // Preserve the first-seen original name if the same upstream name is // produced from multiple call sites; they all map back identically. From fc1ddf365f489ca465e5fd85334d01303e635f11 Mon Sep 17 00:00:00 2001 From: Enzo Lucchesi Date: Sun, 19 Apr 2026 14:36:25 +0000 Subject: [PATCH 10/51] fix(claude): centralize oauth tool-name transform flow --- internal/runtime/executor/claude_executor.go | 74 +++++++++---------- .../runtime/executor/claude_executor_test.go | 64 ++++++++++++++++ 2 files changed, 100 insertions(+), 38 deletions(-) diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 78fa3cd6..2dbff1d3 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -191,14 +191,8 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r bodyForUpstream := body oauthToken := isClaudeOAuthToken(apiKey) var oauthToolNamesReverseMap map[string]string - if oauthToken && !auth.ToolPrefixDisabled() { - bodyForUpstream = applyClaudeToolPrefix(body, claudeToolPrefix) - } - // Remap third-party tool names to Claude Code equivalents and remove - // tools without official counterparts. This prevents Anthropic from - // fingerprinting the request as third-party via tool naming patterns. if oauthToken { - bodyForUpstream, oauthToolNamesReverseMap = remapOAuthToolNames(bodyForUpstream) + bodyForUpstream, oauthToolNamesReverseMap = prepareClaudeOAuthToolNamesForUpstream(bodyForUpstream, claudeToolPrefix, auth.ToolPrefixDisabled()) } // Enable cch signing by default for OAuth tokens (not just experimental flag). // Claude Code always computes cch; missing or invalid cch is a detectable fingerprint. @@ -292,13 +286,7 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r } else { reporter.Publish(ctx, helps.ParseClaudeUsage(data)) } - if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { - data = stripClaudeToolPrefixFromResponse(data, claudeToolPrefix) - } - // Reverse the OAuth tool name remap so the downstream client sees original names. - if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { - data = reverseRemapOAuthToolNames(data, oauthToolNamesReverseMap) - } + data = restoreClaudeOAuthToolNamesFromResponse(data, claudeToolPrefix, auth.ToolPrefixDisabled(), oauthToolNamesReverseMap) var param any out := sdktranslator.TranslateNonStream( ctx, @@ -373,14 +361,8 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A bodyForUpstream := body oauthToken := isClaudeOAuthToken(apiKey) var oauthToolNamesReverseMap map[string]string - if oauthToken && !auth.ToolPrefixDisabled() { - bodyForUpstream = applyClaudeToolPrefix(body, claudeToolPrefix) - } - // Remap third-party tool names to Claude Code equivalents and remove - // tools without official counterparts. This prevents Anthropic from - // fingerprinting the request as third-party via tool naming patterns. if oauthToken { - bodyForUpstream, oauthToolNamesReverseMap = remapOAuthToolNames(bodyForUpstream) + bodyForUpstream, oauthToolNamesReverseMap = prepareClaudeOAuthToolNamesForUpstream(bodyForUpstream, claudeToolPrefix, auth.ToolPrefixDisabled()) } // Enable cch signing by default for OAuth tokens (not just experimental flag). if oauthToken || experimentalCCHSigningEnabled(e.cfg, auth) { @@ -471,12 +453,7 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A if detail, ok := helps.ParseClaudeStreamUsage(line); ok { reporter.Publish(ctx, detail) } - if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { - line = stripClaudeToolPrefixFromStreamLine(line, claudeToolPrefix) - } - if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { - line = reverseRemapOAuthToolNamesFromStreamLine(line, oauthToolNamesReverseMap) - } + line = restoreClaudeOAuthToolNamesFromStreamLine(line, claudeToolPrefix, auth.ToolPrefixDisabled(), oauthToolNamesReverseMap) // Forward the line as-is to preserve SSE format cloned := make([]byte, len(line)+1) copy(cloned, line) @@ -501,12 +478,7 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A if detail, ok := helps.ParseClaudeStreamUsage(line); ok { reporter.Publish(ctx, detail) } - if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { - line = stripClaudeToolPrefixFromStreamLine(line, claudeToolPrefix) - } - if isClaudeOAuthToken(apiKey) && len(oauthToolNamesReverseMap) > 0 { - line = reverseRemapOAuthToolNamesFromStreamLine(line, oauthToolNamesReverseMap) - } + line = restoreClaudeOAuthToolNamesFromStreamLine(line, claudeToolPrefix, auth.ToolPrefixDisabled(), oauthToolNamesReverseMap) chunks := sdktranslator.TranslateStream( ctx, to, @@ -556,12 +528,8 @@ func (e *ClaudeExecutor) CountTokens(ctx context.Context, auth *cliproxyauth.Aut // Extract betas from body and convert to header (for count_tokens too) var extraBetas []string extraBetas, body = extractAndRemoveBetas(body) - if isClaudeOAuthToken(apiKey) && !auth.ToolPrefixDisabled() { - body = applyClaudeToolPrefix(body, claudeToolPrefix) - } - // Remap tool names for OAuth token requests to avoid third-party fingerprinting. if isClaudeOAuthToken(apiKey) { - body, _ = remapOAuthToolNames(body) + body, _ = prepareClaudeOAuthToolNamesForUpstream(body, claudeToolPrefix, auth.ToolPrefixDisabled()) } url := fmt.Sprintf("%s/v1/messages/count_tokens?beta=true", baseURL) @@ -1001,6 +969,36 @@ func isClaudeOAuthToken(apiKey string) bool { return strings.Contains(apiKey, "sk-ant-oat") } +// prepareClaudeOAuthToolNamesForUpstream applies the Claude OAuth tool-name +// transforms in the same order across request paths. Remap runs before prefixing +// so any future non-empty prefix still composes correctly with the per-request +// reverse map. +func prepareClaudeOAuthToolNamesForUpstream(body []byte, prefix string, prefixDisabled bool) ([]byte, map[string]string) { + body, reverseMap := remapOAuthToolNames(body) + if !prefixDisabled { + body = applyClaudeToolPrefix(body, prefix) + } + return body, reverseMap +} + +// restoreClaudeOAuthToolNamesFromResponse undoes the Claude OAuth tool-name +// transforms for non-stream responses in reverse order. +func restoreClaudeOAuthToolNamesFromResponse(body []byte, prefix string, prefixDisabled bool, reverseMap map[string]string) []byte { + if !prefixDisabled { + body = stripClaudeToolPrefixFromResponse(body, prefix) + } + return reverseRemapOAuthToolNames(body, reverseMap) +} + +// restoreClaudeOAuthToolNamesFromStreamLine undoes the Claude OAuth tool-name +// transforms for SSE lines in reverse order. +func restoreClaudeOAuthToolNamesFromStreamLine(line []byte, prefix string, prefixDisabled bool, reverseMap map[string]string) []byte { + if !prefixDisabled { + line = stripClaudeToolPrefixFromStreamLine(line, prefix) + } + return reverseRemapOAuthToolNamesFromStreamLine(line, reverseMap) +} + // remapOAuthToolNames renames third-party tool names to Claude Code equivalents // and removes tools without an official counterpart. This prevents Anthropic from // fingerprinting the request as a third-party client via tool naming patterns. diff --git a/internal/runtime/executor/claude_executor_test.go b/internal/runtime/executor/claude_executor_test.go index 0176340b..9011be04 100644 --- a/internal/runtime/executor/claude_executor_test.go +++ b/internal/runtime/executor/claude_executor_test.go @@ -2090,3 +2090,67 @@ func TestReverseRemapOAuthToolNamesFromStreamLine_HonorsPerRequestMap(t *testing t.Fatalf("Glob should be restored to glob, got: %s", string(out)) } } + +func TestPrepareClaudeOAuthToolNamesForUpstream_MixedCaseWithPrefix(t *testing.T) { + body := []byte(`{"tools":[` + + `{"name":"Bash","input_schema":{"type":"object","properties":{"cmd":{"type":"string"}}}},` + + `{"name":"glob","input_schema":{"type":"object","properties":{"filePattern":{"type":"string"}}}}` + + `],"messages":[{"role":"assistant","content":[` + + `{"type":"tool_use","id":"toolu_01","name":"Bash","input":{}},` + + `{"type":"tool_use","id":"toolu_02","name":"glob","input":{}}` + + `]}]}`) + + out, reverseMap := prepareClaudeOAuthToolNamesForUpstream(body, "proxy_", false) + + if got := gjson.GetBytes(out, "tools.0.name").String(); got != "proxy_Bash" { + t.Fatalf("tools.0.name = %q, want %q", got, "proxy_Bash") + } + if got := gjson.GetBytes(out, "tools.1.name").String(); got != "proxy_Glob" { + t.Fatalf("tools.1.name = %q, want %q", got, "proxy_Glob") + } + if got := gjson.GetBytes(out, "messages.0.content.0.name").String(); got != "proxy_Bash" { + t.Fatalf("messages.0.content.0.name = %q, want %q", got, "proxy_Bash") + } + if got := gjson.GetBytes(out, "messages.0.content.1.name").String(); got != "proxy_Glob" { + t.Fatalf("messages.0.content.1.name = %q, want %q", got, "proxy_Glob") + } + if len(reverseMap) != 1 || reverseMap["Glob"] != "glob" { + t.Fatalf("reverseMap = %v, want {Glob:glob}", reverseMap) + } +} + +func TestRestoreClaudeOAuthToolNamesFromResponse_MixedCaseWithPrefix(t *testing.T) { + reverseMap := map[string]string{"Glob": "glob"} + resp := []byte(`{"content":[` + + `{"type":"tool_use","id":"toolu_01","name":"proxy_Bash","input":{}},` + + `{"type":"tool_use","id":"toolu_02","name":"proxy_Glob","input":{}}` + + `]}`) + + out := restoreClaudeOAuthToolNamesFromResponse(resp, "proxy_", false, reverseMap) + + if got := gjson.GetBytes(out, "content.0.name").String(); got != "Bash" { + t.Fatalf("content.0.name = %q, want %q", got, "Bash") + } + if got := gjson.GetBytes(out, "content.1.name").String(); got != "glob" { + t.Fatalf("content.1.name = %q, want %q", got, "glob") + } +} + +func TestRestoreClaudeOAuthToolNamesFromStreamLine_MixedCaseWithPrefix(t *testing.T) { + reverseMap := map[string]string{"Glob": "glob"} + + bashLine := []byte(`data: {"type":"content_block_start","index":0,"content_block":{"type":"tool_use","id":"toolu_01","name":"proxy_Bash","input":{}}}`) + out := restoreClaudeOAuthToolNamesFromStreamLine(bashLine, "proxy_", false, reverseMap) + if !bytes.Contains(out, []byte(`"name":"Bash"`)) { + t.Fatalf("Bash should be preserved, got: %s", string(out)) + } + if bytes.Contains(out, []byte(`"name":"bash"`)) { + t.Fatalf("Bash must not be lowercased, got: %s", string(out)) + } + + globLine := []byte(`data: {"type":"content_block_start","index":0,"content_block":{"type":"tool_use","id":"toolu_02","name":"proxy_Glob","input":{}}}`) + out = restoreClaudeOAuthToolNamesFromStreamLine(globLine, "proxy_", false, reverseMap) + if !bytes.Contains(out, []byte(`"name":"glob"`)) { + t.Fatalf("Glob should be restored to glob, got: %s", string(out)) + } +} From 95318ad46dce78e19a74e06872b4901b83985bc9 Mon Sep 17 00:00:00 2001 From: edlsh Date: Mon, 13 Apr 2026 09:39:01 -0400 Subject: [PATCH 11/51] fix(amp): preserve lowercase glob tool name --- internal/api/modules/amp/response_rewriter.go | 50 ++++++++++++++++++ .../api/modules/amp/response_rewriter_test.go | 51 +++++++++++++++++++ 2 files changed, 101 insertions(+) diff --git a/internal/api/modules/amp/response_rewriter.go b/internal/api/modules/amp/response_rewriter.go index 707fe576..895c494e 100644 --- a/internal/api/modules/amp/response_rewriter.go +++ b/internal/api/modules/amp/response_rewriter.go @@ -123,6 +123,52 @@ func (rw *ResponseRewriter) Flush() { var modelFieldPaths = []string{"message.model", "model", "modelVersion", "response.model", "response.modelVersion"} +// ampCanonicalToolNames maps tool names to the exact casing expected by the +// Amp mode tool whitelist (case-sensitive match). +var ampCanonicalToolNames = map[string]string{ + "bash": "Bash", + "read": "Read", + "grep": "Grep", + "glob": "glob", + "task": "Task", + "check": "Check", +} + +// normalizeAmpToolNames fixes tool_use block names to match Amp's canonical casing. +// Some upstream models return lowercase tool names (e.g. "bash" instead of "Bash") +// which causes Amp's case-sensitive mode whitelist to reject them. +func normalizeAmpToolNames(data []byte) []byte { + // Non-streaming: content[].name in tool_use blocks + for index, block := range gjson.GetBytes(data, "content").Array() { + if block.Get("type").String() != "tool_use" { + continue + } + name := block.Get("name").String() + if canonical, ok := ampCanonicalToolNames[strings.ToLower(name)]; ok && name != canonical { + path := fmt.Sprintf("content.%d.name", index) + var err error + data, err = sjson.SetBytes(data, path, canonical) + if err != nil { + log.Warnf("Amp ResponseRewriter: failed to normalize tool name %q to %q: %v", name, canonical, err) + } + } + } + + // Streaming: content_block.name in content_block_start events + if gjson.GetBytes(data, "content_block.type").String() == "tool_use" { + name := gjson.GetBytes(data, "content_block.name").String() + if canonical, ok := ampCanonicalToolNames[strings.ToLower(name)]; ok && name != canonical { + var err error + data, err = sjson.SetBytes(data, "content_block.name", canonical) + if err != nil { + log.Warnf("Amp ResponseRewriter: failed to normalize streaming tool name %q to %q: %v", name, canonical, err) + } + } + } + + return data +} + // ensureAmpSignature injects empty signature fields into tool_use/thinking blocks // in API responses so that the Amp TUI does not crash on P.signature.length. func ensureAmpSignature(data []byte) []byte { @@ -179,6 +225,7 @@ func (rw *ResponseRewriter) suppressAmpThinking(data []byte) []byte { func (rw *ResponseRewriter) rewriteModelInResponse(data []byte) []byte { data = ensureAmpSignature(data) + data = normalizeAmpToolNames(data) data = rw.suppressAmpThinking(data) if len(data) == 0 { return data @@ -278,6 +325,9 @@ func (rw *ResponseRewriter) rewriteStreamEvent(data []byte) []byte { // Inject empty signature where needed data = ensureAmpSignature(data) + // Normalize tool names to canonical casing + data = normalizeAmpToolNames(data) + // Rewrite model name if rw.originalModel != "" { for _, path := range modelFieldPaths { diff --git a/internal/api/modules/amp/response_rewriter_test.go b/internal/api/modules/amp/response_rewriter_test.go index ac95dfc6..a3a350cb 100644 --- a/internal/api/modules/amp/response_rewriter_test.go +++ b/internal/api/modules/amp/response_rewriter_test.go @@ -175,6 +175,57 @@ func TestSanitizeAmpRequestBody_MixedInvalidThinkingAndToolUseSignature(t *testi } } +func TestNormalizeAmpToolNames_NonStreaming(t *testing.T) { + input := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"bash","input":{"cmd":"ls"}},{"type":"tool_use","id":"toolu_02","name":"read","input":{"path":"/tmp"}},{"type":"text","text":"hello"}]}`) + result := normalizeAmpToolNames(input) + + if !contains(result, []byte(`"name":"Bash"`)) { + t.Errorf("expected bash->Bash, got %s", string(result)) + } + if !contains(result, []byte(`"name":"Read"`)) { + t.Errorf("expected read->Read, got %s", string(result)) + } + if contains(result, []byte(`"name":"bash"`)) { + t.Errorf("expected lowercase bash to be replaced, got %s", string(result)) + } +} + +func TestNormalizeAmpToolNames_Streaming(t *testing.T) { + input := []byte(`{"type":"content_block_start","index":1,"content_block":{"type":"tool_use","name":"grep","id":"toolu_01","input":{}}}`) + result := normalizeAmpToolNames(input) + + if !contains(result, []byte(`"name":"Grep"`)) { + t.Errorf("expected grep->Grep in streaming, got %s", string(result)) + } +} + +func TestNormalizeAmpToolNames_AlreadyCorrect(t *testing.T) { + input := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"Bash","input":{"cmd":"ls"}}]}`) + result := normalizeAmpToolNames(input) + + if string(result) != string(input) { + t.Errorf("expected no modification for correctly-cased tool, got %s", string(result)) + } +} + +func TestNormalizeAmpToolNames_GlobPreserved(t *testing.T) { + input := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"glob","input":{"pattern":"*.go"}}]}`) + result := normalizeAmpToolNames(input) + + if string(result) != string(input) { + t.Errorf("expected glob to remain lowercase, got %s", string(result)) + } +} + +func TestNormalizeAmpToolNames_UnknownToolUntouched(t *testing.T) { + input := []byte(`{"content":[{"type":"tool_use","id":"toolu_01","name":"edit_file","input":{"path":"/tmp/x"}}]}`) + result := normalizeAmpToolNames(input) + + if string(result) != string(input) { + t.Errorf("expected no modification for unknown tool, got %s", string(result)) + } +} + func contains(data, substr []byte) bool { for i := 0; i <= len(data)-len(substr); i++ { if string(data[i:i+len(substr)]) == string(substr) { From fd45dece7f027ef198f00cad8c6455a333a36ca4 Mon Sep 17 00:00:00 2001 From: edlsh Date: Fri, 24 Apr 2026 15:15:01 -0400 Subject: [PATCH 12/51] fix(openai): repair empty responses stream output --- .../openai/openai_responses_handlers.go | 122 +++++++++++++++++- .../openai_responses_handlers_stream_test.go | 34 ++++- 2 files changed, 151 insertions(+), 5 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_handlers.go b/sdk/api/handlers/openai/openai_responses_handlers.go index 8969ce2f..67c648dc 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers.go +++ b/sdk/api/handlers/openai/openai_responses_handlers.go @@ -13,6 +13,7 @@ import ( "fmt" "io" "net/http" + "sort" "github.com/gin-gonic/gin" . "github.com/router-for-me/CLIProxyAPI/v6/internal/constant" @@ -45,7 +46,9 @@ func writeResponsesSSEChunk(w io.Writer, chunk []byte) { } type responsesSSEFramer struct { - pending []byte + pending []byte + outputItems map[int][]byte + outputOrder []int } func (f *responsesSSEFramer) WriteChunk(w io.Writer, chunk []byte) { @@ -61,7 +64,7 @@ func (f *responsesSSEFramer) WriteChunk(w io.Writer, chunk []byte) { if frameLen == 0 { break } - writeResponsesSSEChunk(w, f.pending[:frameLen]) + f.writeFrame(w, f.pending[:frameLen]) copy(f.pending, f.pending[frameLen:]) f.pending = f.pending[:len(f.pending)-frameLen] } @@ -72,7 +75,7 @@ func (f *responsesSSEFramer) WriteChunk(w io.Writer, chunk []byte) { if len(f.pending) == 0 || !responsesSSECanEmitWithoutDelimiter(f.pending) { return } - writeResponsesSSEChunk(w, f.pending) + f.writeFrame(w, f.pending) f.pending = f.pending[:0] } @@ -88,10 +91,121 @@ func (f *responsesSSEFramer) Flush(w io.Writer) { f.pending = f.pending[:0] return } - writeResponsesSSEChunk(w, f.pending) + f.writeFrame(w, f.pending) f.pending = f.pending[:0] } +func (f *responsesSSEFramer) writeFrame(w io.Writer, frame []byte) { + writeResponsesSSEChunk(w, f.repairFrame(frame)) +} + +func (f *responsesSSEFramer) repairFrame(frame []byte) []byte { + payload, ok := responsesSSEDataPayload(frame) + if !ok || len(payload) == 0 || bytes.Equal(payload, []byte("[DONE]")) || !json.Valid(payload) { + return frame + } + + switch gjson.GetBytes(payload, "type").String() { + case "response.output_item.done": + f.recordOutputItem(payload) + case "response.completed": + repaired := f.repairCompletedPayload(payload) + if !bytes.Equal(repaired, payload) { + return responsesSSEFrameWithData(frame, repaired) + } + } + return frame +} + +func responsesSSEDataPayload(frame []byte) ([]byte, bool) { + var payload []byte + found := false + for _, line := range bytes.Split(frame, []byte("\n")) { + line = bytes.TrimRight(line, "\r") + trimmed := bytes.TrimSpace(line) + if !bytes.HasPrefix(trimmed, []byte("data:")) { + continue + } + data := bytes.TrimSpace(trimmed[len("data:"):]) + if found { + payload = append(payload, '\n') + } + payload = append(payload, data...) + found = true + } + return payload, found +} + +func responsesSSEFrameWithData(frame, payload []byte) []byte { + var out bytes.Buffer + for _, line := range bytes.Split(frame, []byte("\n")) { + line = bytes.TrimRight(line, "\r") + trimmed := bytes.TrimSpace(line) + if len(trimmed) == 0 || bytes.HasPrefix(trimmed, []byte("data:")) { + continue + } + out.Write(line) + out.WriteByte('\n') + } + out.WriteString("data: ") + out.Write(payload) + out.WriteString("\n\n") + return out.Bytes() +} + +func (f *responsesSSEFramer) recordOutputItem(payload []byte) { + item := gjson.GetBytes(payload, "item") + if !item.Exists() || !item.IsObject() || item.Get("type").String() == "" { + return + } + + index := len(f.outputOrder) + if outputIndex := gjson.GetBytes(payload, "output_index"); outputIndex.Exists() { + index = int(outputIndex.Int()) + } + if f.outputItems == nil { + f.outputItems = make(map[int][]byte) + } + if _, exists := f.outputItems[index]; !exists { + f.outputOrder = append(f.outputOrder, index) + } + f.outputItems[index] = append([]byte(nil), item.Raw...) +} + +func (f *responsesSSEFramer) repairCompletedPayload(payload []byte) []byte { + if len(f.outputOrder) == 0 { + return payload + } + output := gjson.GetBytes(payload, "response.output") + if output.Exists() && (!output.IsArray() || len(output.Array()) > 0) { + return payload + } + + var outputJSON bytes.Buffer + outputJSON.WriteByte('[') + indexes := append([]int(nil), f.outputOrder...) + sort.Ints(indexes) + written := 0 + for _, index := range indexes { + item, ok := f.outputItems[index] + if !ok { + continue + } + if written > 0 { + outputJSON.WriteByte(',') + } + outputJSON.Write(item) + written++ + } + outputJSON.WriteByte(']') + + repaired, err := sjson.SetRawBytes(payload, "response.output", outputJSON.Bytes()) + if err != nil { + return payload + } + return repaired +} + func responsesSSEFrameLen(chunk []byte) int { if len(chunk) == 0 { return 0 diff --git a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go index ef16fe80..8b3f79e3 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go +++ b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go @@ -10,6 +10,7 @@ import ( "github.com/router-for-me/CLIProxyAPI/v6/internal/interfaces" "github.com/router-for-me/CLIProxyAPI/v6/sdk/api/handlers" sdkconfig "github.com/router-for-me/CLIProxyAPI/v6/sdk/config" + "github.com/tidwall/gjson" ) func newResponsesStreamTestHandler(t *testing.T) (*OpenAIResponsesAPIHandler, *httptest.ResponseRecorder, *gin.Context, http.Flusher) { @@ -53,12 +54,43 @@ func TestForwardResponsesStreamSeparatesDataOnlySSEChunks(t *testing.T) { t.Errorf("unexpected first event.\nGot: %q\nWant: %q", parts[0], expectedPart1) } - expectedPart2 := "data: {\"type\":\"response.completed\",\"response\":{\"id\":\"resp-1\",\"output\":[]}}" + expectedPart2 := "data: {\"type\":\"response.completed\",\"response\":{\"id\":\"resp-1\",\"output\":[{\"type\":\"function_call\",\"arguments\":\"{}\"}]}}" if parts[1] != expectedPart2 { t.Errorf("unexpected second event.\nGot: %q\nWant: %q", parts[1], expectedPart2) } } +func TestForwardResponsesStreamRepairsEmptyCompletedOutputFromDoneItems(t *testing.T) { + h, recorder, c, flusher := newResponsesStreamTestHandler(t) + + data := make(chan []byte, 3) + errs := make(chan *interfaces.ErrorMessage) + data <- []byte(`data: {"type":"response.output_item.done","output_index":0,"item":{"type":"reasoning","id":"rs-1","summary":[]}}`) + data <- []byte(`data: {"type":"response.output_item.done","output_index":1,"item":{"type":"function_call","id":"fc-1","call_id":"call-1","name":"shell","arguments":"{\"cmd\":\"pwd\"}","status":"completed"}}`) + data <- []byte(`data: {"type":"response.completed","response":{"id":"resp-1","output":[]}}`) + close(data) + close(errs) + + h.forwardResponsesStream(c, flusher, func(error) {}, data, errs, nil) + + parts := strings.Split(strings.TrimSpace(recorder.Body.String()), "\n\n") + if len(parts) != 3 { + t.Fatalf("expected 3 SSE events, got %d. Body: %q", len(parts), recorder.Body.String()) + } + + payload := strings.TrimPrefix(parts[2], "data: ") + output := gjson.Get(payload, "response.output") + if !output.IsArray() || len(output.Array()) != 2 { + t.Fatalf("expected repaired completed output with 2 items, got %s", output.Raw) + } + if got := gjson.Get(payload, "response.output.1.name").String(); got != "shell" { + t.Fatalf("expected function_call name to be preserved, got %q in %s", got, payload) + } + if got := gjson.Get(payload, "response.output.1.arguments").String(); got != `{"cmd":"pwd"}` { + t.Fatalf("expected function_call arguments to be preserved, got %q in %s", got, payload) + } +} + func TestForwardResponsesStreamReassemblesSplitSSEEventChunks(t *testing.T) { h, recorder, c, flusher := newResponsesStreamTestHandler(t) From d36e70e9dcfd5e4a79f2165a582e76e385423895 Mon Sep 17 00:00:00 2001 From: edlsh Date: Sat, 25 Apr 2026 18:06:00 -0400 Subject: [PATCH 13/51] fix(openai): preserve unindexed response output items --- .../openai/openai_responses_handlers.go | 36 ++++++++++++------- .../openai_responses_handlers_stream_test.go | 31 ++++++++++++++++ 2 files changed, 54 insertions(+), 13 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_handlers.go b/sdk/api/handlers/openai/openai_responses_handlers.go index 67c648dc..578977d6 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers.go +++ b/sdk/api/handlers/openai/openai_responses_handlers.go @@ -46,9 +46,10 @@ func writeResponsesSSEChunk(w io.Writer, chunk []byte) { } type responsesSSEFramer struct { - pending []byte - outputItems map[int][]byte - outputOrder []int + pending []byte + outputItems map[int][]byte + outputOrder []int + unindexedOutputItems [][]byte } func (f *responsesSSEFramer) WriteChunk(w io.Writer, chunk []byte) { @@ -159,21 +160,23 @@ func (f *responsesSSEFramer) recordOutputItem(payload []byte) { return } - index := len(f.outputOrder) if outputIndex := gjson.GetBytes(payload, "output_index"); outputIndex.Exists() { - index = int(outputIndex.Int()) + index := int(outputIndex.Int()) + if f.outputItems == nil { + f.outputItems = make(map[int][]byte) + } + if _, exists := f.outputItems[index]; !exists { + f.outputOrder = append(f.outputOrder, index) + } + f.outputItems[index] = append([]byte(nil), item.Raw...) + return } - if f.outputItems == nil { - f.outputItems = make(map[int][]byte) - } - if _, exists := f.outputItems[index]; !exists { - f.outputOrder = append(f.outputOrder, index) - } - f.outputItems[index] = append([]byte(nil), item.Raw...) + + f.unindexedOutputItems = append(f.unindexedOutputItems, append([]byte(nil), item.Raw...)) } func (f *responsesSSEFramer) repairCompletedPayload(payload []byte) []byte { - if len(f.outputOrder) == 0 { + if len(f.outputOrder) == 0 && len(f.unindexedOutputItems) == 0 { return payload } output := gjson.GetBytes(payload, "response.output") @@ -197,6 +200,13 @@ func (f *responsesSSEFramer) repairCompletedPayload(payload []byte) []byte { outputJSON.Write(item) written++ } + for _, item := range f.unindexedOutputItems { + if written > 0 { + outputJSON.WriteByte(',') + } + outputJSON.Write(item) + written++ + } outputJSON.WriteByte(']') repaired, err := sjson.SetRawBytes(payload, "response.output", outputJSON.Bytes()) diff --git a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go index 8b3f79e3..3851278f 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go +++ b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go @@ -91,6 +91,37 @@ func TestForwardResponsesStreamRepairsEmptyCompletedOutputFromDoneItems(t *testi } } +func TestForwardResponsesStreamRepairsMixedIndexedAndUnindexedDoneItems(t *testing.T) { + h, recorder, c, flusher := newResponsesStreamTestHandler(t) + + data := make(chan []byte, 3) + errs := make(chan *interfaces.ErrorMessage) + data <- []byte(`data: {"type":"response.output_item.done","output_index":1,"item":{"type":"function_call","id":"fc-1","call_id":"call-1","name":"shell","arguments":"{}","status":"completed"}}`) + data <- []byte(`data: {"type":"response.output_item.done","item":{"type":"message","id":"msg-1","role":"assistant","content":[{"type":"output_text","text":"done"}]}}`) + data <- []byte(`data: {"type":"response.completed","response":{"id":"resp-1","output":[]}}`) + close(data) + close(errs) + + h.forwardResponsesStream(c, flusher, func(error) {}, data, errs, nil) + + parts := strings.Split(strings.TrimSpace(recorder.Body.String()), "\n\n") + if len(parts) != 3 { + t.Fatalf("expected 3 SSE events, got %d. Body: %q", len(parts), recorder.Body.String()) + } + + payload := strings.TrimPrefix(parts[2], "data: ") + output := gjson.Get(payload, "response.output") + if !output.IsArray() || len(output.Array()) != 2 { + t.Fatalf("expected repaired completed output with 2 items, got %s", output.Raw) + } + if got := gjson.Get(payload, "response.output.0.name").String(); got != "shell" { + t.Fatalf("expected indexed function_call to be preserved first, got %q in %s", got, payload) + } + if got := gjson.Get(payload, "response.output.1.id").String(); got != "msg-1" { + t.Fatalf("expected unindexed message to be appended, got %q in %s", got, payload) + } +} + func TestForwardResponsesStreamReassemblesSplitSSEEventChunks(t *testing.T) { h, recorder, c, flusher := newResponsesStreamTestHandler(t) From 80eb03709a569a7620b6bba4f1e4f30f8170d3bd Mon Sep 17 00:00:00 2001 From: edlsh Date: Sat, 25 Apr 2026 18:12:27 -0400 Subject: [PATCH 14/51] fix(openai): preserve multiline repaired SSE data --- .../openai/openai_responses_handlers.go | 9 +++-- .../openai_responses_handlers_stream_test.go | 34 +++++++++++++++++++ 2 files changed, 40 insertions(+), 3 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_handlers.go b/sdk/api/handlers/openai/openai_responses_handlers.go index 578977d6..8dd1a0a7 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers.go +++ b/sdk/api/handlers/openai/openai_responses_handlers.go @@ -148,9 +148,12 @@ func responsesSSEFrameWithData(frame, payload []byte) []byte { out.Write(line) out.WriteByte('\n') } - out.WriteString("data: ") - out.Write(payload) - out.WriteString("\n\n") + for _, line := range bytes.Split(payload, []byte("\n")) { + out.WriteString("data: ") + out.Write(line) + out.WriteByte('\n') + } + out.WriteByte('\n') return out.Bytes() } diff --git a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go index 3851278f..151da9a7 100644 --- a/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go +++ b/sdk/api/handlers/openai/openai_responses_handlers_stream_test.go @@ -122,6 +122,40 @@ func TestForwardResponsesStreamRepairsMixedIndexedAndUnindexedDoneItems(t *testi } } +func TestForwardResponsesStreamRepairsMultilineCompletedOutputAsSSEDataLines(t *testing.T) { + h, recorder, c, flusher := newResponsesStreamTestHandler(t) + + data := make(chan []byte, 2) + errs := make(chan *interfaces.ErrorMessage) + data <- []byte(`data: {"type":"response.output_item.done","item":{"type":"function_call","arguments":"{}"}}`) + data <- []byte("data: {\"type\":\"response.completed\",\ndata: \"response\":{\"id\":\"resp-1\",\"output\":[]}}\n\n") + close(data) + close(errs) + + h.forwardResponsesStream(c, flusher, func(error) {}, data, errs, nil) + + parts := strings.Split(strings.TrimSpace(recorder.Body.String()), "\n\n") + if len(parts) != 2 { + t.Fatalf("expected 2 SSE events, got %d. Body: %q", len(parts), recorder.Body.String()) + } + + completedFrame := []byte(parts[1]) + for _, line := range strings.Split(parts[1], "\n") { + if line != "" && !strings.HasPrefix(line, "data: ") { + t.Fatalf("expected every completed payload line to be an SSE data line, got %q in %q", line, parts[1]) + } + } + + payload, ok := responsesSSEDataPayload(completedFrame) + if !ok { + t.Fatalf("expected completed frame to contain data payload: %q", parts[1]) + } + output := gjson.GetBytes(payload, "response.output") + if !output.IsArray() || len(output.Array()) != 1 { + t.Fatalf("expected repaired completed output with 1 item, got %s from %q", output.Raw, payload) + } +} + func TestForwardResponsesStreamReassemblesSplitSSEEventChunks(t *testing.T) { h, recorder, c, flusher := newResponsesStreamTestHandler(t) From aa70d13f606e96d570c65e4eeb1792913f0a51ce Mon Sep 17 00:00:00 2001 From: C4AL <104809382+C4AL@users.noreply.github.com> Date: Thu, 30 Apr 2026 20:36:37 +0800 Subject: [PATCH 15/51] docs: add CodexCliPlus to README ecosystem list --- README.md | 4 ++++ README_CN.md | 4 ++++ README_JA.md | 4 ++++ 3 files changed, 12 insertions(+) diff --git a/README.md b/README.md index 70f5a044..93ef6f71 100644 --- a/README.md +++ b/README.md @@ -187,6 +187,10 @@ Ready-to-use cross-platform quota inspector for CLIProxyAPI, supporting per-acco Standalone persistence and visualization service for CLIProxyAPI, with periodic data sync, SQLite storage, aggregate APIs, and a built-in dashboard for usage and statistics. +### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) + +Windows-focused, local-first desktop management platform for Codex CLI built on CLIProxyAPI, focused on simplifying local setup, account and runtime management, and providing a more complete Codex CLI experience for local users. + > [!NOTE] > If you developed a project based on CLIProxyAPI, please open a PR to add it to this list. diff --git a/README_CN.md b/README_CN.md index e08e4ed1..6199095c 100644 --- a/README_CN.md +++ b/README_CN.md @@ -183,6 +183,10 @@ Shadow AI 是一款专为受限环境设计的 AI 辅助工具。提供无窗口 独立的 CLIProxyAPI 使用量持久化与可视化服务,定期同步 CPA 数据,存储到 SQLite,提供聚合 API,并内置使用量分析与统计仪表盘。 +### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) + +基于 CLIProxyAPI 的 Windows Codex CLI 本地优先桌面管理平台,聚焦简化本机配置、账号与运行状态管理,并为本地用户提供更完整的 Codex CLI 使用体验。 + > [!NOTE] > 如果你开发了基于 CLIProxyAPI 的项目,请提交一个 PR(拉取请求)将其添加到此列表中。 diff --git a/README_JA.md b/README_JA.md index 6360320c..1bb30d48 100644 --- a/README_JA.md +++ b/README_JA.md @@ -182,6 +182,10 @@ CLIProxyAPI向けのすぐに使えるクロスプラットフォームのクォ CLIProxyAPI向けの独立した使用量永続化・可視化サービス。CPAデータを定期同期してSQLiteに保存し、集計APIと、使用量や各種統計を確認できる組み込みダッシュボードを提供します。 +### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) + +CLIProxyAPIを基盤にしたWindows向けのローカル優先Codex CLIデスクトップ管理プラットフォーム。ローカル設定、アカウント、実行状態の管理を簡素化し、ローカルユーザーにより包括的なCodex CLI体験を提供します。 + > [!NOTE] > CLIProxyAPIをベースにプロジェクトを開発した場合は、PRを送ってこのリストに追加してください。 From 4035abc0cd6b7dabdff49b695256d6d6ceb03245 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Thu, 30 Apr 2026 23:36:07 +0800 Subject: [PATCH 16/51] refactor(logging): replace gin-specific context handling with generic context-based request metadata utilities - Introduced reusable utilities in `requestmeta` to manage endpoint and response status in request contexts. - Refactored plugins and handlers to use context-based metadata, removing direct dependency on `gin`. - Updated tests to validate new context utilities and replaced `gin`-based context handling. Fixed: #3166 --- internal/logging/requestmeta.go | 62 ++++++++++++++++++++++ internal/redisqueue/plugin.go | 42 ++------------- internal/redisqueue/plugin_test.go | 84 +++++++++++++++++++++++++++--- internal/usage/logger_plugin.go | 28 ++-------- sdk/api/handlers/handlers.go | 26 ++++++++- 5 files changed, 174 insertions(+), 68 deletions(-) create mode 100644 internal/logging/requestmeta.go diff --git a/internal/logging/requestmeta.go b/internal/logging/requestmeta.go new file mode 100644 index 00000000..a28d7c62 --- /dev/null +++ b/internal/logging/requestmeta.go @@ -0,0 +1,62 @@ +package logging + +import ( + "context" + "sync/atomic" +) + +type endpointKey struct{} +type responseStatusKey struct{} + +type responseStatusHolder struct { + status atomic.Int32 +} + +func WithEndpoint(ctx context.Context, endpoint string) context.Context { + if ctx == nil { + ctx = context.Background() + } + return context.WithValue(ctx, endpointKey{}, endpoint) +} + +func GetEndpoint(ctx context.Context) string { + if ctx == nil { + return "" + } + if endpoint, ok := ctx.Value(endpointKey{}).(string); ok { + return endpoint + } + return "" +} + +func WithResponseStatusHolder(ctx context.Context) context.Context { + if ctx == nil { + ctx = context.Background() + } + if holder, ok := ctx.Value(responseStatusKey{}).(*responseStatusHolder); ok && holder != nil { + return ctx + } + return context.WithValue(ctx, responseStatusKey{}, &responseStatusHolder{}) +} + +func SetResponseStatus(ctx context.Context, status int) { + if ctx == nil || status <= 0 { + return + } + holder, ok := ctx.Value(responseStatusKey{}).(*responseStatusHolder) + if !ok || holder == nil { + return + } + holder.status.Store(int32(status)) +} + +func GetResponseStatus(ctx context.Context) int { + if ctx == nil { + return 0 + } + holder, ok := ctx.Value(responseStatusKey{}).(*responseStatusHolder) + if !ok || holder == nil { + return 0 + } + return int(holder.status.Load()) +} diff --git a/internal/redisqueue/plugin.go b/internal/redisqueue/plugin.go index a805e5da..39739dbe 100644 --- a/internal/redisqueue/plugin.go +++ b/internal/redisqueue/plugin.go @@ -3,11 +3,9 @@ package redisqueue import ( "context" "encoding/json" - "net/http" "strings" "time" - "github.com/gin-gonic/gin" internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" internalusage "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" @@ -46,11 +44,6 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec } apiKey := strings.TrimSpace(record.APIKey) requestID := strings.TrimSpace(internallogging.GetRequestID(ctx)) - if requestID == "" { - if ginCtx, ok := ctx.Value("gin").(*gin.Context); ok && ginCtx != nil { - requestID = strings.TrimSpace(internallogging.GetGinRequestID(ginCtx)) - } - } tokens := internalusage.TokenStats{ InputTokens: record.Detail.InputTokens, @@ -106,40 +99,15 @@ type queuedUsageDetail struct { } func resolveSuccess(ctx context.Context) bool { - if ctx == nil { - return true - } - ginCtx, ok := ctx.Value("gin").(*gin.Context) - if !ok || ginCtx == nil { - return true - } - status := ginCtx.Writer.Status() + status := internallogging.GetResponseStatus(ctx) if status == 0 { return true } - return status < http.StatusBadRequest + return status < httpStatusBadRequest } func resolveEndpoint(ctx context.Context) string { - if ctx == nil { - return "" - } - ginCtx, ok := ctx.Value("gin").(*gin.Context) - if !ok || ginCtx == nil || ginCtx.Request == nil { - return "" - } - - path := strings.TrimSpace(ginCtx.FullPath()) - if path == "" && ginCtx.Request.URL != nil { - path = strings.TrimSpace(ginCtx.Request.URL.Path) - } - if path == "" { - return "" - } - - method := strings.TrimSpace(ginCtx.Request.Method) - if method == "" { - return path - } - return method + " " + path + return strings.TrimSpace(internallogging.GetEndpoint(ctx)) } + +const httpStatusBadRequest = 400 diff --git a/internal/redisqueue/plugin_test.go b/internal/redisqueue/plugin_test.go index 907b8aee..1e8bda48 100644 --- a/internal/redisqueue/plugin_test.go +++ b/internal/redisqueue/plugin_test.go @@ -16,9 +16,10 @@ import ( func TestUsageQueuePluginPayloadIncludesStableFieldsAndSuccess(t *testing.T) { withEnabledQueue(t, func() { - ginCtx := newTestGinContext(t, http.MethodPost, "/v1/chat/completions", http.StatusOK) - internallogging.SetGinRequestID(ginCtx, "gin-request-id-ignored") - ctx := context.WithValue(internallogging.WithRequestID(context.Background(), "ctx-request-id"), "gin", ginCtx) + ctx := internallogging.WithRequestID(context.Background(), "ctx-request-id") + ctx = internallogging.WithEndpoint(ctx, "POST /v1/chat/completions") + ctx = internallogging.WithResponseStatusHolder(ctx) + internallogging.SetResponseStatus(ctx, http.StatusOK) plugin := &usageQueuePlugin{} plugin.HandleUsage(ctx, coreusage.Record{ @@ -49,9 +50,10 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndSuccess(t *testing.T) { func TestUsageQueuePluginPayloadIncludesStableFieldsAndFailureAndGinRequestID(t *testing.T) { withEnabledQueue(t, func() { - ginCtx := newTestGinContext(t, http.MethodGet, "/v1/responses", http.StatusInternalServerError) - internallogging.SetGinRequestID(ginCtx, "gin-request-id") - ctx := context.WithValue(context.Background(), "gin", ginCtx) + ctx := internallogging.WithRequestID(context.Background(), "gin-request-id") + ctx = internallogging.WithEndpoint(ctx, "GET /v1/responses") + ctx = internallogging.WithResponseStatusHolder(ctx) + internallogging.SetResponseStatus(ctx, http.StatusInternalServerError) plugin := &usageQueuePlugin{} plugin.HandleUsage(ctx, coreusage.Record{ @@ -80,6 +82,47 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndFailureAndGinRequestID(t }) } +func TestUsageQueuePluginAsyncIgnoresRecycledGinContext(t *testing.T) { + withEnabledQueue(t, func() { + ginCtx := newTestGinContext(t, http.MethodPost, "/v1/chat/completions", http.StatusOK) + ctx := context.WithValue(context.Background(), "gin", ginCtx) + ctx = internallogging.WithRequestID(ctx, "ctx-request-id") + ctx = internallogging.WithEndpoint(ctx, "POST /v1/chat/completions") + ctx = internallogging.WithResponseStatusHolder(ctx) + internallogging.SetResponseStatus(ctx, http.StatusInternalServerError) + + mgr := coreusage.NewManager(16) + defer mgr.Stop() + + mgr.Register(pluginFunc(func(_ context.Context, _ coreusage.Record) { + ginCtx.Request = httptest.NewRequest(http.MethodGet, "http://example.com/v1/responses", nil) + ginCtx.Status(http.StatusOK) + })) + mgr.Register(&usageQueuePlugin{}) + + mgr.Publish(ctx, coreusage.Record{ + Provider: "openai", + Model: "gpt-5.4", + APIKey: "test-key", + AuthIndex: "0", + AuthType: "apikey", + Source: "user@example.com", + RequestedAt: time.Date(2026, 4, 25, 0, 0, 0, 0, time.UTC), + Latency: 1500 * time.Millisecond, + Detail: coreusage.Detail{ + InputTokens: 10, + OutputTokens: 20, + TotalTokens: 30, + }, + }) + + payload := waitForSinglePayload(t, 2*time.Second) + requireStringField(t, payload, "endpoint", "POST /v1/chat/completions") + requireStringField(t, payload, "request_id", "ctx-request-id") + requireBoolField(t, payload, "failed", true) + }) +} + func withEnabledQueue(t *testing.T, fn func()) { t.Helper() @@ -127,6 +170,29 @@ func popSinglePayload(t *testing.T) map[string]json.RawMessage { return payload } +func waitForSinglePayload(t *testing.T, timeout time.Duration) map[string]json.RawMessage { + t.Helper() + + deadline := time.Now().Add(timeout) + for time.Now().Before(deadline) { + items := PopOldest(10) + if len(items) == 0 { + time.Sleep(10 * time.Millisecond) + continue + } + if len(items) != 1 { + t.Fatalf("PopOldest() items = %d, want 1", len(items)) + } + var payload map[string]json.RawMessage + if err := json.Unmarshal(items[0], &payload); err != nil { + t.Fatalf("unmarshal payload: %v", err) + } + return payload + } + t.Fatalf("timeout waiting for queued payload") + return nil +} + func requireStringField(t *testing.T, payload map[string]json.RawMessage, key, want string) { t.Helper() @@ -143,6 +209,12 @@ func requireStringField(t *testing.T, payload map[string]json.RawMessage, key, w } } +type pluginFunc func(context.Context, coreusage.Record) + +func (fn pluginFunc) HandleUsage(ctx context.Context, record coreusage.Record) { + fn(ctx, record) +} + func requireBoolField(t *testing.T, payload map[string]json.RawMessage, key string, want bool) { t.Helper() diff --git a/internal/usage/logger_plugin.go b/internal/usage/logger_plugin.go index 803d005e..9d59de4f 100644 --- a/internal/usage/logger_plugin.go +++ b/internal/usage/logger_plugin.go @@ -11,7 +11,7 @@ import ( "sync/atomic" "time" - "github.com/gin-gonic/gin" + internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" ) @@ -401,21 +401,8 @@ func dedupKey(apiName, modelName string, detail RequestDetail) string { func resolveAPIIdentifier(ctx context.Context, record coreusage.Record) string { if ctx != nil { - if ginCtx, ok := ctx.Value("gin").(*gin.Context); ok && ginCtx != nil { - path := ginCtx.FullPath() - if path == "" && ginCtx.Request != nil { - path = ginCtx.Request.URL.Path - } - method := "" - if ginCtx.Request != nil { - method = ginCtx.Request.Method - } - if path != "" { - if method != "" { - return method + " " + path - } - return path - } + if endpoint := strings.TrimSpace(internallogging.GetEndpoint(ctx)); endpoint != "" { + return endpoint } } if record.Provider != "" { @@ -425,14 +412,7 @@ func resolveAPIIdentifier(ctx context.Context, record coreusage.Record) string { } func resolveSuccess(ctx context.Context) bool { - if ctx == nil { - return true - } - ginCtx, ok := ctx.Value("gin").(*gin.Context) - if !ok || ginCtx == nil { - return true - } - status := ginCtx.Writer.Status() + status := internallogging.GetResponseStatus(ctx) if status == 0 { return true } diff --git a/sdk/api/handlers/handlers.go b/sdk/api/handlers/handlers.go index 22f7c41a..52b2a4fd 100644 --- a/sdk/api/handlers/handlers.go +++ b/sdk/api/handlers/handlers.go @@ -375,11 +375,32 @@ func (h *BaseAPIHandler) GetContextWithCancel(handler interfaces.APIHandler, c * if requestCtx != nil && logging.GetRequestID(parentCtx) == "" { if requestID := logging.GetRequestID(requestCtx); requestID != "" { parentCtx = logging.WithRequestID(parentCtx, requestID) - } else if requestID := logging.GetGinRequestID(c); requestID != "" { + } else if requestID = logging.GetGinRequestID(c); requestID != "" { parentCtx = logging.WithRequestID(parentCtx, requestID) } } newCtx, cancel := context.WithCancel(parentCtx) + + endpoint := "" + if c != nil && c.Request != nil { + path := strings.TrimSpace(c.FullPath()) + if path == "" && c.Request.URL != nil { + path = strings.TrimSpace(c.Request.URL.Path) + } + if path != "" { + method := strings.TrimSpace(c.Request.Method) + if method != "" { + endpoint = method + " " + path + } else { + endpoint = path + } + } + } + if endpoint != "" { + newCtx = logging.WithEndpoint(newCtx, endpoint) + } + newCtx = logging.WithResponseStatusHolder(newCtx) + cancelCtx := newCtx if requestCtx != nil && requestCtx != parentCtx { go func() { @@ -393,6 +414,9 @@ func (h *BaseAPIHandler) GetContextWithCancel(handler interfaces.APIHandler, c * newCtx = context.WithValue(newCtx, "gin", c) newCtx = context.WithValue(newCtx, "handler", handler) return newCtx, func(params ...interface{}) { + if c != nil { + logging.SetResponseStatus(cancelCtx, c.Writer.Status()) + } if h.Cfg.RequestLog && len(params) == 1 { if existing, exists := c.Get("API_RESPONSE"); exists { if existingBytes, ok := existing.([]byte); ok && len(bytes.TrimSpace(existingBytes)) > 0 { From 61879190002c267d70ad0dd3992c817ad0014b23 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Fri, 1 May 2026 22:55:22 +0800 Subject: [PATCH 17/51] feat: add support for recent request tracking in auth records - Implemented `RecentRequestsSnapshot` in `Auth` to capture bucketed recent request data. - Added new fields and methods to `Auth` for tracking request success and failure counts over time. - Updated `/v0/management/auth-files` response to include recent request data for each auth record. - Introduced unit tests to validate request tracking and snapshot generation logic. --- .../api/handlers/management/auth_files.go | 1 + .../auth_files_recent_requests_test.go | 87 ++++++++++++++++++ sdk/cliproxy/auth/conductor.go | 1 + .../auth/conductor_recent_requests_test.go | 44 ++++++++++ sdk/cliproxy/auth/types.go | 88 ++++++++++++++++++- sdk/cliproxy/auth/types_test.go | 75 +++++++++++++++- 6 files changed, 294 insertions(+), 2 deletions(-) create mode 100644 internal/api/handlers/management/auth_files_recent_requests_test.go create mode 100644 sdk/cliproxy/auth/conductor_recent_requests_test.go diff --git a/internal/api/handlers/management/auth_files.go b/internal/api/handlers/management/auth_files.go index 8f7b8c5e..2bcfaac4 100644 --- a/internal/api/handlers/management/auth_files.go +++ b/internal/api/handlers/management/auth_files.go @@ -388,6 +388,7 @@ func (h *Handler) buildAuthFileEntry(auth *coreauth.Auth) gin.H { "source": "memory", "size": int64(0), } + entry["recent_requests"] = auth.RecentRequestsSnapshot(time.Now()) if email := authEmail(auth); email != "" { entry["email"] = email } diff --git a/internal/api/handlers/management/auth_files_recent_requests_test.go b/internal/api/handlers/management/auth_files_recent_requests_test.go new file mode 100644 index 00000000..fd28ca1d --- /dev/null +++ b/internal/api/handlers/management/auth_files_recent_requests_test.go @@ -0,0 +1,87 @@ +package management + +import ( + "context" + "encoding/json" + "net/http" + "net/http/httptest" + "testing" + + "github.com/gin-gonic/gin" + "github.com/router-for-me/CLIProxyAPI/v6/internal/config" + coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" +) + +func TestListAuthFiles_IncludesRecentRequestsBuckets(t *testing.T) { + t.Setenv("MANAGEMENT_PASSWORD", "") + gin.SetMode(gin.TestMode) + + manager := coreauth.NewManager(nil, nil, nil) + record := &coreauth.Auth{ + ID: "runtime-only-auth-1", + Provider: "codex", + Attributes: map[string]string{ + "runtime_only": "true", + }, + Metadata: map[string]any{ + "type": "codex", + }, + } + if _, errRegister := manager.Register(context.Background(), record); errRegister != nil { + t.Fatalf("failed to register auth record: %v", errRegister) + } + + h := NewHandlerWithoutConfigFilePath(&config.Config{AuthDir: t.TempDir()}, manager) + h.tokenStore = &memoryAuthStore{} + + rec := httptest.NewRecorder() + ginCtx, _ := gin.CreateTestContext(rec) + req := httptest.NewRequest(http.MethodGet, "/v0/management/auth-files", nil) + ginCtx.Request = req + + h.ListAuthFiles(ginCtx) + + if rec.Code != http.StatusOK { + t.Fatalf("expected list status %d, got %d with body %s", http.StatusOK, rec.Code, rec.Body.String()) + } + + var payload map[string]any + if errUnmarshal := json.Unmarshal(rec.Body.Bytes(), &payload); errUnmarshal != nil { + t.Fatalf("failed to decode list payload: %v", errUnmarshal) + } + filesRaw, ok := payload["files"].([]any) + if !ok { + t.Fatalf("expected files array, payload: %#v", payload) + } + if len(filesRaw) != 1 { + t.Fatalf("expected 1 auth entry, got %d", len(filesRaw)) + } + + fileEntry, ok := filesRaw[0].(map[string]any) + if !ok { + t.Fatalf("expected file entry object, got %#v", filesRaw[0]) + } + + recentRaw, ok := fileEntry["recent_requests"].([]any) + if !ok { + t.Fatalf("expected recent_requests array, got %#v", fileEntry["recent_requests"]) + } + if len(recentRaw) != 20 { + t.Fatalf("expected 20 recent_requests buckets, got %d", len(recentRaw)) + } + for idx, item := range recentRaw { + bucket, ok := item.(map[string]any) + if !ok { + t.Fatalf("expected bucket object at %d, got %#v", idx, item) + } + if _, ok := bucket["time"].(string); !ok { + t.Fatalf("expected bucket time string at %d, got %#v", idx, bucket["time"]) + } + if _, ok := bucket["success"].(float64); !ok { + t.Fatalf("expected bucket success number at %d, got %#v", idx, bucket["success"]) + } + if _, ok := bucket["failed"].(float64); !ok { + t.Fatalf("expected bucket failed number at %d, got %#v", idx, bucket["failed"]) + } + } +} diff --git a/sdk/cliproxy/auth/conductor.go b/sdk/cliproxy/auth/conductor.go index 6571518d..61a0e413 100644 --- a/sdk/cliproxy/auth/conductor.go +++ b/sdk/cliproxy/auth/conductor.go @@ -2021,6 +2021,7 @@ func (m *Manager) MarkResult(ctx context.Context, result Result) { m.mu.Lock() if auth, ok := m.auths[result.AuthID]; ok && auth != nil { now := time.Now() + auth.recordRecentRequest(now, result.Success) if result.Success { if result.Model != "" { diff --git a/sdk/cliproxy/auth/conductor_recent_requests_test.go b/sdk/cliproxy/auth/conductor_recent_requests_test.go new file mode 100644 index 00000000..3f5a7212 --- /dev/null +++ b/sdk/cliproxy/auth/conductor_recent_requests_test.go @@ -0,0 +1,44 @@ +package auth + +import ( + "context" + "testing" + "time" +) + +func TestManagerMarkResultRecordsRecentRequests(t *testing.T) { + mgr := NewManager(nil, nil, nil) + auth := &Auth{ + ID: "auth-1", + Provider: "antigravity", + Attributes: map[string]string{ + "runtime_only": "true", + }, + Metadata: map[string]any{ + "type": "antigravity", + }, + } + + if _, err := mgr.Register(WithSkipPersist(context.Background()), auth); err != nil { + t.Fatalf("Register returned error: %v", err) + } + + mgr.MarkResult(context.Background(), Result{AuthID: "auth-1", Provider: "antigravity", Model: "gpt-5", Success: true}) + mgr.MarkResult(context.Background(), Result{AuthID: "auth-1", Provider: "antigravity", Model: "gpt-5", Success: false}) + + gotAuth, ok := mgr.GetByID("auth-1") + if !ok || gotAuth == nil { + t.Fatalf("GetByID returned ok=%v auth=%v", ok, gotAuth) + } + + snapshot := gotAuth.RecentRequestsSnapshot(time.Now()) + var successTotal int64 + var failedTotal int64 + for _, bucket := range snapshot { + successTotal += bucket.Success + failedTotal += bucket.Failed + } + if successTotal != 1 || failedTotal != 1 { + t.Fatalf("totals = success=%d failed=%d, want 1/1", successTotal, failedTotal) + } +} diff --git a/sdk/cliproxy/auth/types.go b/sdk/cliproxy/auth/types.go index f30f4dc0..93dd3881 100644 --- a/sdk/cliproxy/auth/types.go +++ b/sdk/cliproxy/auth/types.go @@ -92,7 +92,29 @@ type Auth struct { // Runtime carries non-serialisable data used during execution (in-memory only). Runtime any `json:"-"` - indexAssigned bool `json:"-"` + recentRequests recentRequestRing `json:"-"` + indexAssigned bool `json:"-"` +} + +const ( + recentRequestBucketSeconds int64 = 10 * 60 + recentRequestBucketCount = 20 +) + +type recentRequestBucket struct { + bucketID int64 + success int64 + failed int64 +} + +type recentRequestRing struct { + buckets [recentRequestBucketCount]recentRequestBucket +} + +type RecentRequestBucket struct { + Time string `json:"time"` + Success int64 `json:"success"` + Failed int64 `json:"failed"` } // QuotaState contains limiter tracking data for a credential. @@ -125,6 +147,70 @@ type ModelState struct { UpdatedAt time.Time `json:"updated_at"` } +func recentRequestBucketID(now time.Time) int64 { + if now.IsZero() { + return 0 + } + return now.Unix() / recentRequestBucketSeconds +} + +func recentRequestBucketIndex(bucketID int64) int { + mod := bucketID % int64(recentRequestBucketCount) + if mod < 0 { + mod += int64(recentRequestBucketCount) + } + return int(mod) +} + +func formatRecentRequestBucketLabel(bucketID int64) string { + start := time.Unix(bucketID*recentRequestBucketSeconds, 0).In(time.Local) + end := start.Add(10 * time.Minute) + return start.Format("15:04") + "-" + end.Format("15:04") +} + +func (a *Auth) recordRecentRequest(now time.Time, success bool) { + if a == nil { + return + } + bucketID := recentRequestBucketID(now) + idx := recentRequestBucketIndex(bucketID) + bucket := &a.recentRequests.buckets[idx] + if bucket.bucketID != bucketID { + bucket.bucketID = bucketID + bucket.success = 0 + bucket.failed = 0 + } + if success { + bucket.success++ + return + } + bucket.failed++ +} + +func (a *Auth) RecentRequestsSnapshot(now time.Time) []RecentRequestBucket { + out := make([]RecentRequestBucket, 0, recentRequestBucketCount) + if a == nil { + return out + } + + currentBucketID := recentRequestBucketID(now) + for i := recentRequestBucketCount - 1; i >= 0; i-- { + bucketID := currentBucketID - int64(i) + idx := recentRequestBucketIndex(bucketID) + bucket := a.recentRequests.buckets[idx] + entry := RecentRequestBucket{ + Time: formatRecentRequestBucketLabel(bucketID), + } + if bucket.bucketID == bucketID { + entry.Success = bucket.success + entry.Failed = bucket.failed + } + out = append(out, entry) + } + + return out +} + // Clone shallow copies the Auth structure, duplicating maps to avoid accidental mutation. func (a *Auth) Clone() *Auth { if a == nil { diff --git a/sdk/cliproxy/auth/types_test.go b/sdk/cliproxy/auth/types_test.go index e7029385..06836da1 100644 --- a/sdk/cliproxy/auth/types_test.go +++ b/sdk/cliproxy/auth/types_test.go @@ -1,6 +1,10 @@ package auth -import "testing" +import ( + "strings" + "testing" + "time" +) func TestToolPrefixDisabled(t *testing.T) { var a *Auth @@ -96,3 +100,72 @@ func TestEnsureIndexUsesCredentialIdentity(t *testing.T) { t.Fatalf("duplicate config entries should be separated by source-derived seed, got %q", geminiIndex) } } + +func TestRecentRequestsSnapshotEmptyReturnsTwentyBuckets(t *testing.T) { + now := time.Unix(1_700_000_000, 0).In(time.Local) + a := &Auth{} + + got := a.RecentRequestsSnapshot(now) + if len(got) != recentRequestBucketCount { + t.Fatalf("len = %d, want %d", len(got), recentRequestBucketCount) + } + + currentBucketID := now.Unix() / recentRequestBucketSeconds + baseBucketID := currentBucketID - int64(recentRequestBucketCount-1) + for i, bucket := range got { + if bucket.Success != 0 || bucket.Failed != 0 { + t.Fatalf("bucket[%d] counts = %d/%d, want 0/0", i, bucket.Success, bucket.Failed) + } + if strings.TrimSpace(bucket.Time) == "" { + t.Fatalf("bucket[%d] time label is empty", i) + } + expectedBucketID := baseBucketID + int64(i) + start := time.Unix(expectedBucketID*recentRequestBucketSeconds, 0).In(time.Local) + end := start.Add(10 * time.Minute) + expected := start.Format("15:04") + "-" + end.Format("15:04") + if bucket.Time != expected { + t.Fatalf("bucket[%d] time = %q, want %q", i, bucket.Time, expected) + } + } +} + +func TestRecentRequestsSnapshotIncludesCounts(t *testing.T) { + now := time.Unix(1_700_000_000, 0).In(time.Local) + a := &Auth{} + + a.recordRecentRequest(now, true) + a.recordRecentRequest(now, false) + + got := a.RecentRequestsSnapshot(now) + if len(got) != recentRequestBucketCount { + t.Fatalf("len = %d, want %d", len(got), recentRequestBucketCount) + } + + newest := got[len(got)-1] + if newest.Success != 1 || newest.Failed != 1 { + t.Fatalf("newest bucket = success=%d failed=%d, want 1/1", newest.Success, newest.Failed) + } +} + +func TestRecentRequestsSnapshotBucketAdvanceMovesCounts(t *testing.T) { + now := time.Unix(1_700_000_000, 0).In(time.Local) + next := now.Add(10 * time.Minute) + a := &Auth{} + + a.recordRecentRequest(now, true) + a.recordRecentRequest(next, false) + + got := a.RecentRequestsSnapshot(next) + if len(got) != recentRequestBucketCount { + t.Fatalf("len = %d, want %d", len(got), recentRequestBucketCount) + } + + secondNewest := got[len(got)-2] + newest := got[len(got)-1] + if secondNewest.Success != 1 || secondNewest.Failed != 0 { + t.Fatalf("second newest bucket = success=%d failed=%d, want 1/0", secondNewest.Success, secondNewest.Failed) + } + if newest.Success != 0 || newest.Failed != 1 { + t.Fatalf("newest bucket = success=%d failed=%d, want 0/1", newest.Success, newest.Failed) + } +} From b0dc9df887ef9f8fd9fad5bd9e4ebd639d6de8f3 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Fri, 1 May 2026 23:34:18 +0800 Subject: [PATCH 18/51] feat: add API key usage endpoint with provider and key grouping - Implemented `GetAPIKeyUsage` to expose recent request data grouped by provider and API key. - Added supporting function `mergeRecentRequestBuckets` for bucket aggregation. - Registered new endpoint `/v0/management/api-key-usage` in the management API. - Included extensive unit tests for provider and key-based grouping validation. - Updated `formatRecentRequestBucketLabel` to support configurable bucket duration. --- .../api/handlers/management/api_key_usage.go | 86 ++++++++++++++++++ .../handlers/management/api_key_usage_test.go | 87 +++++++++++++++++++ internal/api/server.go | 1 + sdk/cliproxy/auth/types.go | 2 +- 4 files changed, 175 insertions(+), 1 deletion(-) create mode 100644 internal/api/handlers/management/api_key_usage.go create mode 100644 internal/api/handlers/management/api_key_usage_test.go diff --git a/internal/api/handlers/management/api_key_usage.go b/internal/api/handlers/management/api_key_usage.go new file mode 100644 index 00000000..599fbad9 --- /dev/null +++ b/internal/api/handlers/management/api_key_usage.go @@ -0,0 +1,86 @@ +package management + +import ( + "net/http" + "strings" + "time" + + "github.com/gin-gonic/gin" + coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" +) + +func mergeRecentRequestBuckets(dst, src []coreauth.RecentRequestBucket) []coreauth.RecentRequestBucket { + if len(dst) == 0 { + return src + } + if len(src) == 0 { + return dst + } + if len(dst) != len(src) { + n := len(dst) + if len(src) < n { + n = len(src) + } + for i := 0; i < n; i++ { + dst[i].Success += src[i].Success + dst[i].Failed += src[i].Failed + } + return dst + } + for i := range dst { + dst[i].Success += src[i].Success + dst[i].Failed += src[i].Failed + } + return dst +} + +// GetAPIKeyUsage returns recent request buckets for all in-memory api_key auths, +// grouped by provider and keyed by the raw api-key value. +func (h *Handler) GetAPIKeyUsage(c *gin.Context) { + if h == nil { + c.JSON(http.StatusInternalServerError, gin.H{"error": "handler not initialized"}) + return + } + + h.mu.Lock() + manager := h.authManager + h.mu.Unlock() + if manager == nil { + c.JSON(http.StatusServiceUnavailable, gin.H{"error": "core auth manager unavailable"}) + return + } + + now := time.Now() + out := make(map[string]map[string][]coreauth.RecentRequestBucket) + for _, auth := range manager.List() { + if auth == nil { + continue + } + kind, apiKey := auth.AccountInfo() + if !strings.EqualFold(strings.TrimSpace(kind), "api_key") { + continue + } + apiKey = strings.TrimSpace(apiKey) + if apiKey == "" { + continue + } + provider := strings.ToLower(strings.TrimSpace(auth.Provider)) + if provider == "" { + provider = "unknown" + } + + recent := auth.RecentRequestsSnapshot(now) + providerBucket, ok := out[provider] + if !ok { + providerBucket = make(map[string][]coreauth.RecentRequestBucket) + out[provider] = providerBucket + } + if existing, exists := providerBucket[apiKey]; exists { + providerBucket[apiKey] = mergeRecentRequestBuckets(existing, recent) + continue + } + providerBucket[apiKey] = recent + } + + c.JSON(http.StatusOK, out) +} diff --git a/internal/api/handlers/management/api_key_usage_test.go b/internal/api/handlers/management/api_key_usage_test.go new file mode 100644 index 00000000..230dca4a --- /dev/null +++ b/internal/api/handlers/management/api_key_usage_test.go @@ -0,0 +1,87 @@ +package management + +import ( + "context" + "encoding/json" + "net/http" + "net/http/httptest" + "testing" + + "github.com/gin-gonic/gin" + "github.com/router-for-me/CLIProxyAPI/v6/internal/config" + coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" +) + +func sumRecentRequestBuckets(buckets []coreauth.RecentRequestBucket) (int64, int64) { + var success int64 + var failed int64 + for _, bucket := range buckets { + success += bucket.Success + failed += bucket.Failed + } + return success, failed +} + +func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { + t.Setenv("MANAGEMENT_PASSWORD", "") + gin.SetMode(gin.TestMode) + + manager := coreauth.NewManager(nil, nil, nil) + if _, err := manager.Register(context.Background(), &coreauth.Auth{ + ID: "codex-auth", + Provider: "codex", + Attributes: map[string]string{ + "api_key": "codex-key", + }, + }); err != nil { + t.Fatalf("register codex auth: %v", err) + } + if _, err := manager.Register(context.Background(), &coreauth.Auth{ + ID: "claude-auth", + Provider: "claude", + Attributes: map[string]string{ + "api_key": "claude-key", + }, + }); err != nil { + t.Fatalf("register claude auth: %v", err) + } + + manager.MarkResult(context.Background(), coreauth.Result{AuthID: "codex-auth", Provider: "codex", Model: "gpt-5", Success: true}) + manager.MarkResult(context.Background(), coreauth.Result{AuthID: "codex-auth", Provider: "codex", Model: "gpt-5", Success: false}) + manager.MarkResult(context.Background(), coreauth.Result{AuthID: "claude-auth", Provider: "claude", Model: "claude-4", Success: true}) + + h := NewHandlerWithoutConfigFilePath(&config.Config{AuthDir: t.TempDir()}, manager) + + rec := httptest.NewRecorder() + ginCtx, _ := gin.CreateTestContext(rec) + req := httptest.NewRequest(http.MethodGet, "/v0/management/api-key-usage", nil) + ginCtx.Request = req + h.GetAPIKeyUsage(ginCtx) + + if rec.Code != http.StatusOK { + t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusOK, rec.Body.String()) + } + + var payload map[string]map[string][]coreauth.RecentRequestBucket + if err := json.Unmarshal(rec.Body.Bytes(), &payload); err != nil { + t.Fatalf("decode payload: %v", err) + } + + codexBuckets := payload["codex"]["codex-key"] + if len(codexBuckets) != 20 { + t.Fatalf("codex buckets len = %d, want 20", len(codexBuckets)) + } + codexSuccess, codexFailed := sumRecentRequestBuckets(codexBuckets) + if codexSuccess != 1 || codexFailed != 1 { + t.Fatalf("codex totals = %d/%d, want 1/1", codexSuccess, codexFailed) + } + + claudeBuckets := payload["claude"]["claude-key"] + if len(claudeBuckets) != 20 { + t.Fatalf("claude buckets len = %d, want 20", len(claudeBuckets)) + } + claudeSuccess, claudeFailed := sumRecentRequestBuckets(claudeBuckets) + if claudeSuccess != 1 || claudeFailed != 0 { + t.Fatalf("claude totals = %d/%d, want 1/0", claudeSuccess, claudeFailed) + } +} diff --git a/internal/api/server.go b/internal/api/server.go index 8421357b..4d51460d 100644 --- a/internal/api/server.go +++ b/internal/api/server.go @@ -554,6 +554,7 @@ func (s *Server) registerManagementRoutes() { mgmt.PUT("/api-keys", s.mgmt.PutAPIKeys) mgmt.PATCH("/api-keys", s.mgmt.PatchAPIKeys) mgmt.DELETE("/api-keys", s.mgmt.DeleteAPIKeys) + mgmt.GET("/api-key-usage", s.mgmt.GetAPIKeyUsage) mgmt.GET("/gemini-api-key", s.mgmt.GetGeminiKeys) mgmt.PUT("/gemini-api-key", s.mgmt.PutGeminiKeys) diff --git a/sdk/cliproxy/auth/types.go b/sdk/cliproxy/auth/types.go index 93dd3881..4a394ad4 100644 --- a/sdk/cliproxy/auth/types.go +++ b/sdk/cliproxy/auth/types.go @@ -164,7 +164,7 @@ func recentRequestBucketIndex(bucketID int64) int { func formatRecentRequestBucketLabel(bucketID int64) string { start := time.Unix(bucketID*recentRequestBucketSeconds, 0).In(time.Local) - end := start.Add(10 * time.Minute) + end := start.Add(time.Duration(recentRequestBucketSeconds) * time.Second) return start.Format("15:04") + "-" + end.Format("15:04") } From e37f3be0bfc482934d9669b58df1562e59f1196e Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 00:09:08 +0800 Subject: [PATCH 19/51] chore: update .goreleaser.yml to include custom archive naming with arch override logic --- .goreleaser.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.goreleaser.yml b/.goreleaser.yml index f8bebfc1..c479255e 100644 --- a/.goreleaser.yml +++ b/.goreleaser.yml @@ -19,6 +19,8 @@ builds: archives: - id: "cli-proxy-api" format: tar.gz + name_template: >- + {{ .ProjectName }}_{{ .Version }}_{{ .Os }}_{{- if eq .Arch "arm64" -}}aarch64{{- else -}}{{ .Arch }}{{- end -}} format_overrides: - goos: windows format: zip From 8c2f1a80d39e542d3d85d569d035d7df8d5c39f6 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 02:20:49 +0800 Subject: [PATCH 20/51] feat: enhance API key usage grouping with base URL inclusion - Updated `GetAPIKeyUsage` to group API key usage by "base_url|api_key" composite keys. - Adjusted logic to handle `base_url` extraction from auth attributes. - Revised unit tests to validate "base_url|api_key" grouping behavior. --- .../api/handlers/management/api_key_usage.go | 16 ++++++++++++---- .../handlers/management/api_key_usage_test.go | 10 ++++++---- 2 files changed, 18 insertions(+), 8 deletions(-) diff --git a/internal/api/handlers/management/api_key_usage.go b/internal/api/handlers/management/api_key_usage.go index 599fbad9..76b32bbb 100644 --- a/internal/api/handlers/management/api_key_usage.go +++ b/internal/api/handlers/management/api_key_usage.go @@ -35,7 +35,7 @@ func mergeRecentRequestBuckets(dst, src []coreauth.RecentRequestBucket) []coreau } // GetAPIKeyUsage returns recent request buckets for all in-memory api_key auths, -// grouped by provider and keyed by the raw api-key value. +// grouped by provider and keyed by "base_url|api_key". func (h *Handler) GetAPIKeyUsage(c *gin.Context) { if h == nil { c.JSON(http.StatusInternalServerError, gin.H{"error": "handler not initialized"}) @@ -64,6 +64,14 @@ func (h *Handler) GetAPIKeyUsage(c *gin.Context) { if apiKey == "" { continue } + baseURL := "" + if auth.Attributes != nil { + baseURL = strings.TrimSpace(auth.Attributes["base_url"]) + if baseURL == "" { + baseURL = strings.TrimSpace(auth.Attributes["base-url"]) + } + } + compositeKey := baseURL + "|" + apiKey provider := strings.ToLower(strings.TrimSpace(auth.Provider)) if provider == "" { provider = "unknown" @@ -75,11 +83,11 @@ func (h *Handler) GetAPIKeyUsage(c *gin.Context) { providerBucket = make(map[string][]coreauth.RecentRequestBucket) out[provider] = providerBucket } - if existing, exists := providerBucket[apiKey]; exists { - providerBucket[apiKey] = mergeRecentRequestBuckets(existing, recent) + if existing, exists := providerBucket[compositeKey]; exists { + providerBucket[compositeKey] = mergeRecentRequestBuckets(existing, recent) continue } - providerBucket[apiKey] = recent + providerBucket[compositeKey] = recent } c.JSON(http.StatusOK, out) diff --git a/internal/api/handlers/management/api_key_usage_test.go b/internal/api/handlers/management/api_key_usage_test.go index 230dca4a..56617161 100644 --- a/internal/api/handlers/management/api_key_usage_test.go +++ b/internal/api/handlers/management/api_key_usage_test.go @@ -31,7 +31,8 @@ func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { ID: "codex-auth", Provider: "codex", Attributes: map[string]string{ - "api_key": "codex-key", + "api_key": "codex-key", + "base_url": "https://codex.example.com", }, }); err != nil { t.Fatalf("register codex auth: %v", err) @@ -40,7 +41,8 @@ func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { ID: "claude-auth", Provider: "claude", Attributes: map[string]string{ - "api_key": "claude-key", + "api_key": "claude-key", + "base_url": "https://claude.example.com", }, }); err != nil { t.Fatalf("register claude auth: %v", err) @@ -67,7 +69,7 @@ func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { t.Fatalf("decode payload: %v", err) } - codexBuckets := payload["codex"]["codex-key"] + codexBuckets := payload["codex"]["https://codex.example.com|codex-key"] if len(codexBuckets) != 20 { t.Fatalf("codex buckets len = %d, want 20", len(codexBuckets)) } @@ -76,7 +78,7 @@ func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { t.Fatalf("codex totals = %d/%d, want 1/1", codexSuccess, codexFailed) } - claudeBuckets := payload["claude"]["claude-key"] + claudeBuckets := payload["claude"]["https://claude.example.com|claude-key"] if len(claudeBuckets) != 20 { t.Fatalf("claude buckets len = %d, want 20", len(claudeBuckets)) } From b8bba053fcdafd80abc2152c88c78f4e7713c05a Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 03:40:00 +0800 Subject: [PATCH 21/51] feat: add tracking for auth request success and failure counts - Introduced `Success` and `Failed` fields in auth records to track request outcomes. - Updated `/v0/management/auth-files` and `/v0/management/api-key-usage` responses to include success and failure counts. - Enhanced tests to validate tracking logic and API responses. --- .../api/handlers/management/api_key_usage.go | 21 ++++++-- .../handlers/management/api_key_usage_test.go | 24 +++++---- .../api/handlers/management/auth_files.go | 2 + .../auth_files_recent_requests_test.go | 7 +++ sdk/cliproxy/auth/conductor.go | 8 +++ .../auth/conductor_recent_requests_test.go | 51 +++++++++++++++++++ sdk/cliproxy/auth/types.go | 3 ++ 7 files changed, 103 insertions(+), 13 deletions(-) diff --git a/internal/api/handlers/management/api_key_usage.go b/internal/api/handlers/management/api_key_usage.go index 76b32bbb..3361da5d 100644 --- a/internal/api/handlers/management/api_key_usage.go +++ b/internal/api/handlers/management/api_key_usage.go @@ -9,6 +9,12 @@ import ( coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" ) +type apiKeyUsageEntry struct { + Success int64 `json:"success"` + Failed int64 `json:"failed"` + RecentRequests []coreauth.RecentRequestBucket `json:"recent_requests"` +} + func mergeRecentRequestBuckets(dst, src []coreauth.RecentRequestBucket) []coreauth.RecentRequestBucket { if len(dst) == 0 { return src @@ -51,7 +57,7 @@ func (h *Handler) GetAPIKeyUsage(c *gin.Context) { } now := time.Now() - out := make(map[string]map[string][]coreauth.RecentRequestBucket) + out := make(map[string]map[string]apiKeyUsageEntry) for _, auth := range manager.List() { if auth == nil { continue @@ -80,14 +86,21 @@ func (h *Handler) GetAPIKeyUsage(c *gin.Context) { recent := auth.RecentRequestsSnapshot(now) providerBucket, ok := out[provider] if !ok { - providerBucket = make(map[string][]coreauth.RecentRequestBucket) + providerBucket = make(map[string]apiKeyUsageEntry) out[provider] = providerBucket } if existing, exists := providerBucket[compositeKey]; exists { - providerBucket[compositeKey] = mergeRecentRequestBuckets(existing, recent) + existing.Success += auth.Success + existing.Failed += auth.Failed + existing.RecentRequests = mergeRecentRequestBuckets(existing.RecentRequests, recent) + providerBucket[compositeKey] = existing continue } - providerBucket[compositeKey] = recent + providerBucket[compositeKey] = apiKeyUsageEntry{ + Success: auth.Success, + Failed: auth.Failed, + RecentRequests: recent, + } } c.JSON(http.StatusOK, out) diff --git a/internal/api/handlers/management/api_key_usage_test.go b/internal/api/handlers/management/api_key_usage_test.go index 56617161..2880567f 100644 --- a/internal/api/handlers/management/api_key_usage_test.go +++ b/internal/api/handlers/management/api_key_usage_test.go @@ -64,25 +64,31 @@ func TestGetAPIKeyUsage_GroupsByProviderAndAPIKey(t *testing.T) { t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusOK, rec.Body.String()) } - var payload map[string]map[string][]coreauth.RecentRequestBucket + var payload map[string]map[string]apiKeyUsageEntry if err := json.Unmarshal(rec.Body.Bytes(), &payload); err != nil { t.Fatalf("decode payload: %v", err) } - codexBuckets := payload["codex"]["https://codex.example.com|codex-key"] - if len(codexBuckets) != 20 { - t.Fatalf("codex buckets len = %d, want 20", len(codexBuckets)) + codexEntry := payload["codex"]["https://codex.example.com|codex-key"] + if codexEntry.Success != 1 || codexEntry.Failed != 1 { + t.Fatalf("codex totals = %d/%d, want 1/1", codexEntry.Success, codexEntry.Failed) } - codexSuccess, codexFailed := sumRecentRequestBuckets(codexBuckets) + if len(codexEntry.RecentRequests) != 20 { + t.Fatalf("codex buckets len = %d, want 20", len(codexEntry.RecentRequests)) + } + codexSuccess, codexFailed := sumRecentRequestBuckets(codexEntry.RecentRequests) if codexSuccess != 1 || codexFailed != 1 { t.Fatalf("codex totals = %d/%d, want 1/1", codexSuccess, codexFailed) } - claudeBuckets := payload["claude"]["https://claude.example.com|claude-key"] - if len(claudeBuckets) != 20 { - t.Fatalf("claude buckets len = %d, want 20", len(claudeBuckets)) + claudeEntry := payload["claude"]["https://claude.example.com|claude-key"] + if claudeEntry.Success != 1 || claudeEntry.Failed != 0 { + t.Fatalf("claude totals = %d/%d, want 1/0", claudeEntry.Success, claudeEntry.Failed) } - claudeSuccess, claudeFailed := sumRecentRequestBuckets(claudeBuckets) + if len(claudeEntry.RecentRequests) != 20 { + t.Fatalf("claude buckets len = %d, want 20", len(claudeEntry.RecentRequests)) + } + claudeSuccess, claudeFailed := sumRecentRequestBuckets(claudeEntry.RecentRequests) if claudeSuccess != 1 || claudeFailed != 0 { t.Fatalf("claude totals = %d/%d, want 1/0", claudeSuccess, claudeFailed) } diff --git a/internal/api/handlers/management/auth_files.go b/internal/api/handlers/management/auth_files.go index 2bcfaac4..bb94daa9 100644 --- a/internal/api/handlers/management/auth_files.go +++ b/internal/api/handlers/management/auth_files.go @@ -388,6 +388,8 @@ func (h *Handler) buildAuthFileEntry(auth *coreauth.Auth) gin.H { "source": "memory", "size": int64(0), } + entry["success"] = auth.Success + entry["failed"] = auth.Failed entry["recent_requests"] = auth.RecentRequestsSnapshot(time.Now()) if email := authEmail(auth); email != "" { entry["email"] = email diff --git a/internal/api/handlers/management/auth_files_recent_requests_test.go b/internal/api/handlers/management/auth_files_recent_requests_test.go index fd28ca1d..979040f5 100644 --- a/internal/api/handlers/management/auth_files_recent_requests_test.go +++ b/internal/api/handlers/management/auth_files_recent_requests_test.go @@ -62,6 +62,13 @@ func TestListAuthFiles_IncludesRecentRequestsBuckets(t *testing.T) { t.Fatalf("expected file entry object, got %#v", filesRaw[0]) } + if _, ok := fileEntry["success"].(float64); !ok { + t.Fatalf("expected success number, got %#v", fileEntry["success"]) + } + if _, ok := fileEntry["failed"].(float64); !ok { + t.Fatalf("expected failed number, got %#v", fileEntry["failed"]) + } + recentRaw, ok := fileEntry["recent_requests"].([]any) if !ok { t.Fatalf("expected recent_requests array, got %#v", fileEntry["recent_requests"]) diff --git a/sdk/cliproxy/auth/conductor.go b/sdk/cliproxy/auth/conductor.go index 61a0e413..d2a3db18 100644 --- a/sdk/cliproxy/auth/conductor.go +++ b/sdk/cliproxy/auth/conductor.go @@ -1126,6 +1126,9 @@ func (m *Manager) Update(ctx context.Context, auth *Auth) (*Auth, error) { auth.Index = existing.Index auth.indexAssigned = existing.indexAssigned } + auth.Success = existing.Success + auth.Failed = existing.Failed + auth.recentRequests = existing.recentRequests if !existing.Disabled && existing.Status != StatusDisabled && !auth.Disabled && auth.Status != StatusDisabled { if len(auth.ModelStates) == 0 && len(existing.ModelStates) > 0 { auth.ModelStates = existing.ModelStates @@ -2022,6 +2025,11 @@ func (m *Manager) MarkResult(ctx context.Context, result Result) { if auth, ok := m.auths[result.AuthID]; ok && auth != nil { now := time.Now() auth.recordRecentRequest(now, result.Success) + if result.Success { + auth.Success++ + } else { + auth.Failed++ + } if result.Success { if result.Model != "" { diff --git a/sdk/cliproxy/auth/conductor_recent_requests_test.go b/sdk/cliproxy/auth/conductor_recent_requests_test.go index 3f5a7212..d2003b7c 100644 --- a/sdk/cliproxy/auth/conductor_recent_requests_test.go +++ b/sdk/cliproxy/auth/conductor_recent_requests_test.go @@ -31,6 +31,10 @@ func TestManagerMarkResultRecordsRecentRequests(t *testing.T) { t.Fatalf("GetByID returned ok=%v auth=%v", ok, gotAuth) } + if gotAuth.Success != 1 || gotAuth.Failed != 1 { + t.Fatalf("auth totals = success=%d failed=%d, want 1/1", gotAuth.Success, gotAuth.Failed) + } + snapshot := gotAuth.RecentRequestsSnapshot(time.Now()) var successTotal int64 var failedTotal int64 @@ -42,3 +46,50 @@ func TestManagerMarkResultRecordsRecentRequests(t *testing.T) { t.Fatalf("totals = success=%d failed=%d, want 1/1", successTotal, failedTotal) } } + +func TestManagerUpdatePreservesRecentRequestsAndTotals(t *testing.T) { + mgr := NewManager(nil, nil, nil) + auth := &Auth{ + ID: "auth-1", + Provider: "antigravity", + Metadata: map[string]any{ + "type": "antigravity", + }, + } + if _, err := mgr.Register(WithSkipPersist(context.Background()), auth); err != nil { + t.Fatalf("Register returned error: %v", err) + } + + mgr.MarkResult(context.Background(), Result{AuthID: "auth-1", Provider: "antigravity", Model: "gpt-5", Success: true}) + + updated := &Auth{ + ID: "auth-1", + Provider: "antigravity", + Metadata: map[string]any{ + "type": "antigravity", + "note": "updated", + }, + } + if _, err := mgr.Update(WithSkipPersist(context.Background()), updated); err != nil { + t.Fatalf("Update returned error: %v", err) + } + + gotAuth, ok := mgr.GetByID("auth-1") + if !ok || gotAuth == nil { + t.Fatalf("GetByID returned ok=%v auth=%v", ok, gotAuth) + } + if gotAuth.Success != 1 || gotAuth.Failed != 0 { + t.Fatalf("auth totals = success=%d failed=%d, want 1/0", gotAuth.Success, gotAuth.Failed) + } + + snapshot := gotAuth.RecentRequestsSnapshot(time.Now()) + var successTotal int64 + var failedTotal int64 + for _, bucket := range snapshot { + successTotal += bucket.Success + failedTotal += bucket.Failed + } + if successTotal != 1 || failedTotal != 0 { + t.Fatalf("bucket totals = success=%d failed=%d, want 1/0", successTotal, failedTotal) + } +} diff --git a/sdk/cliproxy/auth/types.go b/sdk/cliproxy/auth/types.go index 4a394ad4..76f4c396 100644 --- a/sdk/cliproxy/auth/types.go +++ b/sdk/cliproxy/auth/types.go @@ -92,6 +92,9 @@ type Auth struct { // Runtime carries non-serialisable data used during execution (in-memory only). Runtime any `json:"-"` + Success int64 `json:"-"` + Failed int64 `json:"-"` + recentRequests recentRequestRing `json:"-"` indexAssigned bool `json:"-"` } From 18bb9c315fced2c428f57b4a0e66b06183c46c06 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 04:50:58 +0800 Subject: [PATCH 22/51] chore: remove usage tracking and logging functionality - Deleted the `LoggerPlugin` along with associated usage tracking and in-memory statistics logic. - Removed all related tests (`logger_plugin_test.go`, `usage_tab_test.go`) and external-facing handler (`usage.go`) for usage statistics export/import. - Cleaned up TUI integration by deleting `usage_tab.go`. --- cmd/server/main.go | 4 +- docker-build.sh | 136 +----- internal/api/handlers/management/handler.go | 6 - internal/api/handlers/management/usage.go | 79 ---- internal/api/server.go | 6 +- internal/redisqueue/plugin.go | 28 +- internal/redisqueue/plugin_test.go | 7 +- internal/redisqueue/usage_toggle.go | 16 + internal/tui/app.go | 22 +- internal/tui/client.go | 5 - internal/tui/dashboard.go | 77 +--- internal/tui/i18n.go | 4 +- internal/tui/usage_tab.go | 418 ------------------ internal/tui/usage_tab_test.go | 134 ------ internal/usage/logger_plugin.go | 464 -------------------- internal/usage/logger_plugin_test.go | 96 ---- sdk/cliproxy/service.go | 1 - test/usage_logging_test.go | 83 ++-- 18 files changed, 116 insertions(+), 1470 deletions(-) delete mode 100644 internal/api/handlers/management/usage.go create mode 100644 internal/redisqueue/usage_toggle.go delete mode 100644 internal/tui/usage_tab.go delete mode 100644 internal/tui/usage_tab_test.go delete mode 100644 internal/usage/logger_plugin.go delete mode 100644 internal/usage/logger_plugin_test.go diff --git a/cmd/server/main.go b/cmd/server/main.go index b8707f0a..e735b144 100644 --- a/cmd/server/main.go +++ b/cmd/server/main.go @@ -24,11 +24,11 @@ import ( "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" "github.com/router-for-me/CLIProxyAPI/v6/internal/managementasset" "github.com/router-for-me/CLIProxyAPI/v6/internal/misc" + "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" "github.com/router-for-me/CLIProxyAPI/v6/internal/registry" "github.com/router-for-me/CLIProxyAPI/v6/internal/store" _ "github.com/router-for-me/CLIProxyAPI/v6/internal/translator" "github.com/router-for-me/CLIProxyAPI/v6/internal/tui" - "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" "github.com/router-for-me/CLIProxyAPI/v6/internal/util" sdkAuth "github.com/router-for-me/CLIProxyAPI/v6/sdk/auth" coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" @@ -417,7 +417,7 @@ func main() { configFileExists = true } } - usage.SetStatisticsEnabled(cfg.UsageStatisticsEnabled) + redisqueue.SetUsageStatisticsEnabled(cfg.UsageStatisticsEnabled) coreauth.SetQuotaCooldownDisabled(cfg.DisableCooling) if err = logging.ConfigureLogOutput(cfg); err != nil { diff --git a/docker-build.sh b/docker-build.sh index 4538b807..ebe7d923 100644 --- a/docker-build.sh +++ b/docker-build.sh @@ -5,123 +5,13 @@ # This script automates the process of building and running the Docker container # with version information dynamically injected at build time. -# Hidden feature: Preserve usage statistics across rebuilds -# Usage: ./docker-build.sh --with-usage -# First run prompts for management API key, saved to temp/stats/.api_secret - set -euo pipefail -STATS_DIR="temp/stats" -STATS_FILE="${STATS_DIR}/.usage_backup.json" -SECRET_FILE="${STATS_DIR}/.api_secret" -WITH_USAGE=false - -get_port() { - if [[ -f "config.yaml" ]]; then - grep -E "^port:" config.yaml | sed -E 's/^port: *["'"'"']?([0-9]+)["'"'"']?.*$/\1/' - else - echo "8317" - fi -} - -export_stats_api_secret() { - if [[ -f "${SECRET_FILE}" ]]; then - API_SECRET=$(cat "${SECRET_FILE}") - else - if [[ ! -d "${STATS_DIR}" ]]; then - mkdir -p "${STATS_DIR}" - fi - echo "First time using --with-usage. Management API key required." - read -r -p "Enter management key: " -s API_SECRET - echo - echo "${API_SECRET}" > "${SECRET_FILE}" - chmod 600 "${SECRET_FILE}" - fi -} - -check_container_running() { - local port - port=$(get_port) - - if ! curl -s -o /dev/null -w "%{http_code}" "http://localhost:${port}/" | grep -q "200"; then - echo "Error: cli-proxy-api service is not responding at localhost:${port}" - echo "Please start the container first or use without --with-usage flag." - exit 1 - fi -} - -export_stats() { - local port - port=$(get_port) - - if [[ ! -d "${STATS_DIR}" ]]; then - mkdir -p "${STATS_DIR}" - fi - check_container_running - echo "Exporting usage statistics..." - EXPORT_RESPONSE=$(curl -s -w "\n%{http_code}" -H "X-Management-Key: ${API_SECRET}" \ - "http://localhost:${port}/v0/management/usage/export") - HTTP_CODE=$(echo "${EXPORT_RESPONSE}" | tail -n1) - RESPONSE_BODY=$(echo "${EXPORT_RESPONSE}" | sed '$d') - - if [[ "${HTTP_CODE}" != "200" ]]; then - echo "Export failed (HTTP ${HTTP_CODE}): ${RESPONSE_BODY}" - exit 1 - fi - - echo "${RESPONSE_BODY}" > "${STATS_FILE}" - echo "Statistics exported to ${STATS_FILE}" -} - -import_stats() { - local port - port=$(get_port) - - echo "Importing usage statistics..." - IMPORT_RESPONSE=$(curl -s -w "\n%{http_code}" -X POST \ - -H "X-Management-Key: ${API_SECRET}" \ - -H "Content-Type: application/json" \ - -d @"${STATS_FILE}" \ - "http://localhost:${port}/v0/management/usage/import") - IMPORT_CODE=$(echo "${IMPORT_RESPONSE}" | tail -n1) - IMPORT_BODY=$(echo "${IMPORT_RESPONSE}" | sed '$d') - - if [[ "${IMPORT_CODE}" == "200" ]]; then - echo "Statistics imported successfully" - else - echo "Import failed (HTTP ${IMPORT_CODE}): ${IMPORT_BODY}" - fi - - rm -f "${STATS_FILE}" -} - -wait_for_service() { - local port - port=$(get_port) - - echo "Waiting for service to be ready..." - for i in {1..30}; do - if curl -s -o /dev/null -w "%{http_code}" "http://localhost:${port}/" | grep -q "200"; then - break - fi - sleep 1 - done - sleep 2 -} - -case "${1:-}" in - "") - ;; - "--with-usage") - WITH_USAGE=true - export_stats_api_secret - ;; - *) - echo "Error: unknown option '${1}'. Did you mean '--with-usage'?" - echo "Usage: ./docker-build.sh [--with-usage]" - exit 1 - ;; -esac +if [[ "${1:-}" != "" ]]; then + echo "Error: unknown option '${1}'." + echo "Usage: ./docker-build.sh" + exit 1 +fi # --- Step 1: Choose Environment --- echo "Please select an option:" @@ -133,14 +23,7 @@ read -r -p "Enter choice [1-2]: " choice case "$choice" in 1) echo "--- Running with Pre-built Image ---" - if [[ "${WITH_USAGE}" == "true" ]]; then - export_stats - fi docker compose up -d --remove-orphans --no-build - if [[ "${WITH_USAGE}" == "true" ]]; then - wait_for_service - import_stats - fi echo "Services are starting from remote image." echo "Run 'docker compose logs -f' to see the logs." ;; @@ -167,18 +50,9 @@ case "$choice" in --build-arg COMMIT="${COMMIT}" \ --build-arg BUILD_DATE="${BUILD_DATE}" - if [[ "${WITH_USAGE}" == "true" ]]; then - export_stats - fi - echo "Starting the services..." docker compose up -d --remove-orphans --pull never - if [[ "${WITH_USAGE}" == "true" ]]; then - wait_for_service - import_stats - fi - echo "Build complete. Services are starting." echo "Run 'docker compose logs -f' to see the logs." ;; diff --git a/internal/api/handlers/management/handler.go b/internal/api/handlers/management/handler.go index af11366c..9abc8a5c 100644 --- a/internal/api/handlers/management/handler.go +++ b/internal/api/handlers/management/handler.go @@ -15,7 +15,6 @@ import ( "github.com/gin-gonic/gin" "github.com/router-for-me/CLIProxyAPI/v6/internal/buildinfo" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" - "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" sdkAuth "github.com/router-for-me/CLIProxyAPI/v6/sdk/auth" coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" "golang.org/x/crypto/bcrypt" @@ -41,7 +40,6 @@ type Handler struct { attemptsMu sync.Mutex failedAttempts map[string]*attemptInfo // keyed by client IP authManager *coreauth.Manager - usageStats *usage.RequestStatistics tokenStore coreauth.Store localPassword string allowRemoteOverride bool @@ -60,7 +58,6 @@ func NewHandler(cfg *config.Config, configFilePath string, manager *coreauth.Man configFilePath: configFilePath, failedAttempts: make(map[string]*attemptInfo), authManager: manager, - usageStats: usage.GetRequestStatistics(), tokenStore: sdkAuth.GetTokenStore(), allowRemoteOverride: envSecret != "", envSecret: envSecret, @@ -124,9 +121,6 @@ func (h *Handler) SetAuthManager(manager *coreauth.Manager) { h.mu.Unlock() } -// SetUsageStatistics allows replacing the usage statistics reference. -func (h *Handler) SetUsageStatistics(stats *usage.RequestStatistics) { h.usageStats = stats } - // SetLocalPassword configures the runtime-local password accepted for localhost requests. func (h *Handler) SetLocalPassword(password string) { h.localPassword = password } diff --git a/internal/api/handlers/management/usage.go b/internal/api/handlers/management/usage.go deleted file mode 100644 index 5f794089..00000000 --- a/internal/api/handlers/management/usage.go +++ /dev/null @@ -1,79 +0,0 @@ -package management - -import ( - "encoding/json" - "net/http" - "time" - - "github.com/gin-gonic/gin" - "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" -) - -type usageExportPayload struct { - Version int `json:"version"` - ExportedAt time.Time `json:"exported_at"` - Usage usage.StatisticsSnapshot `json:"usage"` -} - -type usageImportPayload struct { - Version int `json:"version"` - Usage usage.StatisticsSnapshot `json:"usage"` -} - -// GetUsageStatistics returns the in-memory request statistics snapshot. -func (h *Handler) GetUsageStatistics(c *gin.Context) { - var snapshot usage.StatisticsSnapshot - if h != nil && h.usageStats != nil { - snapshot = h.usageStats.Snapshot() - } - c.JSON(http.StatusOK, gin.H{ - "usage": snapshot, - "failed_requests": snapshot.FailureCount, - }) -} - -// ExportUsageStatistics returns a complete usage snapshot for backup/migration. -func (h *Handler) ExportUsageStatistics(c *gin.Context) { - var snapshot usage.StatisticsSnapshot - if h != nil && h.usageStats != nil { - snapshot = h.usageStats.Snapshot() - } - c.JSON(http.StatusOK, usageExportPayload{ - Version: 1, - ExportedAt: time.Now().UTC(), - Usage: snapshot, - }) -} - -// ImportUsageStatistics merges a previously exported usage snapshot into memory. -func (h *Handler) ImportUsageStatistics(c *gin.Context) { - if h == nil || h.usageStats == nil { - c.JSON(http.StatusBadRequest, gin.H{"error": "usage statistics unavailable"}) - return - } - - data, err := c.GetRawData() - if err != nil { - c.JSON(http.StatusBadRequest, gin.H{"error": "failed to read request body"}) - return - } - - var payload usageImportPayload - if err := json.Unmarshal(data, &payload); err != nil { - c.JSON(http.StatusBadRequest, gin.H{"error": "invalid json"}) - return - } - if payload.Version != 0 && payload.Version != 1 { - c.JSON(http.StatusBadRequest, gin.H{"error": "unsupported version"}) - return - } - - result := h.usageStats.MergeSnapshot(payload.Usage) - snapshot := h.usageStats.Snapshot() - c.JSON(http.StatusOK, gin.H{ - "added": result.Added, - "skipped": result.Skipped, - "total_requests": snapshot.TotalRequests, - "failed_requests": snapshot.FailureCount, - }) -} diff --git a/internal/api/server.go b/internal/api/server.go index 4d51460d..176bc2a3 100644 --- a/internal/api/server.go +++ b/internal/api/server.go @@ -31,7 +31,6 @@ import ( "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" "github.com/router-for-me/CLIProxyAPI/v6/internal/managementasset" "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" - "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" "github.com/router-for-me/CLIProxyAPI/v6/internal/util" sdkaccess "github.com/router-for-me/CLIProxyAPI/v6/sdk/access" "github.com/router-for-me/CLIProxyAPI/v6/sdk/api/handlers" @@ -507,9 +506,6 @@ func (s *Server) registerManagementRoutes() { mgmt := s.engine.Group("/v0/management") mgmt.Use(s.managementAvailabilityMiddleware(), s.mgmt.Middleware()) { - mgmt.GET("/usage", s.mgmt.GetUsageStatistics) - mgmt.GET("/usage/export", s.mgmt.ExportUsageStatistics) - mgmt.POST("/usage/import", s.mgmt.ImportUsageStatistics) mgmt.GET("/config", s.mgmt.GetConfig) mgmt.GET("/config.yaml", s.mgmt.GetConfigYAML) mgmt.PUT("/config.yaml", s.mgmt.PutConfigYAML) @@ -1001,7 +997,7 @@ func (s *Server) UpdateClients(cfg *config.Config) { } if oldCfg == nil || oldCfg.UsageStatisticsEnabled != cfg.UsageStatisticsEnabled { - usage.SetStatisticsEnabled(cfg.UsageStatisticsEnabled) + redisqueue.SetUsageStatisticsEnabled(cfg.UsageStatisticsEnabled) } if s.requestLogger != nil && (oldCfg == nil || oldCfg.ErrorLogsMaxFiles != cfg.ErrorLogsMaxFiles) { diff --git a/internal/redisqueue/plugin.go b/internal/redisqueue/plugin.go index 39739dbe..97168419 100644 --- a/internal/redisqueue/plugin.go +++ b/internal/redisqueue/plugin.go @@ -7,7 +7,6 @@ import ( "time" internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" - internalusage "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" ) @@ -21,7 +20,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec if p == nil { return } - if !Enabled() || !internalusage.StatisticsEnabled() { + if !Enabled() || !UsageStatisticsEnabled() { return } @@ -45,7 +44,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec apiKey := strings.TrimSpace(record.APIKey) requestID := strings.TrimSpace(internallogging.GetRequestID(ctx)) - tokens := internalusage.TokenStats{ + tokens := tokenStats{ InputTokens: record.Detail.InputTokens, OutputTokens: record.Detail.OutputTokens, ReasoningTokens: record.Detail.ReasoningTokens, @@ -64,7 +63,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec failed = !resolveSuccess(ctx) } - detail := internalusage.RequestDetail{ + detail := requestDetail{ Timestamp: timestamp, LatencyMs: record.Latency.Milliseconds(), Source: record.Source, @@ -74,7 +73,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec } payload, err := json.Marshal(queuedUsageDetail{ - RequestDetail: detail, + requestDetail: detail, Provider: provider, Model: modelName, Endpoint: resolveEndpoint(ctx), @@ -89,7 +88,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec } type queuedUsageDetail struct { - internalusage.RequestDetail + requestDetail Provider string `json:"provider"` Model string `json:"model"` Endpoint string `json:"endpoint"` @@ -98,6 +97,23 @@ type queuedUsageDetail struct { RequestID string `json:"request_id"` } +type requestDetail struct { + Timestamp time.Time `json:"timestamp"` + LatencyMs int64 `json:"latency_ms"` + Source string `json:"source"` + AuthIndex string `json:"auth_index"` + Tokens tokenStats `json:"tokens"` + Failed bool `json:"failed"` +} + +type tokenStats struct { + InputTokens int64 `json:"input_tokens"` + OutputTokens int64 `json:"output_tokens"` + ReasoningTokens int64 `json:"reasoning_tokens"` + CachedTokens int64 `json:"cached_tokens"` + TotalTokens int64 `json:"total_tokens"` +} + func resolveSuccess(ctx context.Context) bool { status := internallogging.GetResponseStatus(ctx) if status == 0 { diff --git a/internal/redisqueue/plugin_test.go b/internal/redisqueue/plugin_test.go index 1e8bda48..0cc8b9b9 100644 --- a/internal/redisqueue/plugin_test.go +++ b/internal/redisqueue/plugin_test.go @@ -10,7 +10,6 @@ import ( "github.com/gin-gonic/gin" internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" - internalusage "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" ) @@ -127,16 +126,16 @@ func withEnabledQueue(t *testing.T, fn func()) { t.Helper() prevQueueEnabled := Enabled() - prevStatsEnabled := internalusage.StatisticsEnabled() + prevUsageEnabled := UsageStatisticsEnabled() SetEnabled(false) SetEnabled(true) - internalusage.SetStatisticsEnabled(true) + SetUsageStatisticsEnabled(true) defer func() { SetEnabled(false) SetEnabled(prevQueueEnabled) - internalusage.SetStatisticsEnabled(prevStatsEnabled) + SetUsageStatisticsEnabled(prevUsageEnabled) }() fn() diff --git a/internal/redisqueue/usage_toggle.go b/internal/redisqueue/usage_toggle.go new file mode 100644 index 00000000..dddbeca6 --- /dev/null +++ b/internal/redisqueue/usage_toggle.go @@ -0,0 +1,16 @@ +package redisqueue + +import "sync/atomic" + +var usageStatisticsEnabled atomic.Bool + +func init() { + usageStatisticsEnabled.Store(true) +} + +// SetUsageStatisticsEnabled toggles whether usage records are enqueued into the redisqueue payload buffer. +// This is controlled by the config field `usage-statistics-enabled` and the corresponding management API. +func SetUsageStatisticsEnabled(enabled bool) { usageStatisticsEnabled.Store(enabled) } + +// UsageStatisticsEnabled reports whether the usage queue plugin should publish records. +func UsageStatisticsEnabled() bool { return usageStatisticsEnabled.Load() } diff --git a/internal/tui/app.go b/internal/tui/app.go index b9ee9e1a..c0a7c3a8 100644 --- a/internal/tui/app.go +++ b/internal/tui/app.go @@ -18,7 +18,6 @@ const ( tabAuthFiles tabAPIKeys tabOAuth - tabUsage tabLogs ) @@ -40,7 +39,6 @@ type App struct { auth authTabModel keys keysTabModel oauth oauthTabModel - usage usageTabModel logs logsTabModel client *Client @@ -50,7 +48,7 @@ type App struct { ready bool // Track which tabs have been initialized (fetched data) - initialized [7]bool + initialized [6]bool } type authConnectMsg struct { @@ -81,10 +79,9 @@ func NewApp(port int, secretKey string, hook *LogHook) App { auth: newAuthTabModel(client), keys: newKeysTabModel(client), oauth: newOAuthTabModel(client), - usage: newUsageTabModel(client), logs: newLogsTabModel(client, hook), client: client, - initialized: [7]bool{ + initialized: [6]bool{ tabDashboard: true, tabLogs: true, }, @@ -92,7 +89,7 @@ func NewApp(port int, secretKey string, hook *LogHook) App { app.refreshTabs() if authRequired { - app.initialized = [7]bool{} + app.initialized = [6]bool{} } app.setAuthInputPrompt() return app @@ -128,7 +125,6 @@ func (a App) Update(msg tea.Msg) (tea.Model, tea.Cmd) { a.auth.SetSize(contentW, contentH) a.keys.SetSize(contentW, contentH) a.oauth.SetSize(contentW, contentH) - a.usage.SetSize(contentW, contentH) a.logs.SetSize(contentW, contentH) return a, nil @@ -142,7 +138,7 @@ func (a App) Update(msg tea.Msg) (tea.Model, tea.Cmd) { a.authenticated = true a.logsEnabled = a.standalone || isLogsEnabledFromConfig(msg.cfg) a.refreshTabs() - a.initialized = [7]bool{} + a.initialized = [6]bool{} a.initialized[tabDashboard] = true cmds := []tea.Cmd{a.dashboard.Init()} if a.logsEnabled { @@ -258,8 +254,6 @@ func (a App) Update(msg tea.Msg) (tea.Model, tea.Cmd) { a.keys, cmd = a.keys.Update(msg) case tabOAuth: a.oauth, cmd = a.oauth.Update(msg) - case tabUsage: - a.usage, cmd = a.usage.Update(msg) case tabLogs: a.logs, cmd = a.logs.Update(msg) } @@ -322,8 +316,6 @@ func (a *App) initTabIfNeeded(_ int) tea.Cmd { return a.keys.Init() case tabOAuth: return a.oauth.Init() - case tabUsage: - return a.usage.Init() case tabLogs: if !a.logsEnabled { return nil @@ -360,8 +352,6 @@ func (a App) View() string { sb.WriteString(a.keys.View()) case tabOAuth: sb.WriteString(a.oauth.View()) - case tabUsage: - sb.WriteString(a.usage.View()) case tabLogs: if a.logsEnabled { sb.WriteString(a.logs.View()) @@ -529,10 +519,6 @@ func (a App) broadcastToAllTabs(msg tea.Msg) (tea.Model, tea.Cmd) { if cmd != nil { cmds = append(cmds, cmd) } - a.usage, cmd = a.usage.Update(msg) - if cmd != nil { - cmds = append(cmds, cmd) - } a.logs, cmd = a.logs.Update(msg) if cmd != nil { cmds = append(cmds, cmd) diff --git a/internal/tui/client.go b/internal/tui/client.go index 6f75d6be..747f30b9 100644 --- a/internal/tui/client.go +++ b/internal/tui/client.go @@ -140,11 +140,6 @@ func (c *Client) PutConfigYAML(yamlContent string) error { return err } -// GetUsage fetches usage statistics. -func (c *Client) GetUsage() (map[string]any, error) { - return c.getJSON("/v0/management/usage") -} - // GetAuthFiles lists auth credential files. // API returns {"files": [...]}. func (c *Client) GetAuthFiles() ([]map[string]any, error) { diff --git a/internal/tui/dashboard.go b/internal/tui/dashboard.go index 8561fe9c..99b5409c 100644 --- a/internal/tui/dashboard.go +++ b/internal/tui/dashboard.go @@ -22,14 +22,12 @@ type dashboardModel struct { // Cached data for re-rendering on locale change lastConfig map[string]any - lastUsage map[string]any lastAuthFiles []map[string]any lastAPIKeys []string } type dashboardDataMsg struct { config map[string]any - usage map[string]any authFiles []map[string]any apiKeys []string err error @@ -47,25 +45,24 @@ func (m dashboardModel) Init() tea.Cmd { func (m dashboardModel) fetchData() tea.Msg { cfg, cfgErr := m.client.GetConfig() - usage, usageErr := m.client.GetUsage() authFiles, authErr := m.client.GetAuthFiles() apiKeys, keysErr := m.client.GetAPIKeys() var err error - for _, e := range []error{cfgErr, usageErr, authErr, keysErr} { + for _, e := range []error{cfgErr, authErr, keysErr} { if e != nil { err = e break } } - return dashboardDataMsg{config: cfg, usage: usage, authFiles: authFiles, apiKeys: apiKeys, err: err} + return dashboardDataMsg{config: cfg, authFiles: authFiles, apiKeys: apiKeys, err: err} } func (m dashboardModel) Update(msg tea.Msg) (dashboardModel, tea.Cmd) { switch msg := msg.(type) { case localeChangedMsg: // Re-render immediately with cached data using new locale - m.content = m.renderDashboard(m.lastConfig, m.lastUsage, m.lastAuthFiles, m.lastAPIKeys) + m.content = m.renderDashboard(m.lastConfig, m.lastAuthFiles, m.lastAPIKeys) m.viewport.SetContent(m.content) // Also fetch fresh data in background return m, m.fetchData @@ -78,11 +75,10 @@ func (m dashboardModel) Update(msg tea.Msg) (dashboardModel, tea.Cmd) { m.err = nil // Cache data for locale switching m.lastConfig = msg.config - m.lastUsage = msg.usage m.lastAuthFiles = msg.authFiles m.lastAPIKeys = msg.apiKeys - m.content = m.renderDashboard(msg.config, msg.usage, msg.authFiles, msg.apiKeys) + m.content = m.renderDashboard(msg.config, msg.authFiles, msg.apiKeys) } m.viewport.SetContent(m.content) return m, nil @@ -121,7 +117,7 @@ func (m dashboardModel) View() string { return m.viewport.View() } -func (m dashboardModel) renderDashboard(cfg, usage map[string]any, authFiles []map[string]any, apiKeys []string) string { +func (m dashboardModel) renderDashboard(cfg map[string]any, authFiles []map[string]any, apiKeys []string) string { var sb strings.Builder sb.WriteString(titleStyle.Render(T("dashboard_title"))) @@ -138,7 +134,7 @@ func (m dashboardModel) renderDashboard(cfg, usage map[string]any, authFiles []m // ━━━ Stats Cards ━━━ cardWidth := 25 if m.width > 0 { - cardWidth = (m.width - 6) / 4 + cardWidth = (m.width - 2) / 2 if cardWidth < 18 { cardWidth = 18 } @@ -173,34 +169,7 @@ func (m dashboardModel) renderDashboard(cfg, usage map[string]any, authFiles []m lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%s (%d %s)", T("auth_files_label"), activeAuth, T("active_suffix"))), )) - // Card 3: Total Requests - totalReqs := int64(0) - successReqs := int64(0) - failedReqs := int64(0) - totalTokens := int64(0) - if usage != nil { - if usageMap, ok := usage["usage"].(map[string]any); ok { - totalReqs = int64(getFloat(usageMap, "total_requests")) - successReqs = int64(getFloat(usageMap, "success_count")) - failedReqs = int64(getFloat(usageMap, "failure_count")) - totalTokens = int64(getFloat(usageMap, "total_tokens")) - } - } - card3 := cardStyle.Render(fmt.Sprintf( - "%s\n%s", - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("214")).Render(fmt.Sprintf("📈 %d", totalReqs)), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%s (✓%d ✗%d)", T("total_requests"), successReqs, failedReqs)), - )) - - // Card 4: Total Tokens - tokenStr := formatLargeNumber(totalTokens) - card4 := cardStyle.Render(fmt.Sprintf( - "%s\n%s", - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("170")).Render(fmt.Sprintf("🔤 %s", tokenStr)), - lipgloss.NewStyle().Foreground(colorMuted).Render(T("total_tokens")), - )) - - sb.WriteString(lipgloss.JoinHorizontal(lipgloss.Top, card1, " ", card2, " ", card3, " ", card4)) + sb.WriteString(lipgloss.JoinHorizontal(lipgloss.Top, card1, " ", card2)) sb.WriteString("\n\n") // ━━━ Current Config ━━━ @@ -258,38 +227,6 @@ func (m dashboardModel) renderDashboard(cfg, usage map[string]any, authFiles []m sb.WriteString("\n") - // ━━━ Per-Model Usage ━━━ - if usage != nil { - if usageMap, ok := usage["usage"].(map[string]any); ok { - if apis, ok := usageMap["apis"].(map[string]any); ok && len(apis) > 0 { - sb.WriteString(lipgloss.NewStyle().Bold(true).Foreground(colorHighlight).Render(T("model_stats"))) - sb.WriteString("\n") - sb.WriteString(strings.Repeat("─", minInt(m.width, 60))) - sb.WriteString("\n") - - header := fmt.Sprintf(" %-40s %10s %12s", T("model"), T("requests"), T("tokens")) - sb.WriteString(tableHeaderStyle.Render(header)) - sb.WriteString("\n") - - for _, apiSnap := range apis { - if apiMap, ok := apiSnap.(map[string]any); ok { - if models, ok := apiMap["models"].(map[string]any); ok { - for model, v := range models { - if stats, ok := v.(map[string]any); ok { - reqs := int64(getFloat(stats, "total_requests")) - toks := int64(getFloat(stats, "total_tokens")) - row := fmt.Sprintf(" %-40s %10d %12s", truncate(model, 40), reqs, formatLargeNumber(toks)) - sb.WriteString(tableCellStyle.Render(row)) - sb.WriteString("\n") - } - } - } - } - } - } - } - } - return sb.String() } diff --git a/internal/tui/i18n.go b/internal/tui/i18n.go index f6a33ca4..a4c0ac16 100644 --- a/internal/tui/i18n.go +++ b/internal/tui/i18n.go @@ -50,8 +50,8 @@ var locales = map[string]map[string]string{ // ────────────────────────────────────────── // Tab names // ────────────────────────────────────────── -var zhTabNames = []string{"仪表盘", "配置", "认证文件", "API 密钥", "OAuth", "使用统计", "日志"} -var enTabNames = []string{"Dashboard", "Config", "Auth Files", "API Keys", "OAuth", "Usage", "Logs"} +var zhTabNames = []string{"仪表盘", "配置", "认证文件", "API 密钥", "OAuth", "日志"} +var enTabNames = []string{"Dashboard", "Config", "Auth Files", "API Keys", "OAuth", "Logs"} // TabNames returns tab names in the current locale. func TabNames() []string { diff --git a/internal/tui/usage_tab.go b/internal/tui/usage_tab.go deleted file mode 100644 index 6b9fef5e..00000000 --- a/internal/tui/usage_tab.go +++ /dev/null @@ -1,418 +0,0 @@ -package tui - -import ( - "fmt" - "sort" - "strings" - - "github.com/charmbracelet/bubbles/viewport" - tea "github.com/charmbracelet/bubbletea" - "github.com/charmbracelet/lipgloss" -) - -// usageTabModel displays usage statistics with charts and breakdowns. -type usageTabModel struct { - client *Client - viewport viewport.Model - usage map[string]any - err error - width int - height int - ready bool -} - -type usageDataMsg struct { - usage map[string]any - err error -} - -func newUsageTabModel(client *Client) usageTabModel { - return usageTabModel{ - client: client, - } -} - -func (m usageTabModel) Init() tea.Cmd { - return m.fetchData -} - -func (m usageTabModel) fetchData() tea.Msg { - usage, err := m.client.GetUsage() - return usageDataMsg{usage: usage, err: err} -} - -func (m usageTabModel) Update(msg tea.Msg) (usageTabModel, tea.Cmd) { - switch msg := msg.(type) { - case localeChangedMsg: - m.viewport.SetContent(m.renderContent()) - return m, nil - case usageDataMsg: - if msg.err != nil { - m.err = msg.err - } else { - m.err = nil - m.usage = msg.usage - } - m.viewport.SetContent(m.renderContent()) - return m, nil - - case tea.KeyMsg: - if msg.String() == "r" { - return m, m.fetchData - } - var cmd tea.Cmd - m.viewport, cmd = m.viewport.Update(msg) - return m, cmd - } - - var cmd tea.Cmd - m.viewport, cmd = m.viewport.Update(msg) - return m, cmd -} - -func (m *usageTabModel) SetSize(w, h int) { - m.width = w - m.height = h - if !m.ready { - m.viewport = viewport.New(w, h) - m.viewport.SetContent(m.renderContent()) - m.ready = true - } else { - m.viewport.Width = w - m.viewport.Height = h - } -} - -func (m usageTabModel) View() string { - if !m.ready { - return T("loading") - } - return m.viewport.View() -} - -func (m usageTabModel) renderContent() string { - var sb strings.Builder - - sb.WriteString(titleStyle.Render(T("usage_title"))) - sb.WriteString("\n") - sb.WriteString(helpStyle.Render(T("usage_help"))) - sb.WriteString("\n\n") - - if m.err != nil { - sb.WriteString(errorStyle.Render("⚠ Error: " + m.err.Error())) - sb.WriteString("\n") - return sb.String() - } - - if m.usage == nil { - sb.WriteString(subtitleStyle.Render(T("usage_no_data"))) - sb.WriteString("\n") - return sb.String() - } - - usageMap, _ := m.usage["usage"].(map[string]any) - if usageMap == nil { - sb.WriteString(subtitleStyle.Render(T("usage_no_data"))) - sb.WriteString("\n") - return sb.String() - } - - totalReqs := int64(getFloat(usageMap, "total_requests")) - successCnt := int64(getFloat(usageMap, "success_count")) - failureCnt := int64(getFloat(usageMap, "failure_count")) - totalTokens := int64(getFloat(usageMap, "total_tokens")) - - // ━━━ Overview Cards ━━━ - cardWidth := 20 - if m.width > 0 { - cardWidth = (m.width - 6) / 4 - if cardWidth < 16 { - cardWidth = 16 - } - } - cardStyle := lipgloss.NewStyle(). - Border(lipgloss.RoundedBorder()). - BorderForeground(lipgloss.Color("240")). - Padding(0, 1). - Width(cardWidth). - Height(3) - - // Total Requests - card1 := cardStyle.Copy().BorderForeground(lipgloss.Color("111")).Render(fmt.Sprintf( - "%s\n%s\n%s", - lipgloss.NewStyle().Foreground(colorMuted).Render(T("usage_total_reqs")), - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("111")).Render(fmt.Sprintf("%d", totalReqs)), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("● %s: %d ● %s: %d", T("usage_success"), successCnt, T("usage_failure"), failureCnt)), - )) - - // Total Tokens - card2 := cardStyle.Copy().BorderForeground(lipgloss.Color("214")).Render(fmt.Sprintf( - "%s\n%s\n%s", - lipgloss.NewStyle().Foreground(colorMuted).Render(T("usage_total_tokens")), - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("214")).Render(formatLargeNumber(totalTokens)), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%s: %s", T("usage_total_token_l"), formatLargeNumber(totalTokens))), - )) - - // RPM - rpm := float64(0) - if totalReqs > 0 { - if rByH, ok := usageMap["requests_by_hour"].(map[string]any); ok && len(rByH) > 0 { - rpm = float64(totalReqs) / float64(len(rByH)) / 60.0 - } - } - card3 := cardStyle.Copy().BorderForeground(lipgloss.Color("76")).Render(fmt.Sprintf( - "%s\n%s\n%s", - lipgloss.NewStyle().Foreground(colorMuted).Render(T("usage_rpm")), - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("76")).Render(fmt.Sprintf("%.2f", rpm)), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%s: %d", T("usage_total_reqs"), totalReqs)), - )) - - // TPM - tpm := float64(0) - if totalTokens > 0 { - if tByH, ok := usageMap["tokens_by_hour"].(map[string]any); ok && len(tByH) > 0 { - tpm = float64(totalTokens) / float64(len(tByH)) / 60.0 - } - } - card4 := cardStyle.Copy().BorderForeground(lipgloss.Color("170")).Render(fmt.Sprintf( - "%s\n%s\n%s", - lipgloss.NewStyle().Foreground(colorMuted).Render(T("usage_tpm")), - lipgloss.NewStyle().Bold(true).Foreground(lipgloss.Color("170")).Render(fmt.Sprintf("%.2f", tpm)), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%s: %s", T("usage_total_tokens"), formatLargeNumber(totalTokens))), - )) - - sb.WriteString(lipgloss.JoinHorizontal(lipgloss.Top, card1, " ", card2, " ", card3, " ", card4)) - sb.WriteString("\n\n") - - // ━━━ Requests by Hour (ASCII bar chart) ━━━ - if rByH, ok := usageMap["requests_by_hour"].(map[string]any); ok && len(rByH) > 0 { - sb.WriteString(lipgloss.NewStyle().Bold(true).Foreground(colorHighlight).Render(T("usage_req_by_hour"))) - sb.WriteString("\n") - sb.WriteString(strings.Repeat("─", minInt(m.width, 60))) - sb.WriteString("\n") - sb.WriteString(renderBarChart(rByH, m.width-6, lipgloss.Color("111"))) - sb.WriteString("\n") - } - - // ━━━ Tokens by Hour ━━━ - if tByH, ok := usageMap["tokens_by_hour"].(map[string]any); ok && len(tByH) > 0 { - sb.WriteString(lipgloss.NewStyle().Bold(true).Foreground(colorHighlight).Render(T("usage_tok_by_hour"))) - sb.WriteString("\n") - sb.WriteString(strings.Repeat("─", minInt(m.width, 60))) - sb.WriteString("\n") - sb.WriteString(renderBarChart(tByH, m.width-6, lipgloss.Color("214"))) - sb.WriteString("\n") - } - - // ━━━ Requests by Day ━━━ - if rByD, ok := usageMap["requests_by_day"].(map[string]any); ok && len(rByD) > 0 { - sb.WriteString(lipgloss.NewStyle().Bold(true).Foreground(colorHighlight).Render(T("usage_req_by_day"))) - sb.WriteString("\n") - sb.WriteString(strings.Repeat("─", minInt(m.width, 60))) - sb.WriteString("\n") - sb.WriteString(renderBarChart(rByD, m.width-6, lipgloss.Color("76"))) - sb.WriteString("\n") - } - - // ━━━ API Detail Stats ━━━ - if apis, ok := usageMap["apis"].(map[string]any); ok && len(apis) > 0 { - sb.WriteString(lipgloss.NewStyle().Bold(true).Foreground(colorHighlight).Render(T("usage_api_detail"))) - sb.WriteString("\n") - sb.WriteString(strings.Repeat("─", minInt(m.width, 80))) - sb.WriteString("\n") - - header := fmt.Sprintf(" %-30s %10s %12s", "API", T("requests"), T("tokens")) - sb.WriteString(tableHeaderStyle.Render(header)) - sb.WriteString("\n") - - for apiName, apiSnap := range apis { - if apiMap, ok := apiSnap.(map[string]any); ok { - apiReqs := int64(getFloat(apiMap, "total_requests")) - apiToks := int64(getFloat(apiMap, "total_tokens")) - - row := fmt.Sprintf(" %-30s %10d %12s", - truncate(maskKey(apiName), 30), apiReqs, formatLargeNumber(apiToks)) - sb.WriteString(lipgloss.NewStyle().Bold(true).Render(row)) - sb.WriteString("\n") - - // Per-model breakdown - if models, ok := apiMap["models"].(map[string]any); ok { - for model, v := range models { - if stats, ok := v.(map[string]any); ok { - mReqs := int64(getFloat(stats, "total_requests")) - mToks := int64(getFloat(stats, "total_tokens")) - mRow := fmt.Sprintf(" ├─ %-28s %10d %12s", - truncate(model, 28), mReqs, formatLargeNumber(mToks)) - sb.WriteString(tableCellStyle.Render(mRow)) - sb.WriteString("\n") - - // Token type breakdown from details - sb.WriteString(m.renderTokenBreakdown(stats)) - - // Latency breakdown from details - sb.WriteString(m.renderLatencyBreakdown(stats)) - } - } - } - } - } - } - - sb.WriteString("\n") - return sb.String() -} - -// renderTokenBreakdown aggregates input/output/cached/reasoning tokens from model details. -func (m usageTabModel) renderTokenBreakdown(modelStats map[string]any) string { - details, ok := modelStats["details"] - if !ok { - return "" - } - detailList, ok := details.([]any) - if !ok || len(detailList) == 0 { - return "" - } - - var inputTotal, outputTotal, cachedTotal, reasoningTotal int64 - for _, d := range detailList { - dm, ok := d.(map[string]any) - if !ok { - continue - } - tokens, ok := dm["tokens"].(map[string]any) - if !ok { - continue - } - inputTotal += int64(getFloat(tokens, "input_tokens")) - outputTotal += int64(getFloat(tokens, "output_tokens")) - cachedTotal += int64(getFloat(tokens, "cached_tokens")) - reasoningTotal += int64(getFloat(tokens, "reasoning_tokens")) - } - - if inputTotal == 0 && outputTotal == 0 && cachedTotal == 0 && reasoningTotal == 0 { - return "" - } - - parts := []string{} - if inputTotal > 0 { - parts = append(parts, fmt.Sprintf("%s:%s", T("usage_input"), formatLargeNumber(inputTotal))) - } - if outputTotal > 0 { - parts = append(parts, fmt.Sprintf("%s:%s", T("usage_output"), formatLargeNumber(outputTotal))) - } - if cachedTotal > 0 { - parts = append(parts, fmt.Sprintf("%s:%s", T("usage_cached"), formatLargeNumber(cachedTotal))) - } - if reasoningTotal > 0 { - parts = append(parts, fmt.Sprintf("%s:%s", T("usage_reasoning"), formatLargeNumber(reasoningTotal))) - } - - return fmt.Sprintf(" │ %s\n", - lipgloss.NewStyle().Foreground(colorMuted).Render(strings.Join(parts, " "))) -} - -// renderLatencyBreakdown aggregates latency_ms from model details and displays avg/min/max. -func (m usageTabModel) renderLatencyBreakdown(modelStats map[string]any) string { - details, ok := modelStats["details"] - if !ok { - return "" - } - detailList, ok := details.([]any) - if !ok || len(detailList) == 0 { - return "" - } - - var totalLatency int64 - var count int - var minLatency, maxLatency int64 - first := true - - for _, d := range detailList { - dm, ok := d.(map[string]any) - if !ok { - continue - } - latencyMs := int64(getFloat(dm, "latency_ms")) - if latencyMs <= 0 { - continue - } - totalLatency += latencyMs - count++ - if first { - minLatency = latencyMs - maxLatency = latencyMs - first = false - } else { - if latencyMs < minLatency { - minLatency = latencyMs - } - if latencyMs > maxLatency { - maxLatency = latencyMs - } - } - } - - if count == 0 { - return "" - } - - avgLatency := totalLatency / int64(count) - return fmt.Sprintf(" │ %s: avg %dms min %dms max %dms\n", - lipgloss.NewStyle().Foreground(colorMuted).Render(T("usage_time")), - avgLatency, minLatency, maxLatency) -} - -// renderBarChart renders a simple ASCII horizontal bar chart. -func renderBarChart(data map[string]any, maxBarWidth int, barColor lipgloss.Color) string { - if maxBarWidth < 10 { - maxBarWidth = 10 - } - - // Sort keys - keys := make([]string, 0, len(data)) - for k := range data { - keys = append(keys, k) - } - sort.Strings(keys) - - // Find max value - maxVal := float64(0) - for _, k := range keys { - v := getFloat(data, k) - if v > maxVal { - maxVal = v - } - } - if maxVal == 0 { - return "" - } - - barStyle := lipgloss.NewStyle().Foreground(barColor) - var sb strings.Builder - - labelWidth := 12 - barAvail := maxBarWidth - labelWidth - 12 - if barAvail < 5 { - barAvail = 5 - } - - for _, k := range keys { - v := getFloat(data, k) - barLen := int(v / maxVal * float64(barAvail)) - if barLen < 1 && v > 0 { - barLen = 1 - } - bar := strings.Repeat("█", barLen) - label := k - if len(label) > labelWidth { - label = label[:labelWidth] - } - sb.WriteString(fmt.Sprintf(" %-*s %s %s\n", - labelWidth, label, - barStyle.Render(bar), - lipgloss.NewStyle().Foreground(colorMuted).Render(fmt.Sprintf("%.0f", v)), - )) - } - - return sb.String() -} diff --git a/internal/tui/usage_tab_test.go b/internal/tui/usage_tab_test.go deleted file mode 100644 index 4fffcd98..00000000 --- a/internal/tui/usage_tab_test.go +++ /dev/null @@ -1,134 +0,0 @@ -package tui - -import ( - "strings" - "testing" -) - -func TestRenderLatencyBreakdown(t *testing.T) { - tests := []struct { - name string - modelStats map[string]any - wantEmpty bool - wantContains string - }{ - { - name: "no details", - modelStats: map[string]any{}, - wantEmpty: true, - }, - { - name: "empty details", - modelStats: map[string]any{ - "details": []any{}, - }, - wantEmpty: true, - }, - { - name: "details with zero latency", - modelStats: map[string]any{ - "details": []any{ - map[string]any{ - "latency_ms": float64(0), - }, - }, - }, - wantEmpty: true, - }, - { - name: "single request with latency", - modelStats: map[string]any{ - "details": []any{ - map[string]any{ - "latency_ms": float64(1500), - }, - }, - }, - wantEmpty: false, - wantContains: "avg 1500ms min 1500ms max 1500ms", - }, - { - name: "multiple requests with varying latency", - modelStats: map[string]any{ - "details": []any{ - map[string]any{ - "latency_ms": float64(100), - }, - map[string]any{ - "latency_ms": float64(200), - }, - map[string]any{ - "latency_ms": float64(300), - }, - }, - }, - wantEmpty: false, - wantContains: "avg 200ms min 100ms max 300ms", - }, - { - name: "mixed valid and invalid latency values", - modelStats: map[string]any{ - "details": []any{ - map[string]any{ - "latency_ms": float64(500), - }, - map[string]any{ - "latency_ms": float64(0), - }, - map[string]any{ - "latency_ms": float64(1500), - }, - }, - }, - wantEmpty: false, - wantContains: "avg 1000ms min 500ms max 1500ms", - }, - } - - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - m := usageTabModel{} - result := m.renderLatencyBreakdown(tt.modelStats) - - if tt.wantEmpty { - if result != "" { - t.Errorf("renderLatencyBreakdown() = %q, want empty string", result) - } - return - } - - if result == "" { - t.Errorf("renderLatencyBreakdown() = empty, want non-empty string") - return - } - - if tt.wantContains != "" && !strings.Contains(result, tt.wantContains) { - t.Errorf("renderLatencyBreakdown() = %q, want to contain %q", result, tt.wantContains) - } - }) - } -} - -func TestUsageTimeTranslations(t *testing.T) { - prevLocale := CurrentLocale() - t.Cleanup(func() { - SetLocale(prevLocale) - }) - - tests := []struct { - locale string - want string - }{ - {locale: "en", want: "Time"}, - {locale: "zh", want: "时间"}, - } - - for _, tt := range tests { - t.Run(tt.locale, func(t *testing.T) { - SetLocale(tt.locale) - if got := T("usage_time"); got != tt.want { - t.Fatalf("T(usage_time) = %q, want %q", got, tt.want) - } - }) - } -} diff --git a/internal/usage/logger_plugin.go b/internal/usage/logger_plugin.go deleted file mode 100644 index 9d59de4f..00000000 --- a/internal/usage/logger_plugin.go +++ /dev/null @@ -1,464 +0,0 @@ -// Package usage provides usage tracking and logging functionality for the CLI Proxy API server. -// It includes plugins for monitoring API usage, token consumption, and other metrics -// to help with observability and billing purposes. -package usage - -import ( - "context" - "fmt" - "strings" - "sync" - "sync/atomic" - "time" - - internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" - coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" -) - -var statisticsEnabled atomic.Bool - -func init() { - statisticsEnabled.Store(true) - coreusage.RegisterPlugin(NewLoggerPlugin()) -} - -// LoggerPlugin collects in-memory request statistics for usage analysis. -// It implements coreusage.Plugin to receive usage records emitted by the runtime. -type LoggerPlugin struct { - stats *RequestStatistics -} - -// NewLoggerPlugin constructs a new logger plugin instance. -// -// Returns: -// - *LoggerPlugin: A new logger plugin instance wired to the shared statistics store. -func NewLoggerPlugin() *LoggerPlugin { return &LoggerPlugin{stats: defaultRequestStatistics} } - -// HandleUsage implements coreusage.Plugin. -// It updates the in-memory statistics store whenever a usage record is received. -// -// Parameters: -// - ctx: The context for the usage record -// - record: The usage record to aggregate -func (p *LoggerPlugin) HandleUsage(ctx context.Context, record coreusage.Record) { - if !statisticsEnabled.Load() { - return - } - if p == nil || p.stats == nil { - return - } - p.stats.Record(ctx, record) -} - -// SetStatisticsEnabled toggles whether in-memory statistics are recorded. -func SetStatisticsEnabled(enabled bool) { statisticsEnabled.Store(enabled) } - -// StatisticsEnabled reports the current recording state. -func StatisticsEnabled() bool { return statisticsEnabled.Load() } - -// RequestStatistics maintains aggregated request metrics in memory. -type RequestStatistics struct { - mu sync.RWMutex - - totalRequests int64 - successCount int64 - failureCount int64 - totalTokens int64 - - apis map[string]*apiStats - - requestsByDay map[string]int64 - requestsByHour map[int]int64 - tokensByDay map[string]int64 - tokensByHour map[int]int64 -} - -// apiStats holds aggregated metrics for a single API key. -type apiStats struct { - TotalRequests int64 - TotalTokens int64 - Models map[string]*modelStats -} - -// modelStats holds aggregated metrics for a specific model within an API. -type modelStats struct { - TotalRequests int64 - TotalTokens int64 - Details []RequestDetail -} - -// RequestDetail stores the timestamp, latency, and token usage for a single request. -type RequestDetail struct { - Timestamp time.Time `json:"timestamp"` - LatencyMs int64 `json:"latency_ms"` - Source string `json:"source"` - AuthIndex string `json:"auth_index"` - Tokens TokenStats `json:"tokens"` - Failed bool `json:"failed"` -} - -// TokenStats captures the token usage breakdown for a request. -type TokenStats struct { - InputTokens int64 `json:"input_tokens"` - OutputTokens int64 `json:"output_tokens"` - ReasoningTokens int64 `json:"reasoning_tokens"` - CachedTokens int64 `json:"cached_tokens"` - TotalTokens int64 `json:"total_tokens"` -} - -// StatisticsSnapshot represents an immutable view of the aggregated metrics. -type StatisticsSnapshot struct { - TotalRequests int64 `json:"total_requests"` - SuccessCount int64 `json:"success_count"` - FailureCount int64 `json:"failure_count"` - TotalTokens int64 `json:"total_tokens"` - - APIs map[string]APISnapshot `json:"apis"` - - RequestsByDay map[string]int64 `json:"requests_by_day"` - RequestsByHour map[string]int64 `json:"requests_by_hour"` - TokensByDay map[string]int64 `json:"tokens_by_day"` - TokensByHour map[string]int64 `json:"tokens_by_hour"` -} - -// APISnapshot summarises metrics for a single API key. -type APISnapshot struct { - TotalRequests int64 `json:"total_requests"` - TotalTokens int64 `json:"total_tokens"` - Models map[string]ModelSnapshot `json:"models"` -} - -// ModelSnapshot summarises metrics for a specific model. -type ModelSnapshot struct { - TotalRequests int64 `json:"total_requests"` - TotalTokens int64 `json:"total_tokens"` - Details []RequestDetail `json:"details"` -} - -var defaultRequestStatistics = NewRequestStatistics() - -// GetRequestStatistics returns the shared statistics store. -func GetRequestStatistics() *RequestStatistics { return defaultRequestStatistics } - -// NewRequestStatistics constructs an empty statistics store. -func NewRequestStatistics() *RequestStatistics { - return &RequestStatistics{ - apis: make(map[string]*apiStats), - requestsByDay: make(map[string]int64), - requestsByHour: make(map[int]int64), - tokensByDay: make(map[string]int64), - tokensByHour: make(map[int]int64), - } -} - -// Record ingests a new usage record and updates the aggregates. -func (s *RequestStatistics) Record(ctx context.Context, record coreusage.Record) { - if s == nil { - return - } - if !statisticsEnabled.Load() { - return - } - timestamp := record.RequestedAt - if timestamp.IsZero() { - timestamp = time.Now() - } - detail := normaliseDetail(record.Detail) - totalTokens := detail.TotalTokens - statsKey := record.APIKey - if statsKey == "" { - statsKey = resolveAPIIdentifier(ctx, record) - } - failed := record.Failed - if !failed { - failed = !resolveSuccess(ctx) - } - success := !failed - modelName := record.Model - if modelName == "" { - modelName = "unknown" - } - dayKey := timestamp.Format("2006-01-02") - hourKey := timestamp.Hour() - - s.mu.Lock() - defer s.mu.Unlock() - - s.totalRequests++ - if success { - s.successCount++ - } else { - s.failureCount++ - } - s.totalTokens += totalTokens - - stats, ok := s.apis[statsKey] - if !ok { - stats = &apiStats{Models: make(map[string]*modelStats)} - s.apis[statsKey] = stats - } - s.updateAPIStats(stats, modelName, RequestDetail{ - Timestamp: timestamp, - LatencyMs: normaliseLatency(record.Latency), - Source: record.Source, - AuthIndex: record.AuthIndex, - Tokens: detail, - Failed: failed, - }) - - s.requestsByDay[dayKey]++ - s.requestsByHour[hourKey]++ - s.tokensByDay[dayKey] += totalTokens - s.tokensByHour[hourKey] += totalTokens -} - -func (s *RequestStatistics) updateAPIStats(stats *apiStats, model string, detail RequestDetail) { - stats.TotalRequests++ - stats.TotalTokens += detail.Tokens.TotalTokens - modelStatsValue, ok := stats.Models[model] - if !ok { - modelStatsValue = &modelStats{} - stats.Models[model] = modelStatsValue - } - modelStatsValue.TotalRequests++ - modelStatsValue.TotalTokens += detail.Tokens.TotalTokens - modelStatsValue.Details = append(modelStatsValue.Details, detail) -} - -// Snapshot returns a copy of the aggregated metrics for external consumption. -func (s *RequestStatistics) Snapshot() StatisticsSnapshot { - result := StatisticsSnapshot{} - if s == nil { - return result - } - - s.mu.RLock() - defer s.mu.RUnlock() - - result.TotalRequests = s.totalRequests - result.SuccessCount = s.successCount - result.FailureCount = s.failureCount - result.TotalTokens = s.totalTokens - - result.APIs = make(map[string]APISnapshot, len(s.apis)) - for apiName, stats := range s.apis { - apiSnapshot := APISnapshot{ - TotalRequests: stats.TotalRequests, - TotalTokens: stats.TotalTokens, - Models: make(map[string]ModelSnapshot, len(stats.Models)), - } - for modelName, modelStatsValue := range stats.Models { - requestDetails := make([]RequestDetail, len(modelStatsValue.Details)) - copy(requestDetails, modelStatsValue.Details) - apiSnapshot.Models[modelName] = ModelSnapshot{ - TotalRequests: modelStatsValue.TotalRequests, - TotalTokens: modelStatsValue.TotalTokens, - Details: requestDetails, - } - } - result.APIs[apiName] = apiSnapshot - } - - result.RequestsByDay = make(map[string]int64, len(s.requestsByDay)) - for k, v := range s.requestsByDay { - result.RequestsByDay[k] = v - } - - result.RequestsByHour = make(map[string]int64, len(s.requestsByHour)) - for hour, v := range s.requestsByHour { - key := formatHour(hour) - result.RequestsByHour[key] = v - } - - result.TokensByDay = make(map[string]int64, len(s.tokensByDay)) - for k, v := range s.tokensByDay { - result.TokensByDay[k] = v - } - - result.TokensByHour = make(map[string]int64, len(s.tokensByHour)) - for hour, v := range s.tokensByHour { - key := formatHour(hour) - result.TokensByHour[key] = v - } - - return result -} - -type MergeResult struct { - Added int64 `json:"added"` - Skipped int64 `json:"skipped"` -} - -// MergeSnapshot merges an exported statistics snapshot into the current store. -// Existing data is preserved and duplicate request details are skipped. -func (s *RequestStatistics) MergeSnapshot(snapshot StatisticsSnapshot) MergeResult { - result := MergeResult{} - if s == nil { - return result - } - - s.mu.Lock() - defer s.mu.Unlock() - - seen := make(map[string]struct{}) - for apiName, stats := range s.apis { - if stats == nil { - continue - } - for modelName, modelStatsValue := range stats.Models { - if modelStatsValue == nil { - continue - } - for _, detail := range modelStatsValue.Details { - seen[dedupKey(apiName, modelName, detail)] = struct{}{} - } - } - } - - for apiName, apiSnapshot := range snapshot.APIs { - apiName = strings.TrimSpace(apiName) - if apiName == "" { - continue - } - stats, ok := s.apis[apiName] - if !ok || stats == nil { - stats = &apiStats{Models: make(map[string]*modelStats)} - s.apis[apiName] = stats - } else if stats.Models == nil { - stats.Models = make(map[string]*modelStats) - } - for modelName, modelSnapshot := range apiSnapshot.Models { - modelName = strings.TrimSpace(modelName) - if modelName == "" { - modelName = "unknown" - } - for _, detail := range modelSnapshot.Details { - detail.Tokens = normaliseTokenStats(detail.Tokens) - if detail.LatencyMs < 0 { - detail.LatencyMs = 0 - } - if detail.Timestamp.IsZero() { - detail.Timestamp = time.Now() - } - key := dedupKey(apiName, modelName, detail) - if _, exists := seen[key]; exists { - result.Skipped++ - continue - } - seen[key] = struct{}{} - s.recordImported(apiName, modelName, stats, detail) - result.Added++ - } - } - } - - return result -} - -func (s *RequestStatistics) recordImported(apiName, modelName string, stats *apiStats, detail RequestDetail) { - totalTokens := detail.Tokens.TotalTokens - if totalTokens < 0 { - totalTokens = 0 - } - - s.totalRequests++ - if detail.Failed { - s.failureCount++ - } else { - s.successCount++ - } - s.totalTokens += totalTokens - - s.updateAPIStats(stats, modelName, detail) - - dayKey := detail.Timestamp.Format("2006-01-02") - hourKey := detail.Timestamp.Hour() - - s.requestsByDay[dayKey]++ - s.requestsByHour[hourKey]++ - s.tokensByDay[dayKey] += totalTokens - s.tokensByHour[hourKey] += totalTokens -} - -func dedupKey(apiName, modelName string, detail RequestDetail) string { - timestamp := detail.Timestamp.UTC().Format(time.RFC3339Nano) - tokens := normaliseTokenStats(detail.Tokens) - return fmt.Sprintf( - "%s|%s|%s|%s|%s|%t|%d|%d|%d|%d|%d", - apiName, - modelName, - timestamp, - detail.Source, - detail.AuthIndex, - detail.Failed, - tokens.InputTokens, - tokens.OutputTokens, - tokens.ReasoningTokens, - tokens.CachedTokens, - tokens.TotalTokens, - ) -} - -func resolveAPIIdentifier(ctx context.Context, record coreusage.Record) string { - if ctx != nil { - if endpoint := strings.TrimSpace(internallogging.GetEndpoint(ctx)); endpoint != "" { - return endpoint - } - } - if record.Provider != "" { - return record.Provider - } - return "unknown" -} - -func resolveSuccess(ctx context.Context) bool { - status := internallogging.GetResponseStatus(ctx) - if status == 0 { - return true - } - return status < httpStatusBadRequest -} - -const httpStatusBadRequest = 400 - -func normaliseDetail(detail coreusage.Detail) TokenStats { - tokens := TokenStats{ - InputTokens: detail.InputTokens, - OutputTokens: detail.OutputTokens, - ReasoningTokens: detail.ReasoningTokens, - CachedTokens: detail.CachedTokens, - TotalTokens: detail.TotalTokens, - } - if tokens.TotalTokens == 0 { - tokens.TotalTokens = detail.InputTokens + detail.OutputTokens + detail.ReasoningTokens - } - if tokens.TotalTokens == 0 { - tokens.TotalTokens = detail.InputTokens + detail.OutputTokens + detail.ReasoningTokens + detail.CachedTokens - } - return tokens -} - -func normaliseTokenStats(tokens TokenStats) TokenStats { - if tokens.TotalTokens == 0 { - tokens.TotalTokens = tokens.InputTokens + tokens.OutputTokens + tokens.ReasoningTokens - } - if tokens.TotalTokens == 0 { - tokens.TotalTokens = tokens.InputTokens + tokens.OutputTokens + tokens.ReasoningTokens + tokens.CachedTokens - } - return tokens -} - -func normaliseLatency(latency time.Duration) int64 { - if latency <= 0 { - return 0 - } - return latency.Milliseconds() -} - -func formatHour(hour int) string { - if hour < 0 { - hour = 0 - } - hour = hour % 24 - return fmt.Sprintf("%02d", hour) -} diff --git a/internal/usage/logger_plugin_test.go b/internal/usage/logger_plugin_test.go deleted file mode 100644 index 842b3f0c..00000000 --- a/internal/usage/logger_plugin_test.go +++ /dev/null @@ -1,96 +0,0 @@ -package usage - -import ( - "context" - "testing" - "time" - - coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" -) - -func TestRequestStatisticsRecordIncludesLatency(t *testing.T) { - stats := NewRequestStatistics() - stats.Record(context.Background(), coreusage.Record{ - APIKey: "test-key", - Model: "gpt-5.4", - RequestedAt: time.Date(2026, 3, 20, 12, 0, 0, 0, time.UTC), - Latency: 1500 * time.Millisecond, - Detail: coreusage.Detail{ - InputTokens: 10, - OutputTokens: 20, - TotalTokens: 30, - }, - }) - - snapshot := stats.Snapshot() - details := snapshot.APIs["test-key"].Models["gpt-5.4"].Details - if len(details) != 1 { - t.Fatalf("details len = %d, want 1", len(details)) - } - if details[0].LatencyMs != 1500 { - t.Fatalf("latency_ms = %d, want 1500", details[0].LatencyMs) - } -} - -func TestRequestStatisticsMergeSnapshotDedupIgnoresLatency(t *testing.T) { - stats := NewRequestStatistics() - timestamp := time.Date(2026, 3, 20, 12, 0, 0, 0, time.UTC) - first := StatisticsSnapshot{ - APIs: map[string]APISnapshot{ - "test-key": { - Models: map[string]ModelSnapshot{ - "gpt-5.4": { - Details: []RequestDetail{{ - Timestamp: timestamp, - LatencyMs: 0, - Source: "user@example.com", - AuthIndex: "0", - Tokens: TokenStats{ - InputTokens: 10, - OutputTokens: 20, - TotalTokens: 30, - }, - }}, - }, - }, - }, - }, - } - second := StatisticsSnapshot{ - APIs: map[string]APISnapshot{ - "test-key": { - Models: map[string]ModelSnapshot{ - "gpt-5.4": { - Details: []RequestDetail{{ - Timestamp: timestamp, - LatencyMs: 2500, - Source: "user@example.com", - AuthIndex: "0", - Tokens: TokenStats{ - InputTokens: 10, - OutputTokens: 20, - TotalTokens: 30, - }, - }}, - }, - }, - }, - }, - } - - result := stats.MergeSnapshot(first) - if result.Added != 1 || result.Skipped != 0 { - t.Fatalf("first merge = %+v, want added=1 skipped=0", result) - } - - result = stats.MergeSnapshot(second) - if result.Added != 0 || result.Skipped != 1 { - t.Fatalf("second merge = %+v, want added=0 skipped=1", result) - } - - snapshot := stats.Snapshot() - details := snapshot.APIs["test-key"].Models["gpt-5.4"].Details - if len(details) != 1 { - t.Fatalf("details len = %d, want 1", len(details)) - } -} diff --git a/sdk/cliproxy/service.go b/sdk/cliproxy/service.go index d9613150..9f195f56 100644 --- a/sdk/cliproxy/service.go +++ b/sdk/cliproxy/service.go @@ -16,7 +16,6 @@ import ( _ "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" "github.com/router-for-me/CLIProxyAPI/v6/internal/registry" "github.com/router-for-me/CLIProxyAPI/v6/internal/runtime/executor" - _ "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" "github.com/router-for-me/CLIProxyAPI/v6/internal/watcher" "github.com/router-for-me/CLIProxyAPI/v6/internal/wsrelay" sdkaccess "github.com/router-for-me/CLIProxyAPI/v6/sdk/access" diff --git a/test/usage_logging_test.go b/test/usage_logging_test.go index 41c2ee34..ee03c4d7 100644 --- a/test/usage_logging_test.go +++ b/test/usage_logging_test.go @@ -2,6 +2,7 @@ package test import ( "context" + "encoding/json" "fmt" "net/http" "net/http/httptest" @@ -9,14 +10,14 @@ import ( "time" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" + "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" runtimeexecutor "github.com/router-for-me/CLIProxyAPI/v6/internal/runtime/executor" - internalusage "github.com/router-for-me/CLIProxyAPI/v6/internal/usage" cliproxyauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" cliproxyexecutor "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/executor" sdktranslator "github.com/router-for-me/CLIProxyAPI/v6/sdk/translator" ) -func TestGeminiExecutorRecordsSuccessfulZeroUsageInStatistics(t *testing.T) { +func TestGeminiExecutorRecordsSuccessfulZeroUsageInQueue(t *testing.T) { model := fmt.Sprintf("gemini-2.5-flash-zero-usage-%d", time.Now().UnixNano()) source := fmt.Sprintf("zero-usage-%d@example.com", time.Now().UnixNano()) @@ -42,10 +43,15 @@ func TestGeminiExecutorRecordsSuccessfulZeroUsageInStatistics(t *testing.T) { }, } - prevStatsEnabled := internalusage.StatisticsEnabled() - internalusage.SetStatisticsEnabled(true) + prevQueueEnabled := redisqueue.Enabled() + prevUsageEnabled := redisqueue.UsageStatisticsEnabled() + redisqueue.SetEnabled(false) + redisqueue.SetEnabled(true) + redisqueue.SetUsageStatisticsEnabled(true) t.Cleanup(func() { - internalusage.SetStatisticsEnabled(prevStatsEnabled) + redisqueue.SetEnabled(false) + redisqueue.SetEnabled(prevQueueEnabled) + redisqueue.SetUsageStatisticsEnabled(prevUsageEnabled) }) _, err := executor.Execute(context.Background(), auth, cliproxyexecutor.Request{ @@ -59,39 +65,58 @@ func TestGeminiExecutorRecordsSuccessfulZeroUsageInStatistics(t *testing.T) { t.Fatalf("Execute error: %v", err) } - detail := waitForStatisticsDetail(t, "gemini", model, source) - if detail.Failed { - t.Fatalf("detail failed = true, want false") - } - if detail.Tokens.TotalTokens != 0 { - t.Fatalf("total tokens = %d, want 0", detail.Tokens.TotalTokens) - } + waitForQueuedUsageModelTotalTokens(t, "gemini", model, 0) } -func waitForStatisticsDetail(t *testing.T, apiName, model, source string) internalusage.RequestDetail { +func waitForQueuedUsageModelTotalTokens(t *testing.T, wantProvider, wantModel string, wantTokens int64) { t.Helper() deadline := time.Now().Add(2 * time.Second) for time.Now().Before(deadline) { - snapshot := internalusage.GetRequestStatistics().Snapshot() - apiSnapshot, ok := snapshot.APIs[apiName] - if !ok { - time.Sleep(10 * time.Millisecond) - continue - } - modelSnapshot, ok := apiSnapshot.Models[model] - if !ok { - time.Sleep(10 * time.Millisecond) - continue - } - for _, detail := range modelSnapshot.Details { - if detail.Source == source { - return detail + items := redisqueue.PopOldest(10) + for _, item := range items { + got, ok := parseQueuedUsagePayload(t, item) + if !ok { + continue } + if got.Provider != wantProvider || got.Model != wantModel { + continue + } + if got.Failed { + t.Fatalf("payload failed = true, want false") + } + if got.Tokens.TotalTokens != wantTokens { + t.Fatalf("payload total tokens = %d, want %d", got.Tokens.TotalTokens, wantTokens) + } + return } time.Sleep(10 * time.Millisecond) } - t.Fatalf("timed out waiting for statistics detail for api=%q model=%q source=%q", apiName, model, source) - return internalusage.RequestDetail{} + t.Fatalf("timed out waiting for queued usage payload for provider=%q model=%q", wantProvider, wantModel) +} + +type queuedUsagePayload struct { + Provider string `json:"provider"` + Model string `json:"model"` + Failed bool `json:"failed"` + Tokens struct { + TotalTokens int64 `json:"total_tokens"` + } `json:"tokens"` +} + +func parseQueuedUsagePayload(t *testing.T, payload []byte) (queuedUsagePayload, bool) { + t.Helper() + + var parsed queuedUsagePayload + if len(payload) == 0 { + return parsed, false + } + if err := json.Unmarshal(payload, &parsed); err != nil { + return parsed, false + } + if parsed.Provider == "" || parsed.Model == "" { + return parsed, false + } + return parsed, true } From 79579c34bf9ea72f51ccaea53908741d84d05829 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 13:35:19 +0800 Subject: [PATCH 23/51] docs: update README to consolidate and clarify CPA Usage Keeper details - Moved CPA Usage Keeper from "CLI tools" to a dedicated "Usage Statistics" section. - Added details on its functionality, periodic data sync, SQLite storage, and built-in dashboard. - Applied updates across English, Chinese, and Japanese README files for consistency. --- README.md | 12 ++++++++---- README_CN.md | 12 ++++++++---- README_JA.md | 12 ++++++++---- 3 files changed, 24 insertions(+), 12 deletions(-) diff --git a/README.md b/README.md index 93ef6f71..3958668e 100644 --- a/README.md +++ b/README.md @@ -74,6 +74,14 @@ CLIProxyAPI Guides: [https://help.router-for.me/](https://help.router-for.me/) see [MANAGEMENT_API.md](https://help.router-for.me/management/api) +## Usage Statistics + +Since v6.10.0, CLIProxyAPI and [CPAMC](https://github.com/router-for-me/Cli-Proxy-API-Management-Center) no longer ship built-in usage statistics. If you need usage statistics, use: + +### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) + +Standalone persistence and visualization service for CLIProxyAPI, with periodic data sync, SQLite storage, aggregate APIs, and a built-in dashboard for usage and statistics. + ## Amp CLI Support CLIProxyAPI includes integrated support for [Amp CLI](https://ampcode.com) and Amp IDE extensions, enabling you to use your Google/ChatGPT/Claude OAuth subscriptions with Amp's coding tools: @@ -183,10 +191,6 @@ Cross-platform desktop app (macOS, Windows, Linux) wrapping CLIProxyAPI with a n Ready-to-use cross-platform quota inspector for CLIProxyAPI, supporting per-account codex 5h/7d quota windows, plan-based sorting, status coloring, and multi-account summary analytics. -### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) - -Standalone persistence and visualization service for CLIProxyAPI, with periodic data sync, SQLite storage, aggregate APIs, and a built-in dashboard for usage and statistics. - ### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) Windows-focused, local-first desktop management platform for Codex CLI built on CLIProxyAPI, focused on simplifying local setup, account and runtime management, and providing a more complete Codex CLI experience for local users. diff --git a/README_CN.md b/README_CN.md index 6199095c..5c341d27 100644 --- a/README_CN.md +++ b/README_CN.md @@ -74,6 +74,14 @@ CLIProxyAPI 用户手册: [https://help.router-for.me/](https://help.router-fo 请参见 [MANAGEMENT_API_CN.md](https://help.router-for.me/cn/management/api) +## 使用量统计 + +自v6.10.0版本以后,CLIProxyAPI及 [CPAMC](https://github.com/router-for-me/Cli-Proxy-API-Management-Center) 项目不再预置数据统计功能,如果有数据统计需求的请使用以下项目: + +### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) + +独立的 CLIProxyAPI 使用量持久化与可视化服务,定期同步 CLIProxyAPI 数据,存储到 SQLite,提供聚合 API,并内置使用量分析与统计仪表盘。 + ## Amp CLI 支持 CLIProxyAPI 已内置对 [Amp CLI](https://ampcode.com) 和 Amp IDE 扩展的支持,可让你使用自己的 Google/ChatGPT/Claude OAuth 订阅来配合 Amp 编码工具: @@ -179,10 +187,6 @@ Shadow AI 是一款专为受限环境设计的 AI 辅助工具。提供无窗口 上手即用的面向 CLIProxyAPI 跨平台配额查询工具,支持按账号展示 codex 5h/7d 配额窗口、按计划排序、状态着色及多账号汇总分析。 -### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) - -独立的 CLIProxyAPI 使用量持久化与可视化服务,定期同步 CPA 数据,存储到 SQLite,提供聚合 API,并内置使用量分析与统计仪表盘。 - ### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) 基于 CLIProxyAPI 的 Windows Codex CLI 本地优先桌面管理平台,聚焦简化本机配置、账号与运行状态管理,并为本地用户提供更完整的 Codex CLI 使用体验。 diff --git a/README_JA.md b/README_JA.md index 1bb30d48..cbb37767 100644 --- a/README_JA.md +++ b/README_JA.md @@ -72,6 +72,14 @@ CLIProxyAPIガイド:[https://help.router-for.me/](https://help.router-for.me/ [MANAGEMENT_API.md](https://help.router-for.me/management/api)を参照 +## 使用量統計 + +v6.10.0以降、CLIProxyAPIおよび [CPAMC](https://github.com/router-for-me/Cli-Proxy-API-Management-Center) プロジェクトには使用量統計機能がプリセットされなくなりました。使用量統計が必要な場合は、次のプロジェクトをご利用ください: + +### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) + +CLIProxyAPI向けの独立した使用量永続化・可視化サービス。CLIProxyAPIデータを定期同期してSQLiteに保存し、集計APIと、使用量や各種統計を確認できる組み込みダッシュボードを提供します。 + ## Amp CLIサポート CLIProxyAPIは[Amp CLI](https://ampcode.com)およびAmp IDE拡張機能の統合サポートを含んでおり、Google/ChatGPT/ClaudeのOAuthサブスクリプションをAmpのコーディングツールで使用できます: @@ -178,10 +186,6 @@ CLIProxyAPIをネイティブGUIでラップしたクロスプラットフォー CLIProxyAPI向けのすぐに使えるクロスプラットフォームのクォータ確認ツール。アカウントごとの codex 5h/7d クォータ表示、プラン別ソート、ステータス色分け、複数アカウントの集計分析に対応。 -### [CPA Usage Keeper](https://github.com/Willxup/cpa-usage-keeper) - -CLIProxyAPI向けの独立した使用量永続化・可視化サービス。CPAデータを定期同期してSQLiteに保存し、集計APIと、使用量や各種統計を確認できる組み込みダッシュボードを提供します。 - ### [CodexCliPlus](https://github.com/C4AL/CodexCliPlus) CLIProxyAPIを基盤にしたWindows向けのローカル優先Codex CLIデスクトップ管理プラットフォーム。ローカル設定、アカウント、実行状態の管理を簡素化し、ローカルユーザーにより包括的なCodex CLI体験を提供します。 From 2efa56dbb8191f02a0c43aee0075fa04cb899775 Mon Sep 17 00:00:00 2001 From: daishuge Date: Sat, 2 May 2026 15:34:57 +0800 Subject: [PATCH 24/51] docs: add Playful Proxy API Panel --- README.md | 4 ++++ README_CN.md | 4 ++++ README_JA.md | 4 ++++ 3 files changed, 12 insertions(+) diff --git a/README.md b/README.md index 3958668e..47ea6909 100644 --- a/README.md +++ b/README.md @@ -212,6 +212,10 @@ Never stop coding. Smart routing to FREE & low-cost AI models with automatic fal OmniRoute is an AI gateway for multi-provider LLMs: an OpenAI-compatible endpoint with smart routing, load balancing, retries, and fallbacks. Add policies, rate limits, caching, and observability for reliable, cost-aware inference. +### [Playful Proxy API Panel (PPAP)](https://github.com/daishuge/playful-proxy-api-panel) + +A public CLIProxyAPI-compatible fork and bundled management panel. It keeps upstream-style usage while restoring built-in usage statistics, adding cache hit rate, first-byte latency, TPS tracking, and Docker-oriented self-hosted installation docs. + > [!NOTE] > If you have developed a port of CLIProxyAPI or a project inspired by it, please open a PR to add it to this list. diff --git a/README_CN.md b/README_CN.md index 5c341d27..e9b9c2a4 100644 --- a/README_CN.md +++ b/README_CN.md @@ -208,6 +208,10 @@ Shadow AI 是一款专为受限环境设计的 AI 辅助工具。提供无窗口 OmniRoute 是一个面向多供应商大语言模型的 AI 网关:它提供兼容 OpenAI 的端点,具备智能路由、负载均衡、重试及回退机制。通过添加策略、速率限制、缓存和可观测性,确保推理过程既可靠又具备成本意识。 +### [Playful Proxy API Panel (PPAP)](https://github.com/daishuge/playful-proxy-api-panel) + +一个公开的 CLIProxyAPI 兼容二开版本和配套管理面板,尽量保持与上游一致的使用方式,同时恢复内置使用量统计,并补充缓存命中率、首字响应时间、TPS 记录和面向 Docker 自托管的安装说明。 + > [!NOTE] > 如果你开发了 CLIProxyAPI 的移植或衍生项目,请提交 PR 将其添加到此列表中。 diff --git a/README_JA.md b/README_JA.md index cbb37767..58ad22cf 100644 --- a/README_JA.md +++ b/README_JA.md @@ -207,6 +207,10 @@ CLIProxyAPIに触発されたNext.js実装。インストールと使用が簡 OmniRouteはマルチプロバイダーLLM向けのAIゲートウェイです:スマートルーティング、負荷分散、リトライ、フォールバックを備えたOpenAI互換エンドポイント。ポリシー、レート制限、キャッシュ、可観測性を追加して、信頼性が高くコストを意識した推論を実現します。 +### [Playful Proxy API Panel (PPAP)](https://github.com/daishuge/playful-proxy-api-panel) + +上流に近い使い方を維持する公開CLIProxyAPI互換フォーク兼管理パネルです。内蔵の使用量統計を復元し、キャッシュヒット率、初回バイト待ち時間、TPSの記録、Docker向けのセルフホスト手順を追加しています。 + > [!NOTE] > CLIProxyAPIの移植版またはそれに触発されたプロジェクトを開発した場合は、PRを送ってこのリストに追加してください。 From 56df36895a0ed21720a3aa315f5b394f8b20b1b3 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sat, 2 May 2026 20:43:16 +0800 Subject: [PATCH 25/51] feat: add configurable retention period for Redis usage queue - Introduced `redis-usage-queue-retention-seconds` config parameter with a default of 60 seconds and a max of 3600 seconds. - Updated logic in `redisqueue` to honor configurable retention periods for enqueued usage data. - Modified config validation and initialization to support and enforce retention limits. - Enhanced change tracking in `config_diff` to detect updates to this parameter. --- cmd/server/main.go | 1 + config.example.yaml | 4 ++++ internal/api/server.go | 4 ++++ internal/config/config.go | 13 ++++++++++++ internal/redisqueue/queue.go | 30 ++++++++++++++++++++++++---- internal/watcher/diff/config_diff.go | 3 +++ 6 files changed, 51 insertions(+), 4 deletions(-) diff --git a/cmd/server/main.go b/cmd/server/main.go index e735b144..b10bc9c8 100644 --- a/cmd/server/main.go +++ b/cmd/server/main.go @@ -418,6 +418,7 @@ func main() { } } redisqueue.SetUsageStatisticsEnabled(cfg.UsageStatisticsEnabled) + redisqueue.SetRetentionSeconds(cfg.RedisUsageQueueRetentionSeconds) coreauth.SetQuotaCooldownDisabled(cfg.DisableCooling) if err = logging.ConfigureLogOutput(cfg); err != nil { diff --git a/config.example.yaml b/config.example.yaml index 172e961f..d7d5a9f5 100644 --- a/config.example.yaml +++ b/config.example.yaml @@ -66,6 +66,10 @@ error-logs-max-files: 10 # When false, disable in-memory usage statistics aggregation usage-statistics-enabled: false +# How long (in seconds) Redis usage queue items are retained in memory for the RESP interface (LPOP/RPOP). +# Default: 60. Max: 3600. +redis-usage-queue-retention-seconds: 60 + # Proxy URL. Supports socks5/http/https protocols. Example: socks5://user:pass@192.168.1.1:1080/ # Per-entry proxy-url also supports "direct" or "none" to bypass both the global proxy-url and environment proxies explicitly. proxy-url: "" diff --git a/internal/api/server.go b/internal/api/server.go index 176bc2a3..2e89ac5a 100644 --- a/internal/api/server.go +++ b/internal/api/server.go @@ -1000,6 +1000,10 @@ func (s *Server) UpdateClients(cfg *config.Config) { redisqueue.SetUsageStatisticsEnabled(cfg.UsageStatisticsEnabled) } + if oldCfg == nil || oldCfg.RedisUsageQueueRetentionSeconds != cfg.RedisUsageQueueRetentionSeconds { + redisqueue.SetRetentionSeconds(cfg.RedisUsageQueueRetentionSeconds) + } + if s.requestLogger != nil && (oldCfg == nil || oldCfg.ErrorLogsMaxFiles != cfg.ErrorLogsMaxFiles) { if setter, ok := s.requestLogger.(interface{ SetErrorLogsMaxFiles(int) }); ok { setter.SetErrorLogsMaxFiles(cfg.ErrorLogsMaxFiles) diff --git a/internal/config/config.go b/internal/config/config.go index 39c91127..46ce4f50 100644 --- a/internal/config/config.go +++ b/internal/config/config.go @@ -65,6 +65,11 @@ type Config struct { // UsageStatisticsEnabled toggles in-memory usage aggregation; when false, usage data is discarded. UsageStatisticsEnabled bool `yaml:"usage-statistics-enabled" json:"usage-statistics-enabled"` + // RedisUsageQueueRetentionSeconds controls how long (in seconds) usage queue items + // are retained in memory for the Redis RESP interface (LPOP/RPOP). + // Default: 60. Max: 3600. + RedisUsageQueueRetentionSeconds int `yaml:"redis-usage-queue-retention-seconds" json:"redis-usage-queue-retention-seconds"` + // DisableCooling disables quota cooldown scheduling when true. DisableCooling bool `yaml:"disable-cooling" json:"disable-cooling"` @@ -609,6 +614,7 @@ func LoadConfigOptional(configFile string, optional bool) (*Config, error) { cfg.LogsMaxTotalSizeMB = 0 cfg.ErrorLogsMaxFiles = 10 cfg.UsageStatisticsEnabled = false + cfg.RedisUsageQueueRetentionSeconds = 60 cfg.DisableCooling = false cfg.DisableImageGeneration = DisableImageGenerationOff cfg.Pprof.Enable = false @@ -671,6 +677,13 @@ func LoadConfigOptional(configFile string, optional bool) (*Config, error) { cfg.ErrorLogsMaxFiles = 10 } + if cfg.RedisUsageQueueRetentionSeconds <= 0 { + cfg.RedisUsageQueueRetentionSeconds = 60 + } else if cfg.RedisUsageQueueRetentionSeconds > 3600 { + log.WithField("value", cfg.RedisUsageQueueRetentionSeconds).Warn("redis-usage-queue-retention-seconds too large; clamping to 3600") + cfg.RedisUsageQueueRetentionSeconds = 3600 + } + if cfg.MaxRetryCredentials < 0 { cfg.MaxRetryCredentials = 0 } diff --git a/internal/redisqueue/queue.go b/internal/redisqueue/queue.go index 8a4b6742..2fea5839 100644 --- a/internal/redisqueue/queue.go +++ b/internal/redisqueue/queue.go @@ -6,7 +6,10 @@ import ( "time" ) -const retentionWindow = time.Minute +const ( + defaultRetentionSeconds int64 = 60 + maxRetentionSeconds int64 = 3600 +) type queueItem struct { enqueuedAt time.Time @@ -20,10 +23,15 @@ type queue struct { } var ( - enabled atomic.Bool - global queue + enabled atomic.Bool + retentionSeconds atomic.Int64 + global queue ) +func init() { + retentionSeconds.Store(defaultRetentionSeconds) +} + func SetEnabled(value bool) { enabled.Store(value) if !value { @@ -35,6 +43,16 @@ func Enabled() bool { return enabled.Load() } +func SetRetentionSeconds(value int) { + normalized := int64(value) + if normalized <= 0 { + normalized = defaultRetentionSeconds + } else if normalized > maxRetentionSeconds { + normalized = maxRetentionSeconds + } + retentionSeconds.Store(normalized) +} + func Enqueue(payload []byte) { if !Enabled() { return @@ -110,7 +128,11 @@ func (q *queue) pruneLocked(now time.Time) { return } - cutoff := now.Add(-retentionWindow) + windowSeconds := retentionSeconds.Load() + if windowSeconds <= 0 { + windowSeconds = defaultRetentionSeconds + } + cutoff := now.Add(-time.Duration(windowSeconds) * time.Second) for q.head < len(q.items) && q.items[q.head].enqueuedAt.Before(cutoff) { q.head++ } diff --git a/internal/watcher/diff/config_diff.go b/internal/watcher/diff/config_diff.go index 2be9aa90..b414ed5a 100644 --- a/internal/watcher/diff/config_diff.go +++ b/internal/watcher/diff/config_diff.go @@ -39,6 +39,9 @@ func BuildConfigChangeDetails(oldCfg, newCfg *config.Config) []string { if oldCfg.UsageStatisticsEnabled != newCfg.UsageStatisticsEnabled { changes = append(changes, fmt.Sprintf("usage-statistics-enabled: %t -> %t", oldCfg.UsageStatisticsEnabled, newCfg.UsageStatisticsEnabled)) } + if oldCfg.RedisUsageQueueRetentionSeconds != newCfg.RedisUsageQueueRetentionSeconds { + changes = append(changes, fmt.Sprintf("redis-usage-queue-retention-seconds: %d -> %d", oldCfg.RedisUsageQueueRetentionSeconds, newCfg.RedisUsageQueueRetentionSeconds)) + } if oldCfg.DisableCooling != newCfg.DisableCooling { changes = append(changes, fmt.Sprintf("disable-cooling: %t -> %t", oldCfg.DisableCooling, newCfg.DisableCooling)) } From 101b59cfe872778f5915b0eef0ccac0eec79cae4 Mon Sep 17 00:00:00 2001 From: Vijay Chimmi Date: Sat, 2 May 2026 17:37:38 -0700 Subject: [PATCH 26/51] docs: update Subtitle Translator project description --- README.md | 2 +- README_CN.md | 2 +- README_JA.md | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index 47ea6909..91f38239 100644 --- a/README.md +++ b/README.md @@ -130,7 +130,7 @@ Native macOS menu bar app to use your Claude Code & ChatGPT subscriptions with A ### [Subtitle Translator](https://github.com/VjayC/SRT-Subtitle-Translator-Validator) -Browser-based tool to translate SRT subtitles using your Gemini subscription via CLIProxyAPI with automatic validation/error correction - no API keys needed +A cross-platform desktop and web app to translate and validate SRT subtitles using your existing LLM subscriptions (Gemini, ChatGPT, Claude, etc.) via CLIProxyAPI - no API keys needed. ### [CCS (Claude Code Switch)](https://github.com/kaitranntt/ccs) diff --git a/README_CN.md b/README_CN.md index e9b9c2a4..a307dc95 100644 --- a/README_CN.md +++ b/README_CN.md @@ -129,7 +129,7 @@ CLIProxyAPI 已内置对 [Amp CLI](https://ampcode.com) 和 Amp IDE 扩展的支 ### [Subtitle Translator](https://github.com/VjayC/SRT-Subtitle-Translator-Validator) -一款基于浏览器的 SRT 字幕翻译工具,可通过 CLI 代理 API 使用您的 Gemini 订阅。内置自动验证与错误修正功能,无需 API 密钥。 +一款跨平台的桌面和 Web 应用程序,可通过 CLIProxyAPI 使用您现有的 LLM 订阅(Gemini、ChatGPT、Claude, etc.)来翻译和验证 SRT 字幕 - 无需 API 密钥。 ### [CCS (Claude Code Switch)](https://github.com/kaitranntt/ccs) diff --git a/README_JA.md b/README_JA.md index 58ad22cf..266b612a 100644 --- a/README_JA.md +++ b/README_JA.md @@ -128,7 +128,7 @@ macOSネイティブのメニューバーアプリで、Claude CodeとChatGPTの ### [Subtitle Translator](https://github.com/VjayC/SRT-Subtitle-Translator-Validator) -CLIProxyAPI経由でGeminiサブスクリプションを使用してSRT字幕を翻訳するブラウザベースのツール。自動検証/エラー修正機能付き - APIキー不要 +CLIProxyAPI経由で既存のLLMサブスクリプション(Gemini、ChatGPT、Claude, etc.)を使用してSRT字幕を翻訳および検証する、クロスプラットフォームのデスクトップおよびWebアプリ - APIキー不要。 ### [CCS (Claude Code Switch)](https://github.com/kaitranntt/ccs) From 5fc6f662e14a30be75d3994b3c64270d04358593 Mon Sep 17 00:00:00 2001 From: ziwu Date: Sun, 3 May 2026 18:25:11 +0800 Subject: [PATCH 27/51] docs: add CLIProxy Pool Watch project --- README.md | 4 ++++ README_CN.md | 4 ++++ README_JA.md | 4 ++++ 3 files changed, 12 insertions(+) diff --git a/README.md b/README.md index 91f38239..e404e894 100644 --- a/README.md +++ b/README.md @@ -195,6 +195,10 @@ Ready-to-use cross-platform quota inspector for CLIProxyAPI, supporting per-acco Windows-focused, local-first desktop management platform for Codex CLI built on CLIProxyAPI, focused on simplifying local setup, account and runtime management, and providing a more complete Codex CLI experience for local users. +### [CLIProxy Pool Watch](https://github.com/murasame612/CLIProxyPoolWidget) + +Native macOS SwiftUI app for monitoring ChatGPT/Codex account quotas in CLIProxyAPI pools. Displays account availability, Plus-base capacity, 5-hour and weekly quota bars, plan weights, and restore forecasts through the Management API. + > [!NOTE] > If you developed a project based on CLIProxyAPI, please open a PR to add it to this list. diff --git a/README_CN.md b/README_CN.md index a307dc95..e5d9db1e 100644 --- a/README_CN.md +++ b/README_CN.md @@ -191,6 +191,10 @@ Shadow AI 是一款专为受限环境设计的 AI 辅助工具。提供无窗口 基于 CLIProxyAPI 的 Windows Codex CLI 本地优先桌面管理平台,聚焦简化本机配置、账号与运行状态管理,并为本地用户提供更完整的 Codex CLI 使用体验。 +### [CLIProxy Pool Watch](https://github.com/murasame612/CLIProxyPoolWidget) + +原生 macOS SwiftUI 应用,用于监控 CLIProxyAPI 池中的 ChatGPT/Codex 账号额度。通过 Management API 展示账号可用状态、Plus 基准容量、5 小时与周额度进度条、套餐权重和恢复预测。 + > [!NOTE] > 如果你开发了基于 CLIProxyAPI 的项目,请提交一个 PR(拉取请求)将其添加到此列表中。 diff --git a/README_JA.md b/README_JA.md index 266b612a..84816641 100644 --- a/README_JA.md +++ b/README_JA.md @@ -190,6 +190,10 @@ CLIProxyAPI向けのすぐに使えるクロスプラットフォームのクォ CLIProxyAPIを基盤にしたWindows向けのローカル優先Codex CLIデスクトップ管理プラットフォーム。ローカル設定、アカウント、実行状態の管理を簡素化し、ローカルユーザーにより包括的なCodex CLI体験を提供します。 +### [CLIProxy Pool Watch](https://github.com/murasame612/CLIProxyPoolWidget) + +CLIProxyAPIプール内のChatGPT/Codexアカウントクォータを監視するmacOSネイティブSwiftUIアプリ。Management APIを通じて、アカウントの可用性、Plus基準の容量、5時間/週次クォータバー、プラン重み、復元予測を表示します。 + > [!NOTE] > CLIProxyAPIをベースにプロジェクトを開発した場合は、PRを送ってこのリストに追加してください。 From 81db7fdc1e06b88c6fe9d14e9f1dfb57d5c642d1 Mon Sep 17 00:00:00 2001 From: John Date: Sun, 3 May 2026 20:23:23 +0800 Subject: [PATCH 28/51] Add CLIProxyAPI Usage Dashboard to statistics docs --- README.md | 4 ++++ README_CN.md | 4 ++++ README_JA.md | 4 ++++ 3 files changed, 12 insertions(+) diff --git a/README.md b/README.md index 91f38239..f5bcc4ee 100644 --- a/README.md +++ b/README.md @@ -82,6 +82,10 @@ Since v6.10.0, CLIProxyAPI and [CPAMC](https://github.com/router-for-me/Cli-Prox Standalone persistence and visualization service for CLIProxyAPI, with periodic data sync, SQLite storage, aggregate APIs, and a built-in dashboard for usage and statistics. +### [CLIProxyAPI Usage Dashboard](https://github.com/zhanglunet/cliproxyapi-usage-dashboard) + +Local-first usage and quota dashboard for CLIProxyAPI. It collects per-request token usage from the Redis-compatible usage queue into SQLite, visualizes daily and recent-window usage by account and model, and shows Codex 5h/7d quota remaining in a local web UI. + ## Amp CLI Support CLIProxyAPI includes integrated support for [Amp CLI](https://ampcode.com) and Amp IDE extensions, enabling you to use your Google/ChatGPT/Claude OAuth subscriptions with Amp's coding tools: diff --git a/README_CN.md b/README_CN.md index a307dc95..10bba6e3 100644 --- a/README_CN.md +++ b/README_CN.md @@ -82,6 +82,10 @@ CLIProxyAPI 用户手册: [https://help.router-for.me/](https://help.router-fo 独立的 CLIProxyAPI 使用量持久化与可视化服务,定期同步 CLIProxyAPI 数据,存储到 SQLite,提供聚合 API,并内置使用量分析与统计仪表盘。 +### [CLIProxyAPI Usage Dashboard](https://github.com/zhanglunet/cliproxyapi-usage-dashboard) + +面向 CLIProxyAPI 的本地优先使用量与配额看板。它从 Redis 兼容使用量队列采集每次请求的 token 消耗并写入 SQLite,按账号和模型展示当天及最近时间窗口的用量,并在本地网页中显示 Codex 5h/7d 配额余量。 + ## Amp CLI 支持 CLIProxyAPI 已内置对 [Amp CLI](https://ampcode.com) 和 Amp IDE 扩展的支持,可让你使用自己的 Google/ChatGPT/Claude OAuth 订阅来配合 Amp 编码工具: diff --git a/README_JA.md b/README_JA.md index 266b612a..d5638173 100644 --- a/README_JA.md +++ b/README_JA.md @@ -80,6 +80,10 @@ v6.10.0以降、CLIProxyAPIおよび [CPAMC](https://github.com/router-for-me/Cl CLIProxyAPI向けの独立した使用量永続化・可視化サービス。CLIProxyAPIデータを定期同期してSQLiteに保存し、集計APIと、使用量や各種統計を確認できる組み込みダッシュボードを提供します。 +### [CLIProxyAPI Usage Dashboard](https://github.com/zhanglunet/cliproxyapi-usage-dashboard) + +CLIProxyAPI向けのローカル優先の使用量・クォータダッシュボード。Redis互換の使用量キューからリクエストごとのtoken使用量を収集してSQLiteに保存し、アカウント別・モデル別の日次および直近時間枠の使用量と、Codex 5h/7dクォータ残量をローカルWeb UIで表示します。 + ## Amp CLIサポート CLIProxyAPIは[Amp CLI](https://ampcode.com)およびAmp IDE拡張機能の統合サポートを含んでおり、Google/ChatGPT/ClaudeのOAuthサブスクリプションをAmpのコーディングツールで使用できます: From 7972130513c2f234be29dd733a33ec7c48f6a54c Mon Sep 17 00:00:00 2001 From: zhanglu <1160377+zhanglunet@users.noreply.github.com> Date: Sun, 3 May 2026 20:38:25 +0800 Subject: [PATCH 29/51] Update README_CN.md Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com> --- README_CN.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README_CN.md b/README_CN.md index 10bba6e3..6989cf3f 100644 --- a/README_CN.md +++ b/README_CN.md @@ -84,7 +84,7 @@ CLIProxyAPI 用户手册: [https://help.router-for.me/](https://help.router-fo ### [CLIProxyAPI Usage Dashboard](https://github.com/zhanglunet/cliproxyapi-usage-dashboard) -面向 CLIProxyAPI 的本地优先使用量与配额看板。它从 Redis 兼容使用量队列采集每次请求的 token 消耗并写入 SQLite,按账号和模型展示当天及最近时间窗口的用量,并在本地网页中显示 Codex 5h/7d 配额余量。 +面向 CLIProxyAPI 的本地优先使用量与配额看板。它从 Redis 兼容使用量队列采集每次请求的 Token 消耗并写入 SQLite,按账号和模型可视化每日及最近时间窗口的用量,并在本地网页中显示 Codex 5h/7d 配额余量。 ## Amp CLI 支持 From d2386a31144530c0f6aadd5330f90d8067c0a796 Mon Sep 17 00:00:00 2001 From: zhanglu <1160377+zhanglunet@users.noreply.github.com> Date: Sun, 3 May 2026 20:38:51 +0800 Subject: [PATCH 30/51] Update README_JA.md Co-authored-by: gemini-code-assist[bot] <176961590+gemini-code-assist[bot]@users.noreply.github.com> --- README_JA.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README_JA.md b/README_JA.md index d5638173..b07ca492 100644 --- a/README_JA.md +++ b/README_JA.md @@ -82,7 +82,7 @@ CLIProxyAPI向けの独立した使用量永続化・可視化サービス。CLI ### [CLIProxyAPI Usage Dashboard](https://github.com/zhanglunet/cliproxyapi-usage-dashboard) -CLIProxyAPI向けのローカル優先の使用量・クォータダッシュボード。Redis互換の使用量キューからリクエストごとのtoken使用量を収集してSQLiteに保存し、アカウント別・モデル別の日次および直近時間枠の使用量と、Codex 5h/7dクォータ残量をローカルWeb UIで表示します。 +CLIProxyAPI向けのローカル優先の使用量・クォータダッシュボード。Redis互換の使用量キューからリクエストごとのToken使用量を収集してSQLiteに保存し、アカウント別・モデル別の日次および直近時間枠の使用量を可視化し、Codex 5h/7dクォータ残量をローカルWeb UIで表示します。 ## Amp CLIサポート From af65908cb0e172c91f467cfd6b18b0b596ed47c4 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sun, 3 May 2026 22:26:23 +0800 Subject: [PATCH 31/51] feat: enhance tool mapping with namespace and web search support - Added functions to handle tool conversion, including namespace-based tools and web search tools. - Improved parameter normalization and tool input schema standardization. - Integrated logic to handle qualified tool names and map override functionality. - Refactored existing tool processing for better extensibility and maintainability. Fixed: #3199 --- .../claude_openai-responses_request.go | 206 ++++++++++++++++-- .../claude_openai-responses_response.go | 10 +- 2 files changed, 197 insertions(+), 19 deletions(-) diff --git a/internal/translator/claude/openai/responses/claude_openai-responses_request.go b/internal/translator/claude/openai/responses/claude_openai-responses_request.go index 514129ca..c0479b87 100644 --- a/internal/translator/claude/openai/responses/claude_openai-responses_request.go +++ b/internal/translator/claude/openai/responses/claude_openai-responses_request.go @@ -339,25 +339,21 @@ func ConvertOpenAIResponsesRequestToClaude(modelName string, inputRawJSON []byte }) } + includedToolNames := map[string]struct{}{} + toolNameMap := map[string]string{} + // tools mapping: parameters -> input_schema if tools := root.Get("tools"); tools.Exists() && tools.IsArray() { toolsJSON := []byte("[]") tools.ForEach(func(_, tool gjson.Result) bool { - tJSON := []byte(`{"name":"","description":"","input_schema":{}}`) - if n := tool.Get("name"); n.Exists() { - tJSON, _ = sjson.SetBytes(tJSON, "name", n.String()) + convertedTools := convertResponsesToolToClaudeTools(tool, toolNameMap) + for _, tJSON := range convertedTools { + toolName := gjson.GetBytes(tJSON, "name").String() + if toolName != "" { + includedToolNames[toolName] = struct{}{} + } + toolsJSON, _ = sjson.SetRawBytes(toolsJSON, "-1", tJSON) } - if d := tool.Get("description"); d.Exists() { - tJSON, _ = sjson.SetBytes(tJSON, "description", d.String()) - } - - if params := tool.Get("parameters"); params.Exists() { - tJSON, _ = sjson.SetRawBytes(tJSON, "input_schema", []byte(params.Raw)) - } else if params = tool.Get("parametersJsonSchema"); params.Exists() { - tJSON, _ = sjson.SetRawBytes(tJSON, "input_schema", []byte(params.Raw)) - } - - toolsJSON, _ = sjson.SetRawBytes(toolsJSON, "-1", tJSON) return true }) if parsedTools := gjson.ParseBytes(toolsJSON); parsedTools.IsArray() && len(parsedTools.Array()) > 0 { @@ -375,14 +371,24 @@ func ConvertOpenAIResponsesRequestToClaude(modelName string, inputRawJSON []byte case "none": // Leave unset; implies no tools case "required": - out, _ = sjson.SetRawBytes(out, "tool_choice", []byte(`{"type":"any"}`)) + if len(includedToolNames) > 0 { + out, _ = sjson.SetRawBytes(out, "tool_choice", []byte(`{"type":"any"}`)) + } } case gjson.JSON: if toolChoice.Get("type").String() == "function" { fn := toolChoice.Get("function.name").String() - toolChoiceJSON := []byte(`{"name":"","type":"tool"}`) - toolChoiceJSON, _ = sjson.SetBytes(toolChoiceJSON, "name", fn) - out, _ = sjson.SetRawBytes(out, "tool_choice", toolChoiceJSON) + if fn == "" { + fn = toolChoice.Get("name").String() + } + if mappedName := toolNameMap[fn]; mappedName != "" { + fn = mappedName + } + if _, ok := includedToolNames[fn]; ok { + toolChoiceJSON := []byte(`{"name":"","type":"tool"}`) + toolChoiceJSON, _ = sjson.SetBytes(toolChoiceJSON, "name", fn) + out, _ = sjson.SetRawBytes(out, "tool_choice", toolChoiceJSON) + } } default: @@ -391,3 +397,167 @@ func ConvertOpenAIResponsesRequestToClaude(modelName string, inputRawJSON []byte return out } + +func convertResponsesToolToClaudeTools(tool gjson.Result, toolNameMap map[string]string) [][]byte { + toolType := strings.TrimSpace(tool.Get("type").String()) + switch toolType { + case "", "function": + if tJSON, ok := convertResponsesFunctionToolToClaude(tool, ""); ok { + return [][]byte{tJSON} + } + case "namespace": + return convertResponsesNamespaceToolToClaude(tool, toolNameMap) + case "web_search": + if tJSON, ok := convertResponsesWebSearchToolToClaude(tool); ok { + if name := gjson.GetBytes(tJSON, "name").String(); name != "" { + toolNameMap[name] = name + } + return [][]byte{tJSON} + } + default: + if isUnsupportedOpenAIBuiltinToolType(toolType) { + return nil + } + if tool.Get("name").String() != "" { + return [][]byte{[]byte(tool.Raw)} + } + } + return nil +} + +func convertResponsesNamespaceToolToClaude(tool gjson.Result, toolNameMap map[string]string) [][]byte { + namespaceName := strings.TrimSpace(tool.Get("name").String()) + children := tool.Get("tools") + if !children.Exists() || !children.IsArray() { + return nil + } + + var out [][]byte + children.ForEach(func(_, child gjson.Result) bool { + childName := responsesToolName(child) + qualifiedName := qualifyResponsesNamespaceToolName(namespaceName, childName) + if tJSON, ok := convertResponsesFunctionToolToClaude(child, qualifiedName); ok { + out = append(out, tJSON) + toolNameMap[qualifiedName] = qualifiedName + if childName != "" { + toolNameMap[childName] = qualifiedName + } + } + return true + }) + return out +} + +func convertResponsesFunctionToolToClaude(tool gjson.Result, overrideName string) ([]byte, bool) { + name := strings.TrimSpace(overrideName) + if name == "" { + name = responsesToolName(tool) + } + if name == "" { + return nil, false + } + + tJSON := []byte(`{"name":"","description":"","input_schema":{}}`) + tJSON, _ = sjson.SetBytes(tJSON, "name", name) + if d := responsesToolDescription(tool); d != "" { + tJSON, _ = sjson.SetBytes(tJSON, "description", d) + } + tJSON, _ = sjson.SetRawBytes(tJSON, "input_schema", normalizeClaudeToolInputSchema(responsesToolParameters(tool))) + return tJSON, true +} + +func convertResponsesWebSearchToolToClaude(tool gjson.Result) ([]byte, bool) { + if externalWebAccess := tool.Get("external_web_access"); externalWebAccess.Exists() && !externalWebAccess.Bool() { + return nil, false + } + + name := strings.TrimSpace(tool.Get("name").String()) + if name == "" { + name = "web_search" + } + tJSON := []byte(`{"type":"web_search_20250305","name":""}`) + tJSON, _ = sjson.SetBytes(tJSON, "name", name) + if maxUses := tool.Get("max_uses"); maxUses.Exists() { + tJSON, _ = sjson.SetBytes(tJSON, "max_uses", maxUses.Int()) + } + if allowedDomains := tool.Get("filters.allowed_domains"); allowedDomains.Exists() && allowedDomains.IsArray() { + tJSON, _ = sjson.SetRawBytes(tJSON, "allowed_domains", []byte(allowedDomains.Raw)) + } + if userLocation := tool.Get("user_location"); userLocation.Exists() && userLocation.IsObject() { + tJSON, _ = sjson.SetRawBytes(tJSON, "user_location", []byte(userLocation.Raw)) + } + return tJSON, true +} + +func responsesToolName(tool gjson.Result) string { + if name := strings.TrimSpace(tool.Get("name").String()); name != "" { + return name + } + return strings.TrimSpace(tool.Get("function.name").String()) +} + +func responsesToolDescription(tool gjson.Result) string { + if description := tool.Get("description").String(); description != "" { + return description + } + return tool.Get("function.description").String() +} + +func responsesToolParameters(tool gjson.Result) gjson.Result { + for _, path := range []string{ + "parameters", + "parametersJsonSchema", + "input_schema", + "function.parameters", + "function.parametersJsonSchema", + } { + if parameters := tool.Get(path); parameters.Exists() { + return parameters + } + } + return gjson.Result{} +} + +func normalizeClaudeToolInputSchema(parameters gjson.Result) []byte { + raw := strings.TrimSpace(parameters.Raw) + if raw == "" || raw == "null" || !gjson.Valid(raw) { + return []byte(`{"type":"object","properties":{}}`) + } + result := gjson.Parse(raw) + if !result.IsObject() { + return []byte(`{"type":"object","properties":{}}`) + } + schema := []byte(raw) + schemaType := result.Get("type").String() + if schemaType == "" { + schema, _ = sjson.SetBytes(schema, "type", "object") + schemaType = "object" + } + if schemaType == "object" && !result.Get("properties").Exists() { + schema, _ = sjson.SetRawBytes(schema, "properties", []byte(`{}`)) + } + return schema +} + +func qualifyResponsesNamespaceToolName(namespaceName, childName string) string { + childName = strings.TrimSpace(childName) + if childName == "" || namespaceName == "" || strings.HasPrefix(childName, "mcp__") { + return childName + } + if strings.HasPrefix(childName, namespaceName) { + return childName + } + if strings.HasSuffix(namespaceName, "__") { + return namespaceName + childName + } + return namespaceName + "__" + childName +} + +func isUnsupportedOpenAIBuiltinToolType(toolType string) bool { + switch toolType { + case "image_generation", "file_search", "code_interpreter", "computer_use_preview": + return true + default: + return false + } +} diff --git a/internal/translator/claude/openai/responses/claude_openai-responses_response.go b/internal/translator/claude/openai/responses/claude_openai-responses_response.go index ef2cc1f8..10d12c99 100644 --- a/internal/translator/claude/openai/responses/claude_openai-responses_response.go +++ b/internal/translator/claude/openai/responses/claude_openai-responses_response.go @@ -26,7 +26,8 @@ type claudeToResponsesState struct { FuncNames map[int]string // index -> function name FuncCallIDs map[int]string // index -> call id // message text aggregation - TextBuf strings.Builder + TextBuf strings.Builder + CurrentTextBuf strings.Builder // reasoning state ReasoningActive bool ReasoningItemID string @@ -80,6 +81,7 @@ func ConvertClaudeResponseToOpenAIResponses(ctx context.Context, modelName strin st.CreatedAt = time.Now().Unix() // Reset per-message aggregation state st.TextBuf.Reset() + st.CurrentTextBuf.Reset() st.ReasoningBuf.Reset() st.ReasoningActive = false st.InTextBlock = false @@ -128,6 +130,7 @@ func ConvertClaudeResponseToOpenAIResponses(ctx context.Context, modelName strin if typ == "text" { // open message item + content part st.InTextBlock = true + st.CurrentTextBuf.Reset() st.CurrentMsgID = fmt.Sprintf("msg_%s_0", st.ResponseID) item := []byte(`{"type":"response.output_item.added","sequence_number":0,"output_index":0,"item":{"id":"","type":"message","status":"in_progress","content":[],"role":"assistant"}}`) item, _ = sjson.SetBytes(item, "sequence_number", nextSeq()) @@ -189,6 +192,7 @@ func ConvertClaudeResponseToOpenAIResponses(ctx context.Context, modelName strin out = append(out, emitEvent("response.output_text.delta", msg)) // aggregate text for response.output st.TextBuf.WriteString(t.String()) + st.CurrentTextBuf.WriteString(t.String()) } } else if dt == "input_json_delta" { idx := int(root.Get("index").Int()) @@ -220,17 +224,21 @@ func ConvertClaudeResponseToOpenAIResponses(ctx context.Context, modelName strin case "content_block_stop": idx := int(root.Get("index").Int()) if st.InTextBlock { + fullText := st.CurrentTextBuf.String() done := []byte(`{"type":"response.output_text.done","sequence_number":0,"item_id":"","output_index":0,"content_index":0,"text":"","logprobs":[]}`) done, _ = sjson.SetBytes(done, "sequence_number", nextSeq()) done, _ = sjson.SetBytes(done, "item_id", st.CurrentMsgID) + done, _ = sjson.SetBytes(done, "text", fullText) out = append(out, emitEvent("response.output_text.done", done)) partDone := []byte(`{"type":"response.content_part.done","sequence_number":0,"item_id":"","output_index":0,"content_index":0,"part":{"type":"output_text","annotations":[],"logprobs":[],"text":""}}`) partDone, _ = sjson.SetBytes(partDone, "sequence_number", nextSeq()) partDone, _ = sjson.SetBytes(partDone, "item_id", st.CurrentMsgID) + partDone, _ = sjson.SetBytes(partDone, "part.text", fullText) out = append(out, emitEvent("response.content_part.done", partDone)) final := []byte(`{"type":"response.output_item.done","sequence_number":0,"output_index":0,"item":{"id":"","type":"message","status":"completed","content":[{"type":"output_text","text":""}],"role":"assistant"}}`) final, _ = sjson.SetBytes(final, "sequence_number", nextSeq()) final, _ = sjson.SetBytes(final, "item.id", st.CurrentMsgID) + final, _ = sjson.SetBytes(final, "item.content.0.text", fullText) out = append(out, emitEvent("response.output_item.done", final)) st.InTextBlock = false } else if st.InFuncBlock { From 672fdd14ed0a5d1c0db9e8cd9140023545fa30e8 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Sun, 3 May 2026 22:40:42 +0800 Subject: [PATCH 32/51] feat: filter and drop empty assistant messages in Kimi executor - Added `filterKimiEmptyAssistantMessages` to identify and remove empty assistant messages with no content, tool links, or reasoning. - Integrated logging to track the number of dropped messages. - Updated tests to validate the filtering logic for both empty and valid assistant messages. Fixed: #1730 --- internal/runtime/executor/kimi_executor.go | 103 +++++++++++++++++- .../runtime/executor/kimi_executor_test.go | 67 ++++++++++++ 2 files changed, 168 insertions(+), 2 deletions(-) diff --git a/internal/runtime/executor/kimi_executor.go b/internal/runtime/executor/kimi_executor.go index 3588c962..12c8239f 100644 --- a/internal/runtime/executor/kimi_executor.go +++ b/internal/runtime/executor/kimi_executor.go @@ -322,7 +322,17 @@ func normalizeKimiToolMessageLinks(body []byte) ([]byte, error) { return body, nil } - out := body + msgs := messages.Array() + out, dropped, err := filterKimiEmptyAssistantMessages(body, msgs) + if err != nil { + return body, err + } + if dropped > 0 { + log.WithField("dropped_assistant_messages", dropped).Debug("kimi executor: dropped empty assistant messages") + } + + messages = gjson.GetBytes(out, "messages") + msgs = messages.Array() pending := make([]string, 0) patched := 0 patchedReasoning := 0 @@ -340,7 +350,6 @@ func normalizeKimiToolMessageLinks(body []byte) ([]byte, error) { } } - msgs := messages.Array() for msgIdx := range msgs { msg := msgs[msgIdx] role := strings.TrimSpace(msg.Get("role").String()) @@ -428,6 +437,96 @@ func normalizeKimiToolMessageLinks(body []byte) ([]byte, error) { return out, nil } +func filterKimiEmptyAssistantMessages(body []byte, msgs []gjson.Result) ([]byte, int, error) { + kept := make([]string, 0, len(msgs)) + dropped := 0 + for _, msg := range msgs { + if shouldDropKimiAssistantMessage(msg) { + dropped++ + continue + } + kept = append(kept, msg.Raw) + } + if dropped == 0 { + return body, 0, nil + } + + rawMessages := []byte("[" + strings.Join(kept, ",") + "]") + out, err := sjson.SetRawBytes(body, "messages", rawMessages) + if err != nil { + return body, 0, fmt.Errorf("kimi executor: failed to drop empty assistant messages: %w", err) + } + return out, dropped, nil +} + +func shouldDropKimiAssistantMessage(msg gjson.Result) bool { + if strings.TrimSpace(msg.Get("role").String()) != "assistant" { + return false + } + if hasKimiToolCalls(msg) || hasKimiLegacyFunctionCall(msg) || hasKimiAssistantReasoning(msg) { + return false + } + return isKimiAssistantContentEmpty(msg.Get("content")) +} + +func hasKimiToolCalls(msg gjson.Result) bool { + toolCalls := msg.Get("tool_calls") + return toolCalls.Exists() && toolCalls.IsArray() && len(toolCalls.Array()) > 0 +} + +func hasKimiLegacyFunctionCall(msg gjson.Result) bool { + functionCall := msg.Get("function_call") + if !functionCall.Exists() || functionCall.Type == gjson.Null { + return false + } + if functionCall.IsObject() && strings.TrimSpace(functionCall.Raw) == "{}" { + return false + } + return strings.TrimSpace(functionCall.Raw) != "" +} + +func hasKimiAssistantReasoning(msg gjson.Result) bool { + reasoning := msg.Get("reasoning_content") + return reasoning.Exists() && strings.TrimSpace(reasoning.String()) != "" +} + +func isKimiAssistantContentEmpty(content gjson.Result) bool { + if !content.Exists() || content.Type == gjson.Null { + return true + } + if content.Type == gjson.String { + return strings.TrimSpace(content.String()) == "" + } + if !content.IsArray() { + return false + } + for _, part := range content.Array() { + if !isKimiAssistantContentPartEmpty(part) { + return false + } + } + return true +} + +func isKimiAssistantContentPartEmpty(part gjson.Result) bool { + if !part.Exists() || part.Type == gjson.Null { + return true + } + if part.Type == gjson.String { + return strings.TrimSpace(part.String()) == "" + } + if !part.IsObject() { + return false + } + if text := part.Get("text"); text.Exists() { + return strings.TrimSpace(text.String()) == "" + } + if strings.TrimSpace(part.Get("type").String()) == "text" { + return true + } + return strings.TrimSpace(part.Raw) == "{}" +} + func fallbackAssistantReasoning(msg gjson.Result, hasLatest bool, latest string) string { if hasLatest && strings.TrimSpace(latest) != "" { return latest diff --git a/internal/runtime/executor/kimi_executor_test.go b/internal/runtime/executor/kimi_executor_test.go index 210ddb0e..f3de70f1 100644 --- a/internal/runtime/executor/kimi_executor_test.go +++ b/internal/runtime/executor/kimi_executor_test.go @@ -203,3 +203,70 @@ func TestNormalizeKimiToolMessageLinks_RepairsIDsAndReasoningTogether(t *testing t.Fatalf("messages.2.reasoning_content = %q, want %q", got, "r1") } } + +func TestNormalizeKimiToolMessageLinks_DropsEmptyAssistantWithoutToolLink(t *testing.T) { + body := []byte(`{ + "messages":[ + {"role":"user","content":"start"}, + {"role":"assistant","content":""}, + {"role":"assistant","content":" "}, + {"role":"assistant","content":"","tool_calls":null}, + {"role":"assistant","content":[{"type":"text","text":" "}]}, + {"role":"assistant"}, + {"role":"assistant","content":"keep"}, + {"role":"user","content":"next"} + ] + }`) + + out, err := normalizeKimiToolMessageLinks(body) + if err != nil { + t.Fatalf("normalizeKimiToolMessageLinks() error = %v", err) + } + + messages := gjson.GetBytes(out, "messages").Array() + if len(messages) != 3 { + t.Fatalf("messages length = %d, want 3, raw = %s", len(messages), gjson.GetBytes(out, "messages").Raw) + } + if got := messages[0].Get("content").String(); got != "start" { + t.Fatalf("messages.0.content = %q, want %q", got, "start") + } + if got := messages[1].Get("content").String(); got != "keep" { + t.Fatalf("messages.1.content = %q, want %q", got, "keep") + } + if got := messages[2].Get("content").String(); got != "next" { + t.Fatalf("messages.2.content = %q, want %q", got, "next") + } +} + +func TestNormalizeKimiToolMessageLinks_PreservesAssistantWithToolLinkOrReasoning(t *testing.T) { + body := []byte(`{ + "messages":[ + {"role":"assistant","content":"","tool_calls":[{"id":"call_1","type":"function","function":{"name":"list_directory","arguments":"{}"}}]}, + {"role":"assistant","content":"","function_call":{"name":"legacy_call","arguments":"{}"}}, + {"role":"assistant","content":"","reasoning_content":"thought"}, + {"role":"assistant","content":[{"type":"text","text":" visible "}]} + ] + }`) + + out, err := normalizeKimiToolMessageLinks(body) + if err != nil { + t.Fatalf("normalizeKimiToolMessageLinks() error = %v", err) + } + + messages := gjson.GetBytes(out, "messages").Array() + if len(messages) != 4 { + t.Fatalf("messages length = %d, want 4, raw = %s", len(messages), gjson.GetBytes(out, "messages").Raw) + } + if !messages[0].Get("tool_calls").Exists() { + t.Fatalf("messages.0.tool_calls should exist") + } + if !messages[1].Get("function_call").Exists() { + t.Fatalf("messages.1.function_call should exist") + } + if got := messages[2].Get("reasoning_content").String(); got != "thought" { + t.Fatalf("messages.2.reasoning_content = %q, want %q", got, "thought") + } + if got := messages[3].Get("content.0.text").String(); got != " visible " { + t.Fatalf("messages.3.content.0.text = %q, want %q", got, " visible ") + } +} From bf0e5c23f731e6d80457679e721c535823e5e60e Mon Sep 17 00:00:00 2001 From: 1137043480 <1137043480@users.noreply.github.com> Date: Sun, 3 May 2026 11:25:04 -0400 Subject: [PATCH 33/51] fix: prevent goroutine leaks in streaming executors via context-aware channel sends All streaming executors use bare channel sends (out <- chunk) inside goroutines that process upstream SSE responses. When the downstream consumer disconnects (client timeout, network drop, etc.), these sends block indefinitely, causing the goroutine and all associated resources (HTTP response body, scanner buffers, translation state) to leak permanently. Over time, leaked goroutines accumulate monotonically, leading to RSS growth from ~30MB to 3.7GB+ and eventual OOM kills on resource-constrained VPS hosts. Fix: Replace all bare 'out <- ...' sends with: select { case out <- ...: case <-ctx.Done(): return } This ensures goroutines terminate promptly when the request context is canceled, allowing GC to reclaim all associated resources. Affected executors (9 files, 36+ send sites): - antigravity_executor.go (5 sites) - gemini_cli_executor.go (6 sites) - gemini_vertex_executor.go (6 sites) - aistudio_executor.go (4 sites) - gemini_executor.go (3 sites) - openai_compat_executor.go (3 sites) - claude_executor.go (4 sites) - codex_executor.go (2 sites) - kimi_executor.go (3 sites) --- .../runtime/executor/aistudio_executor.go | 22 +++++++++--- .../runtime/executor/antigravity_executor.go | 28 ++++++++++++--- internal/runtime/executor/claude_executor.go | 22 +++++++++--- internal/runtime/executor/codex_executor.go | 11 ++++-- .../runtime/executor/gemini_cli_executor.go | 34 +++++++++++++++---- internal/runtime/executor/gemini_executor.go | 17 ++++++++-- .../executor/gemini_vertex_executor.go | 34 +++++++++++++++---- internal/runtime/executor/kimi_executor.go | 17 ++++++++-- .../executor/openai_compat_executor.go | 17 ++++++++-- 9 files changed, 166 insertions(+), 36 deletions(-) diff --git a/internal/runtime/executor/aistudio_executor.go b/internal/runtime/executor/aistudio_executor.go index 73491d82..37e85377 100644 --- a/internal/runtime/executor/aistudio_executor.go +++ b/internal/runtime/executor/aistudio_executor.go @@ -285,7 +285,10 @@ func (e *AIStudioExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth if event.Err != nil { helps.RecordAPIResponseError(ctx, e.cfg, event.Err) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: fmt.Errorf("wsrelay: %v", event.Err)} + select { + case out <- cliproxyexecutor.StreamChunk{Err: fmt.Errorf("wsrelay: %v", event.Err)}: + case <-ctx.Done(): + } return false } switch event.Type { @@ -303,7 +306,11 @@ func (e *AIStudioExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth } lines := sdktranslator.TranslateStream(ctx, body.toFormat, opts.SourceFormat, req.Model, opts.OriginalRequest, translatedReq, filtered, ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: ensureColonSpacedJSON(lines[i])} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: ensureColonSpacedJSON(lines[i])}: + case <-ctx.Done(): + return false + } } break } @@ -319,14 +326,21 @@ func (e *AIStudioExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth } lines := sdktranslator.TranslateStream(ctx, body.toFormat, opts.SourceFormat, req.Model, opts.OriginalRequest, translatedReq, event.Payload, ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: ensureColonSpacedJSON(lines[i])} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: ensureColonSpacedJSON(lines[i])}: + case <-ctx.Done(): + return false + } } reporter.Publish(ctx, helps.ParseGeminiUsage(event.Payload)) return false case wsrelay.MessageTypeError: helps.RecordAPIResponseError(ctx, e.cfg, event.Err) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: fmt.Errorf("wsrelay: %v", event.Err)} + select { + case out <- cliproxyexecutor.StreamChunk{Err: fmt.Errorf("wsrelay: %v", event.Err)}: + case <-ctx.Done(): + } return false } return true diff --git a/internal/runtime/executor/antigravity_executor.go b/internal/runtime/executor/antigravity_executor.go index 280c799a..c07680e8 100644 --- a/internal/runtime/executor/antigravity_executor.go +++ b/internal/runtime/executor/antigravity_executor.go @@ -894,12 +894,19 @@ attemptLoop: reporter.Publish(ctx, detail) } - out <- cliproxyexecutor.StreamChunk{Payload: payload} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: payload}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } else { reporter.EnsurePublished(ctx) } @@ -1357,17 +1364,28 @@ attemptLoop: chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, bytes.Clone(payload), ¶m) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } tail := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, []byte("[DONE]"), ¶m) for i := range tail { - out <- cliproxyexecutor.StreamChunk{Payload: tail[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: tail[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } else { reporter.EnsurePublished(ctx) } diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 66432ac4..ea94526e 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -484,12 +484,19 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A cloned := make([]byte, len(line)+1) copy(cloned, line) cloned[len(line)] = '\n' - out <- cliproxyexecutor.StreamChunk{Payload: cloned} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: cloned}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } return } @@ -521,13 +528,20 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A ¶m, ) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil diff --git a/internal/runtime/executor/codex_executor.go b/internal/runtime/executor/codex_executor.go index aa8223f4..6efc25b0 100644 --- a/internal/runtime/executor/codex_executor.go +++ b/internal/runtime/executor/codex_executor.go @@ -515,13 +515,20 @@ func (e *CodexExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.Au chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, originalPayload, body, translatedLine, ¶m) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil diff --git a/internal/runtime/executor/gemini_cli_executor.go b/internal/runtime/executor/gemini_cli_executor.go index 15e84572..b6210e6a 100644 --- a/internal/runtime/executor/gemini_cli_executor.go +++ b/internal/runtime/executor/gemini_cli_executor.go @@ -411,19 +411,30 @@ func (e *GeminiCLIExecutor) ExecuteStream(ctx context.Context, auth *cliproxyaut if bytes.HasPrefix(line, dataTag) { segments := sdktranslator.TranslateStream(respCtx, to, from, attemptModel, opts.OriginalRequest, reqBody, bytes.Clone(line), ¶m) for i := range segments { - out <- cliproxyexecutor.StreamChunk{Payload: segments[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: segments[i]}: + case <-ctx.Done(): + return + } } } } segments := sdktranslator.TranslateStream(respCtx, to, from, attemptModel, opts.OriginalRequest, reqBody, []byte("[DONE]"), ¶m) for i := range segments { - out <- cliproxyexecutor.StreamChunk{Payload: segments[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: segments[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } return } reporter.EnsurePublished(ctx) @@ -434,7 +445,10 @@ func (e *GeminiCLIExecutor) ExecuteStream(ctx context.Context, auth *cliproxyaut if errRead != nil { helps.RecordAPIResponseError(ctx, e.cfg, errRead) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errRead} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errRead}: + case <-ctx.Done(): + } return } helps.AppendAPIResponseChunk(ctx, e.cfg, data) @@ -442,12 +456,20 @@ func (e *GeminiCLIExecutor) ExecuteStream(ctx context.Context, auth *cliproxyaut var param any segments := sdktranslator.TranslateStream(respCtx, to, from, attemptModel, opts.OriginalRequest, reqBody, data, ¶m) for i := range segments { - out <- cliproxyexecutor.StreamChunk{Payload: segments[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: segments[i]}: + case <-ctx.Done(): + return + } } segments = sdktranslator.TranslateStream(respCtx, to, from, attemptModel, opts.OriginalRequest, reqBody, []byte("[DONE]"), ¶m) for i := range segments { - out <- cliproxyexecutor.StreamChunk{Payload: segments[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: segments[i]}: + case <-ctx.Done(): + return + } } }(httpResp, append([]byte(nil), payload...), attemptModel) diff --git a/internal/runtime/executor/gemini_executor.go b/internal/runtime/executor/gemini_executor.go index 0e3c3ec6..2a6e9a6e 100644 --- a/internal/runtime/executor/gemini_executor.go +++ b/internal/runtime/executor/gemini_executor.go @@ -324,17 +324,28 @@ func (e *GeminiExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, bytes.Clone(payload), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, []byte("[DONE]"), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil diff --git a/internal/runtime/executor/gemini_vertex_executor.go b/internal/runtime/executor/gemini_vertex_executor.go index b147fde9..20f5aec1 100644 --- a/internal/runtime/executor/gemini_vertex_executor.go +++ b/internal/runtime/executor/gemini_vertex_executor.go @@ -656,17 +656,28 @@ func (e *GeminiVertexExecutor) executeStreamWithServiceAccount(ctx context.Conte } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, bytes.Clone(line), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, []byte("[DONE]"), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil @@ -786,17 +797,28 @@ func (e *GeminiVertexExecutor) executeStreamWithAPIKey(ctx context.Context, auth } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, bytes.Clone(line), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } } lines := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, []byte("[DONE]"), ¶m) for i := range lines { - out <- cliproxyexecutor.StreamChunk{Payload: lines[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: lines[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil diff --git a/internal/runtime/executor/kimi_executor.go b/internal/runtime/executor/kimi_executor.go index 3588c962..2bb0c7fd 100644 --- a/internal/runtime/executor/kimi_executor.go +++ b/internal/runtime/executor/kimi_executor.go @@ -290,17 +290,28 @@ func (e *KimiExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.Aut } chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, bytes.Clone(line), ¶m) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } doneChunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, body, []byte("[DONE]"), ¶m) for i := range doneChunks { - out <- cliproxyexecutor.StreamChunk{Payload: doneChunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: doneChunks[i]}: + case <-ctx.Done(): + return + } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } }() return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil diff --git a/internal/runtime/executor/openai_compat_executor.go b/internal/runtime/executor/openai_compat_executor.go index 4e44a7ae..ebddfddb 100644 --- a/internal/runtime/executor/openai_compat_executor.go +++ b/internal/runtime/executor/openai_compat_executor.go @@ -293,20 +293,31 @@ func (e *OpenAICompatExecutor) ExecuteStream(ctx context.Context, auth *cliproxy // Pass through translator; it yields one or more chunks for the target schema. chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, bytes.Clone(line), ¶m) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - out <- cliproxyexecutor.StreamChunk{Err: errScan} + select { + case out <- cliproxyexecutor.StreamChunk{Err: errScan}: + case <-ctx.Done(): + } } else { // In case the upstream close the stream without a terminal [DONE] marker. // Feed a synthetic done marker through the translator so pending // response.completed events are still emitted exactly once. chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, []byte("data: [DONE]"), ¶m) for i := range chunks { - out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]} + select { + case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: + case <-ctx.Done(): + return + } } } // Ensure we record the request if no usage chunk was ever seen From 2753d9fb711bb967e5310f35cde43135b04d84e9 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 03:37:31 +0800 Subject: [PATCH 34/51] feat: add validation for Claude streaming responses - Implemented `validateClaudeStreamingResponse` to ensure upstream streaming data integrity. - Added new tests to verify response validation, including empty streams, error events, incomplete streams, and valid streams. - Integrated validation logic into the Claude executor's streaming handler, returning detailed errors for malformed upstream data. Fixed: #2193 --- internal/runtime/executor/claude_executor.go | 62 ++++++++++ .../runtime/executor/claude_executor_test.go | 107 ++++++++++++++++++ 2 files changed, 169 insertions(+) diff --git a/internal/runtime/executor/claude_executor.go b/internal/runtime/executor/claude_executor.go index 66432ac4..3734f262 100644 --- a/internal/runtime/executor/claude_executor.go +++ b/internal/runtime/executor/claude_executor.go @@ -285,6 +285,10 @@ func (e *ClaudeExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth, r } helps.AppendAPIResponseChunk(ctx, e.cfg, data) if stream { + if errValidate := validateClaudeStreamingResponse(data); errValidate != nil { + helps.RecordAPIResponseError(ctx, e.cfg, errValidate) + return resp, errValidate + } lines := bytes.Split(data, []byte("\n")) for _, line := range lines { if detail, ok := helps.ParseClaudeStreamUsage(line); ok { @@ -533,6 +537,64 @@ func (e *ClaudeExecutor) ExecuteStream(ctx context.Context, auth *cliproxyauth.A return &cliproxyexecutor.StreamResult{Headers: httpResp.Header.Clone(), Chunks: out}, nil } +func validateClaudeStreamingResponse(data []byte) error { + scanner := bufio.NewScanner(bytes.NewReader(data)) + scanner.Buffer(nil, 52_428_800) + + hasData := false + hasMessageStart := false + hasMessageDelta := false + + for scanner.Scan() { + line := bytes.TrimSpace(scanner.Bytes()) + if len(line) == 0 || !bytes.HasPrefix(line, []byte("data:")) { + continue + } + payload := bytes.TrimSpace(line[len("data:"):]) + if len(payload) == 0 || bytes.Equal(payload, []byte("[DONE]")) { + continue + } + hasData = true + if !gjson.ValidBytes(payload) { + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream returned malformed stream data"} + } + + root := gjson.ParseBytes(payload) + switch root.Get("type").String() { + case "error": + message := strings.TrimSpace(root.Get("error.message").String()) + if message == "" { + message = strings.TrimSpace(root.Get("error.type").String()) + } + if message == "" { + message = "unknown upstream error" + } + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream returned error event: " + message} + case "message_start": + message := root.Get("message") + if strings.TrimSpace(message.Get("id").String()) == "" || strings.TrimSpace(message.Get("model").String()) == "" { + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream stream message_start is missing id or model"} + } + hasMessageStart = true + case "message_delta": + hasMessageDelta = true + } + } + if errScan := scanner.Err(); errScan != nil { + return errScan + } + if !hasData { + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream returned empty stream response"} + } + if !hasMessageStart { + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream stream response is missing message_start"} + } + if !hasMessageDelta { + return statusErr{code: http.StatusBadGateway, msg: "claude executor: upstream stream response ended before message completion"} + } + return nil +} + func (e *ClaudeExecutor) CountTokens(ctx context.Context, auth *cliproxyauth.Auth, req cliproxyexecutor.Request, opts cliproxyexecutor.Options) (cliproxyexecutor.Response, error) { baseModel := thinking.ParseSuffix(req.Model).ModelName diff --git a/internal/runtime/executor/claude_executor_test.go b/internal/runtime/executor/claude_executor_test.go index c1ce8fc0..6793adda 100644 --- a/internal/runtime/executor/claude_executor_test.go +++ b/internal/runtime/executor/claude_executor_test.go @@ -936,6 +936,113 @@ func TestClaudeExecutor_GeneratesNewUserIDByDefault(t *testing.T) { } } +func TestClaudeExecutor_ExecuteOpenAINonStreamRejectsEmptyClaudeStream(t *testing.T) { + _, err := executeOpenAIChatCompletionThroughClaude(t, "") + if err == nil { + t.Fatal("Execute error = nil, want empty stream error") + } + assertStatusErr(t, err, http.StatusBadGateway) + if !strings.Contains(err.Error(), "empty stream response") { + t.Fatalf("Execute error = %q, want empty stream response", err.Error()) + } +} + +func TestClaudeExecutor_ExecuteOpenAINonStreamRejectsClaudeErrorEvent(t *testing.T) { + body := `data: {"type":"error","error":{"type":"overloaded_error","message":"upstream overloaded"}}` + "\n" + _, err := executeOpenAIChatCompletionThroughClaude(t, body) + if err == nil { + t.Fatal("Execute error = nil, want upstream error event") + } + assertStatusErr(t, err, http.StatusBadGateway) + if !strings.Contains(err.Error(), "upstream overloaded") { + t.Fatalf("Execute error = %q, want upstream overloaded", err.Error()) + } +} + +func TestClaudeExecutor_ExecuteOpenAINonStreamRejectsIncompleteClaudeStream(t *testing.T) { + body := strings.Join([]string{ + `data: {"type":"message_start","message":{"id":"msg_123","model":"claude-3-5-sonnet-20241022"}}`, + `data: {"type":"message_stop"}`, + ``, + }, "\n") + + _, err := executeOpenAIChatCompletionThroughClaude(t, body) + if err == nil { + t.Fatal("Execute error = nil, want incomplete stream error") + } + assertStatusErr(t, err, http.StatusBadGateway) + if !strings.Contains(err.Error(), "ended before message completion") { + t.Fatalf("Execute error = %q, want incomplete stream error", err.Error()) + } +} + +func TestClaudeExecutor_ExecuteOpenAINonStreamConvertsValidClaudeStream(t *testing.T) { + body := strings.Join([]string{ + `event: message_start`, + `data: {"type":"message_start","message":{"id":"msg_123","model":"claude-3-5-sonnet-20241022"}}`, + `event: content_block_delta`, + `data: {"type":"content_block_delta","index":0,"delta":{"type":"text_delta","text":"ok"}}`, + `event: message_delta`, + `data: {"type":"message_delta","delta":{"stop_reason":"end_turn"},"usage":{"input_tokens":2,"output_tokens":1}}`, + `event: message_stop`, + `data: {"type":"message_stop"}`, + ``, + }, "\n") + + resp, err := executeOpenAIChatCompletionThroughClaude(t, body) + if err != nil { + t.Fatalf("Execute error: %v", err) + } + if got := gjson.GetBytes(resp.Payload, "id").String(); got != "msg_123" { + t.Fatalf("response id = %q, want msg_123; payload=%s", got, string(resp.Payload)) + } + if got := gjson.GetBytes(resp.Payload, "model").String(); got != "claude-3-5-sonnet-20241022" { + t.Fatalf("response model = %q, want claude-3-5-sonnet-20241022", got) + } + if got := gjson.GetBytes(resp.Payload, "choices.0.message.content").String(); got != "ok" { + t.Fatalf("response content = %q, want ok", got) + } + if got := gjson.GetBytes(resp.Payload, "usage.total_tokens").Int(); got != 3 { + t.Fatalf("usage.total_tokens = %d, want 3", got) + } +} + +func executeOpenAIChatCompletionThroughClaude(t *testing.T, upstreamBody string) (cliproxyexecutor.Response, error) { + t.Helper() + + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "text/event-stream") + _, _ = w.Write([]byte(upstreamBody)) + })) + defer server.Close() + + executor := NewClaudeExecutor(&config.Config{}) + auth := &cliproxyauth.Auth{Attributes: map[string]string{ + "api_key": "key-123", + "base_url": server.URL, + }} + payload := []byte(`{"model":"claude-3-5-sonnet-20241022","messages":[{"role":"user","content":"hi"}]}`) + + return executor.Execute(context.Background(), auth, cliproxyexecutor.Request{ + Model: "claude-3-5-sonnet-20241022", + Payload: payload, + }, cliproxyexecutor.Options{ + SourceFormat: sdktranslator.FromString("openai"), + }) +} + +func assertStatusErr(t *testing.T, err error, want int) { + t.Helper() + + status, ok := err.(interface{ StatusCode() int }) + if !ok { + t.Fatalf("error %T does not expose StatusCode", err) + } + if got := status.StatusCode(); got != want { + t.Fatalf("StatusCode() = %d, want %d", got, want) + } +} + func TestStripClaudeToolPrefixFromResponse_NestedToolReference(t *testing.T) { input := []byte(`{"content":[{"type":"tool_result","tool_use_id":"toolu_123","content":[{"type":"tool_reference","tool_name":"proxy_mcp__nia__manage_resource"}]}]}`) out := stripClaudeToolPrefixFromResponse(input, "proxy_") From a1487b095855d37998e4c9edbf94aad1670e8e9f Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 05:08:31 +0800 Subject: [PATCH 35/51] fix(translator): handle non-string types in tools result processing - Skip setting values for non-string `type` fields to prevent runtime errors. Closes: #2226 --- .../codex_gemini-cli_request_test.go | 78 +++++++++++++++++++ .../codex/gemini/codex_gemini_request.go | 6 +- 2 files changed, 83 insertions(+), 1 deletion(-) create mode 100644 internal/translator/codex/gemini-cli/codex_gemini-cli_request_test.go diff --git a/internal/translator/codex/gemini-cli/codex_gemini-cli_request_test.go b/internal/translator/codex/gemini-cli/codex_gemini-cli_request_test.go new file mode 100644 index 00000000..fc41452b --- /dev/null +++ b/internal/translator/codex/gemini-cli/codex_gemini-cli_request_test.go @@ -0,0 +1,78 @@ +package geminiCLI + +import ( + "testing" + + "github.com/tidwall/gjson" +) + +func TestConvertGeminiCLIRequestToCodex_PreservesSchemaPropertyNamedType(t *testing.T) { + input := []byte(`{ + "request": { + "tools": [ + { + "functionDeclarations": [ + { + "name": "ask_user", + "description": "Ask the user one or more questions.", + "parametersJsonSchema": { + "type": "object", + "properties": { + "questions": { + "type": "array", + "items": { + "type": "object", + "properties": { + "header": { + "type": "string" + }, + "type": { + "default": "choice", + "description": "Question type.", + "enum": [ + "choice", + "text", + "yesno" + ], + "type": "string" + } + }, + "required": [ + "question", + "header", + "type" + ] + } + } + }, + "required": [ + "questions" + ] + } + } + ] + } + ] + } + }`) + + out := ConvertGeminiCLIRequestToCodex("gpt-5.2", input, true) + tool := gjson.GetBytes(out, "tools.0") + if got := tool.Get("type").String(); got != "function" { + t.Fatalf("expected tool type %q, got %q; output=%s", "function", got, string(out)) + } + + typeProperty := tool.Get("parameters.properties.questions.items.properties.type") + if !typeProperty.IsObject() { + t.Fatalf("expected schema property named type to stay an object; output=%s", string(out)) + } + if got := typeProperty.Get("type").String(); got != "string" { + t.Fatalf("expected schema property type %q, got %q; output=%s", "string", got, string(out)) + } + if got := typeProperty.Get("default").String(); got != "choice" { + t.Fatalf("expected default %q, got %q; output=%s", "choice", got, string(out)) + } + if got := typeProperty.Get("enum.2").String(); got != "yesno" { + t.Fatalf("expected enum value %q, got %q; output=%s", "yesno", got, string(out)) + } +} diff --git a/internal/translator/codex/gemini/codex_gemini_request.go b/internal/translator/codex/gemini/codex_gemini_request.go index 23dae7d7..37399700 100644 --- a/internal/translator/codex/gemini/codex_gemini_request.go +++ b/internal/translator/codex/gemini/codex_gemini_request.go @@ -284,7 +284,11 @@ func ConvertGeminiRequestToCodex(modelName string, inputRawJSON []byte, _ bool) util.Walk(toolsResult, "", "type", &pathsToLower) for _, p := range pathsToLower { fullPath := fmt.Sprintf("tools.%s", p) - out, _ = sjson.SetBytes(out, fullPath, strings.ToLower(gjson.GetBytes(out, fullPath).String())) + typeValue := gjson.GetBytes(out, fullPath) + if typeValue.Type != gjson.String { + continue + } + out, _ = sjson.SetBytes(out, fullPath, strings.ToLower(typeValue.String())) } return out From 8e6ef3fa645caf15466130084760c1ecf6d925bb Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 05:23:23 +0800 Subject: [PATCH 36/51] fix(websocket): ensure state consistency on auth errors in streaming - Added logic to reset `pinnedAuthID` and replay transcript on unauthorized, forbidden, or throttling errors. - Enhanced error handling in `forwardResponsesWebsocket` with detailed status inspection. - Introduced `shouldReleaseResponsesWebsocketPinnedAuth` to determine auth reset conditions. - Updated state management to preserve prior request and response data during forced replay. Fixed: #2230 --- .../openai/openai_responses_websocket.go | 53 ++++- .../openai/openai_responses_websocket_test.go | 187 +++++++++++++++++- 2 files changed, 229 insertions(+), 11 deletions(-) diff --git a/sdk/api/handlers/openai/openai_responses_websocket.go b/sdk/api/handlers/openai/openai_responses_websocket.go index caf26f13..7a9d2224 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket.go +++ b/sdk/api/handlers/openai/openai_responses_websocket.go @@ -79,6 +79,7 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { var lastRequest []byte lastResponseOutput := []byte("[]") pinnedAuthID := "" + forceTranscriptReplayNextRequest := false for { msgType, payload, errReadMessage := conn.ReadMessage() @@ -115,6 +116,9 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { } allowIncrementalInputWithPreviousResponseID = h.websocketUpstreamSupportsIncrementalInputForModel(requestModelName) } + if forceTranscriptReplayNextRequest { + allowIncrementalInputWithPreviousResponseID = false + } allowCompactionReplayBypass := false if pinnedAuthID != "" && h != nil && h.AuthManager != nil { @@ -179,7 +183,13 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { requestJSON = repairResponsesWebsocketToolCalls(downstreamSessionKey, requestJSON) updatedLastRequest = bytes.Clone(requestJSON) + previousLastRequest := bytes.Clone(lastRequest) + previousLastResponseOutput := bytes.Clone(lastResponseOutput) + forcedTranscriptReplay := forceTranscriptReplayNextRequest lastRequest = updatedLastRequest + if forcedTranscriptReplay { + forceTranscriptReplayNextRequest = false + } modelName := gjson.GetBytes(requestJSON, "model").String() cliCtx, cliCancel := h.GetContextWithCancel(h, c, context.Background()) @@ -204,12 +214,19 @@ func (h *OpenAIResponsesAPIHandler) ResponsesWebsocket(c *gin.Context) { } dataChan, _, errChan := h.ExecuteStreamWithAuthManager(cliCtx, h.HandlerType(), modelName, requestJSON, "") - completedOutput, errForward := h.forwardResponsesWebsocket(c, conn, cliCancel, dataChan, errChan, &wsTimelineLog, passthroughSessionID) + completedOutput, forwardErrMsg, errForward := h.forwardResponsesWebsocket(c, conn, cliCancel, dataChan, errChan, &wsTimelineLog, passthroughSessionID) if errForward != nil { wsTerminateErr = errForward log.Warnf("responses websocket: forward failed id=%s error=%v", passthroughSessionID, errForward) return } + if shouldReleaseResponsesWebsocketPinnedAuth(forwardErrMsg) { + pinnedAuthID = "" + forceTranscriptReplayNextRequest = true + lastRequest = previousLastRequest + lastResponseOutput = previousLastResponseOutput + continue + } lastResponseOutput = completedOutput } } @@ -810,7 +827,7 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( errs <-chan *interfaces.ErrorMessage, wsTimelineLog *strings.Builder, sessionID string, -) ([]byte, error) { +) ([]byte, *interfaces.ErrorMessage, error) { completed := false completedOutput := []byte("[]") downstreamSessionKey := "" @@ -822,7 +839,7 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( select { case <-c.Request.Context().Done(): cancel(c.Request.Context().Err()) - return completedOutput, c.Request.Context().Err() + return completedOutput, nil, c.Request.Context().Err() case errMsg, ok := <-errs: if !ok { errs = nil @@ -847,7 +864,7 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( // errWrite, // ) cancel(errMsg.Error) - return completedOutput, errWrite + return completedOutput, errMsg, errWrite } } if errMsg != nil { @@ -855,7 +872,7 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( } else { cancel(nil) } - return completedOutput, nil + return completedOutput, errMsg, nil case chunk, ok := <-data: if !ok { if !completed { @@ -881,13 +898,13 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( errWrite, ) cancel(errMsg.Error) - return completedOutput, errWrite + return completedOutput, errMsg, errWrite } cancel(errMsg.Error) - return completedOutput, nil + return completedOutput, errMsg, nil } cancel(nil) - return completedOutput, nil + return completedOutput, nil, nil } payloads := websocketJSONPayloadsFromChunk(chunk) @@ -914,13 +931,31 @@ func (h *OpenAIResponsesAPIHandler) forwardResponsesWebsocket( errWrite, ) cancel(errWrite) - return completedOutput, errWrite + return completedOutput, nil, errWrite } } } } } +func shouldReleaseResponsesWebsocketPinnedAuth(errMsg *interfaces.ErrorMessage) bool { + if errMsg == nil { + return false + } + status := errMsg.StatusCode + if status <= 0 && errMsg.Error != nil { + if se, ok := errMsg.Error.(interface{ StatusCode() int }); ok && se != nil { + status = se.StatusCode() + } + } + switch status { + case http.StatusUnauthorized, http.StatusPaymentRequired, http.StatusForbidden, http.StatusTooManyRequests: + return true + default: + return false + } +} + func responseCompletedOutputFromPayload(payload []byte) []byte { output := gjson.GetBytes(payload, "response.output") if output.Exists() && output.IsArray() { diff --git a/sdk/api/handlers/openai/openai_responses_websocket_test.go b/sdk/api/handlers/openai/openai_responses_websocket_test.go index f2c4319e..1d397ecd 100644 --- a/sdk/api/handlers/openai/openai_responses_websocket_test.go +++ b/sdk/api/handlers/openai/openai_responses_websocket_test.go @@ -69,6 +69,22 @@ type websocketAuthCaptureExecutor struct { authIDs []string } +type websocketPinnedFailoverExecutor struct { + mu sync.Mutex + authIDs []string + calls map[string]int + payloads map[string][][]byte +} + +type websocketPinnedFailoverStatusError struct { + status int + msg string +} + +func (e websocketPinnedFailoverStatusError) Error() string { return e.msg } + +func (e websocketPinnedFailoverStatusError) StatusCode() int { return e.status } + func (e *websocketAuthCaptureExecutor) Identifier() string { return "test-provider" } func (e *websocketAuthCaptureExecutor) Execute(context.Context, *coreauth.Auth, coreexecutor.Request, coreexecutor.Options) (coreexecutor.Response, error) { @@ -106,6 +122,76 @@ func (e *websocketAuthCaptureExecutor) AuthIDs() []string { return append([]string(nil), e.authIDs...) } +func (e *websocketPinnedFailoverExecutor) Identifier() string { return "test-provider" } + +func (e *websocketPinnedFailoverExecutor) Execute(context.Context, *coreauth.Auth, coreexecutor.Request, coreexecutor.Options) (coreexecutor.Response, error) { + return coreexecutor.Response{}, errors.New("not implemented") +} + +func (e *websocketPinnedFailoverExecutor) ExecuteStream(_ context.Context, auth *coreauth.Auth, req coreexecutor.Request, _ coreexecutor.Options) (*coreexecutor.StreamResult, error) { + authID := "" + if auth != nil { + authID = auth.ID + } + + e.mu.Lock() + if e.calls == nil { + e.calls = make(map[string]int) + } + if e.payloads == nil { + e.payloads = make(map[string][][]byte) + } + e.authIDs = append(e.authIDs, authID) + e.calls[authID]++ + call := e.calls[authID] + e.payloads[authID] = append(e.payloads[authID], bytes.Clone(req.Payload)) + e.mu.Unlock() + + if authID == "auth-a" && call == 2 { + chunks := make(chan coreexecutor.StreamChunk, 1) + chunks <- coreexecutor.StreamChunk{Err: websocketPinnedFailoverStatusError{ + status: http.StatusTooManyRequests, + msg: `{"error":{"message":"quota exhausted","type":"rate_limit_error","code":"rate_limit_exceeded"}}`, + }} + close(chunks) + return &coreexecutor.StreamResult{Chunks: chunks}, nil + } + + chunks := make(chan coreexecutor.StreamChunk, 1) + chunks <- coreexecutor.StreamChunk{Payload: []byte(fmt.Sprintf(`{"type":"response.completed","response":{"id":"resp-%s-%d","output":[{"type":"message","id":"out-%s-%d"}]}}`, authID, call, authID, call))} + close(chunks) + return &coreexecutor.StreamResult{Chunks: chunks}, nil +} + +func (e *websocketPinnedFailoverExecutor) Refresh(_ context.Context, auth *coreauth.Auth) (*coreauth.Auth, error) { + return auth, nil +} + +func (e *websocketPinnedFailoverExecutor) CountTokens(context.Context, *coreauth.Auth, coreexecutor.Request, coreexecutor.Options) (coreexecutor.Response, error) { + return coreexecutor.Response{}, errors.New("not implemented") +} + +func (e *websocketPinnedFailoverExecutor) HttpRequest(context.Context, *coreauth.Auth, *http.Request) (*http.Response, error) { + return nil, errors.New("not implemented") +} + +func (e *websocketPinnedFailoverExecutor) AuthIDs() []string { + e.mu.Lock() + defer e.mu.Unlock() + return append([]string(nil), e.authIDs...) +} + +func (e *websocketPinnedFailoverExecutor) Payloads(authID string) [][]byte { + e.mu.Lock() + defer e.mu.Unlock() + src := e.payloads[authID] + out := make([][]byte, len(src)) + for i := range src { + out[i] = bytes.Clone(src[i]) + } + return out +} + func (e *websocketCaptureExecutor) Identifier() string { return "test-provider" } func (e *websocketCaptureExecutor) Execute(context.Context, *coreauth.Auth, coreexecutor.Request, coreexecutor.Options) (coreexecutor.Response, error) { @@ -681,7 +767,7 @@ func TestForwardResponsesWebsocketPreservesCompletedEvent(t *testing.T) { close(errCh) var timelineLog strings.Builder - completedOutput, err := (*OpenAIResponsesAPIHandler)(nil).forwardResponsesWebsocket( + completedOutput, errMsg, err := (*OpenAIResponsesAPIHandler)(nil).forwardResponsesWebsocket( ctx, conn, func(...interface{}) {}, @@ -694,6 +780,10 @@ func TestForwardResponsesWebsocketPreservesCompletedEvent(t *testing.T) { serverErrCh <- err return } + if errMsg != nil { + serverErrCh <- fmt.Errorf("unexpected websocket error message: %v", errMsg.Error) + return + } if gjson.GetBytes(completedOutput, "0.id").String() != "out-1" { serverErrCh <- errors.New("completed output not captured") return @@ -760,7 +850,7 @@ func TestForwardResponsesWebsocketLogsAttemptedResponseOnWriteFailure(t *testing return } - _, err = (*OpenAIResponsesAPIHandler)(nil).forwardResponsesWebsocket( + _, _, err = (*OpenAIResponsesAPIHandler)(nil).forwardResponsesWebsocket( ctx, conn, func(...interface{}) {}, @@ -1113,6 +1203,99 @@ func TestResponsesWebsocketPinsOnlyWebsocketCapableAuth(t *testing.T) { } } +func TestResponsesWebsocketReleasesPinnedAuthAfterQuotaError(t *testing.T) { + gin.SetMode(gin.TestMode) + + selector := &orderedWebsocketSelector{order: []string{"auth-a", "auth-b"}} + executor := &websocketPinnedFailoverExecutor{} + manager := coreauth.NewManager(nil, selector, nil) + manager.RegisterExecutor(executor) + + authA := &coreauth.Auth{ + ID: "auth-a", + Provider: executor.Identifier(), + Status: coreauth.StatusActive, + Attributes: map[string]string{"websockets": "true"}, + } + if _, err := manager.Register(context.Background(), authA); err != nil { + t.Fatalf("Register auth A: %v", err) + } + authB := &coreauth.Auth{ + ID: "auth-b", + Provider: executor.Identifier(), + Status: coreauth.StatusActive, + Attributes: map[string]string{"websockets": "true"}, + } + if _, err := manager.Register(context.Background(), authB); err != nil { + t.Fatalf("Register auth B: %v", err) + } + + registry.GetGlobalRegistry().RegisterClient(authA.ID, authA.Provider, []*registry.ModelInfo{{ID: "quota-model"}}) + registry.GetGlobalRegistry().RegisterClient(authB.ID, authB.Provider, []*registry.ModelInfo{{ID: "quota-model"}}) + t.Cleanup(func() { + registry.GetGlobalRegistry().UnregisterClient(authA.ID) + registry.GetGlobalRegistry().UnregisterClient(authB.ID) + }) + + base := handlers.NewBaseAPIHandlers(&sdkconfig.SDKConfig{}, manager) + h := NewOpenAIResponsesAPIHandler(base) + router := gin.New() + router.GET("/v1/responses/ws", h.ResponsesWebsocket) + + server := httptest.NewServer(router) + defer server.Close() + + wsURL := "ws" + strings.TrimPrefix(server.URL, "http") + "/v1/responses/ws" + conn, _, err := websocket.DefaultDialer.Dial(wsURL, nil) + if err != nil { + t.Fatalf("dial websocket: %v", err) + } + defer func() { + if errClose := conn.Close(); errClose != nil { + t.Fatalf("close websocket: %v", errClose) + } + }() + + requests := []string{ + `{"type":"response.create","model":"quota-model","input":[{"type":"message","id":"msg-1"}]}`, + `{"type":"response.create","previous_response_id":"resp-auth-a-1","input":[{"type":"message","id":"msg-2"}]}`, + `{"type":"response.create","previous_response_id":"resp-auth-a-1","input":[{"type":"message","id":"msg-3"}]}`, + } + wantTypes := []string{wsEventTypeCompleted, wsEventTypeError, wsEventTypeCompleted} + for i := range requests { + if errWrite := conn.WriteMessage(websocket.TextMessage, []byte(requests[i])); errWrite != nil { + t.Fatalf("write websocket message %d: %v", i+1, errWrite) + } + _, payload, errReadMessage := conn.ReadMessage() + if errReadMessage != nil { + t.Fatalf("read websocket message %d: %v", i+1, errReadMessage) + } + if got := gjson.GetBytes(payload, "type").String(); got != wantTypes[i] { + t.Fatalf("message %d payload type = %s, want %s: %s", i+1, got, wantTypes[i], payload) + } + if i == 1 && int(gjson.GetBytes(payload, "status").Int()) != http.StatusTooManyRequests { + t.Fatalf("quota payload status = %d, want %d: %s", gjson.GetBytes(payload, "status").Int(), http.StatusTooManyRequests, payload) + } + } + + if got := executor.AuthIDs(); len(got) != 3 || got[0] != "auth-a" || got[1] != "auth-a" || got[2] != "auth-b" { + t.Fatalf("selected auth IDs = %v, want [auth-a auth-a auth-b]", got) + } + + authBPayloads := executor.Payloads("auth-b") + if len(authBPayloads) != 1 { + t.Fatalf("auth-b payload count = %d, want 1", len(authBPayloads)) + } + authBPayload := authBPayloads[0] + if gjson.GetBytes(authBPayload, "previous_response_id").Exists() { + t.Fatalf("previous_response_id leaked after auth failover: %s", authBPayload) + } + authBInput := gjson.GetBytes(authBPayload, "input").Raw + if !strings.Contains(authBInput, `"id":"msg-1"`) || !strings.Contains(authBInput, `"id":"msg-3"`) { + t.Fatalf("auth-b replay input missing expected transcript items: %s", authBInput) + } +} + func TestNormalizeResponsesWebsocketRequestTreatsTranscriptReplacementAsReset(t *testing.T) { lastRequest := []byte(`{"model":"test-model","stream":true,"input":[{"type":"message","id":"msg-1"},{"type":"function_call","id":"fc-1","call_id":"call-1"},{"type":"function_call_output","id":"tool-out-1","call_id":"call-1"},{"type":"message","id":"assistant-1","role":"assistant"}]}`) lastResponseOutput := []byte(`[ From 38dad2afdf8280350e0f09b07a32ba0865596a26 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 05:36:09 +0800 Subject: [PATCH 37/51] chore(docker): upgrade base image to alpine 3.23 Fixed: #2265 --- Dockerfile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Dockerfile b/Dockerfile index 3e10c4f9..b4caaee3 100644 --- a/Dockerfile +++ b/Dockerfile @@ -14,7 +14,7 @@ ARG BUILD_DATE=unknown RUN CGO_ENABLED=0 GOOS=linux go build -ldflags="-s -w -X 'main.Version=${VERSION}' -X 'main.Commit=${COMMIT}' -X 'main.BuildDate=${BUILD_DATE}'" -o ./CLIProxyAPI ./cmd/server/ -FROM alpine:3.22.0 +FROM alpine:3.23 RUN apk add --no-cache tzdata @@ -32,4 +32,4 @@ ENV TZ=Asia/Shanghai RUN cp /usr/share/zoneinfo/${TZ} /etc/localtime && echo "${TZ}" > /etc/timezone -CMD ["./CLIProxyAPI"] \ No newline at end of file +CMD ["./CLIProxyAPI"] From 17be6442a8bfebe69e20631d99b5b6961eb54e4c Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 05:50:01 +0800 Subject: [PATCH 38/51] fix(translator): improve tool response handling for non-string content - Added `setToolCallOutputContent` to process various content types, including arrays and fallback cases. - Implemented robust handling for specific tool output types like text, image URLs, and files, ensuring proper serialization. - Improved fallback logic to handle unexpected or missing data. Fixed: #2313 Closes: #2349 --- .../chat-completions/codex_openai_request.go | 89 ++++++++- .../codex_openai_request_test.go | 176 ++++++++++++++++++ 2 files changed, 263 insertions(+), 2 deletions(-) diff --git a/internal/translator/codex/openai/chat-completions/codex_openai_request.go b/internal/translator/codex/openai/chat-completions/codex_openai_request.go index 6cc701e7..569e06e3 100644 --- a/internal/translator/codex/openai/chat-completions/codex_openai_request.go +++ b/internal/translator/codex/openai/chat-completions/codex_openai_request.go @@ -121,13 +121,13 @@ func ConvertOpenAIRequestToCodex(modelName string, inputRawJSON []byte, stream b case "tool": // Handle tool response messages as top-level function_call_output objects toolCallID := m.Get("tool_call_id").String() - content := m.Get("content").String() + content := m.Get("content") // Create function_call_output object funcOutput := []byte(`{}`) funcOutput, _ = sjson.SetBytes(funcOutput, "type", "function_call_output") funcOutput, _ = sjson.SetBytes(funcOutput, "call_id", toolCallID) - funcOutput, _ = sjson.SetBytes(funcOutput, "output", content) + funcOutput = setToolCallOutputContent(funcOutput, content) out, _ = sjson.SetRawBytes(out, "input.-1", funcOutput) default: @@ -359,6 +359,91 @@ func ConvertOpenAIRequestToCodex(modelName string, inputRawJSON []byte, stream b return out } +func setToolCallOutputContent(funcOutput []byte, content gjson.Result) []byte { + switch { + case content.Type == gjson.String: + funcOutput, _ = sjson.SetBytes(funcOutput, "output", content.String()) + case content.IsArray(): + output := []byte(`[]`) + for _, item := range content.Array() { + output = appendToolOutputContentPart(output, item) + } + funcOutput, _ = sjson.SetRawBytes(funcOutput, "output", output) + default: + fallbackOutput := content.Raw + if fallbackOutput == "" { + fallbackOutput = content.String() + } + funcOutput, _ = sjson.SetBytes(funcOutput, "output", fallbackOutput) + } + return funcOutput +} + +func appendToolOutputContentPart(output []byte, item gjson.Result) []byte { + switch item.Get("type").String() { + case "text": + part := []byte(`{}`) + part, _ = sjson.SetBytes(part, "type", "input_text") + part, _ = sjson.SetBytes(part, "text", item.Get("text").String()) + output, _ = sjson.SetRawBytes(output, "-1", part) + case "image_url": + imageURL := item.Get("image_url.url").String() + fileID := item.Get("image_url.file_id").String() + if imageURL == "" && fileID == "" { + return appendToolOutputFallbackPart(output, item) + } + part := []byte(`{}`) + part, _ = sjson.SetBytes(part, "type", "input_image") + if imageURL != "" { + part, _ = sjson.SetBytes(part, "image_url", imageURL) + } + if fileID != "" { + part, _ = sjson.SetBytes(part, "file_id", fileID) + } + if detail := item.Get("image_url.detail").String(); detail != "" { + part, _ = sjson.SetBytes(part, "detail", detail) + } + output, _ = sjson.SetRawBytes(output, "-1", part) + case "file": + fileID := item.Get("file.file_id").String() + fileData := item.Get("file.file_data").String() + fileURL := item.Get("file.file_url").String() + if fileID == "" && fileData == "" && fileURL == "" { + return appendToolOutputFallbackPart(output, item) + } + part := []byte(`{}`) + part, _ = sjson.SetBytes(part, "type", "input_file") + if fileID != "" { + part, _ = sjson.SetBytes(part, "file_id", fileID) + } + if fileData != "" { + part, _ = sjson.SetBytes(part, "file_data", fileData) + } + if fileURL != "" { + part, _ = sjson.SetBytes(part, "file_url", fileURL) + } + if filename := item.Get("file.filename").String(); filename != "" { + part, _ = sjson.SetBytes(part, "filename", filename) + } + output, _ = sjson.SetRawBytes(output, "-1", part) + default: + output = appendToolOutputFallbackPart(output, item) + } + return output +} + +func appendToolOutputFallbackPart(output []byte, item gjson.Result) []byte { + text := item.Raw + if text == "" { + text = item.String() + } + part := []byte(`{}`) + part, _ = sjson.SetBytes(part, "type", "input_text") + part, _ = sjson.SetBytes(part, "text", text) + output, _ = sjson.SetRawBytes(output, "-1", part) + return output +} + // shortenNameIfNeeded applies the simple shortening rule for a single name. // If the name length exceeds 64, it will try to preserve the "mcp__" prefix and last segment. // Otherwise it truncates to 64 characters. diff --git a/internal/translator/codex/openai/chat-completions/codex_openai_request_test.go b/internal/translator/codex/openai/chat-completions/codex_openai_request_test.go index 84c8dad2..e31db6d3 100644 --- a/internal/translator/codex/openai/chat-completions/codex_openai_request_test.go +++ b/internal/translator/codex/openai/chat-completions/codex_openai_request_test.go @@ -176,6 +176,182 @@ func TestToolCallWithContent(t *testing.T) { } } +func TestToolCallOutputWithMultimodalContent(t *testing.T) { + input := []byte(`{ + "model": "gpt-4o", + "messages": [ + {"role": "user", "content": "Show me the generated result."}, + { + "role": "assistant", + "content": null, + "tool_calls": [ + { + "id": "call_output_1", + "type": "function", + "function": {"name": "render_output", "arguments": "{}"} + } + ] + }, + { + "role": "tool", + "tool_call_id": "call_output_1", + "content": [ + {"type":"text","text":"Rendered result attached."}, + {"type":"image_url","image_url":{"url":"https://example.com/generated.png","detail":"high"}}, + {"type":"image_url","image_url":{"file_id":"file-img-123"}}, + {"type":"file","file":{"file_id":"file-doc-123","filename":"doc.pdf"}}, + {"type":"file","file":{"file_data":"SGVsbG8=","filename":"inline.txt"}}, + {"type":"file","file":{"file_url":"https://example.com/report.pdf","filename":"report.pdf"}} + ] + } + ], + "tools": [ + { + "type": "function", + "function": {"name": "render_output", "description": "Render output", "parameters": {"type": "object", "properties": {}}} + } + ] + }`) + + out := ConvertOpenAIRequestToCodex("gpt-4o", input, true) + result := string(out) + + output := gjson.Get(result, "input.2.output") + if !output.IsArray() { + t.Fatalf("expected tool output to be an array, got: %s", output.Raw) + } + + parts := output.Array() + if len(parts) != 6 { + t.Fatalf("expected 6 output parts, got %d: %s", len(parts), output.Raw) + } + if parts[0].Get("type").String() != "input_text" || parts[0].Get("text").String() != "Rendered result attached." { + t.Fatalf("part 0: expected input_text with rendered text, got %s", parts[0].Raw) + } + if parts[1].Get("type").String() != "input_image" { + t.Fatalf("part 1: expected input_image, got %s", parts[1].Raw) + } + if parts[1].Get("image_url").String() != "https://example.com/generated.png" { + t.Errorf("part 1: unexpected image_url %s", parts[1].Get("image_url").String()) + } + if parts[1].Get("detail").String() != "high" { + t.Errorf("part 1: unexpected detail %s", parts[1].Get("detail").String()) + } + if parts[2].Get("type").String() != "input_image" || parts[2].Get("file_id").String() != "file-img-123" { + t.Fatalf("part 2: expected file_id-backed input_image, got %s", parts[2].Raw) + } + if parts[3].Get("type").String() != "input_file" || parts[3].Get("file_id").String() != "file-doc-123" { + t.Fatalf("part 3: expected file_id-backed input_file, got %s", parts[3].Raw) + } + if parts[3].Get("filename").String() != "doc.pdf" { + t.Errorf("part 3: unexpected filename %s", parts[3].Get("filename").String()) + } + if parts[4].Get("type").String() != "input_file" || parts[4].Get("file_data").String() != "SGVsbG8=" { + t.Fatalf("part 4: expected file_data-backed input_file, got %s", parts[4].Raw) + } + if parts[5].Get("type").String() != "input_file" || parts[5].Get("file_url").String() != "https://example.com/report.pdf" { + t.Fatalf("part 5: expected file_url-backed input_file, got %s", parts[5].Raw) + } +} + +func TestToolCallOutputFallsBackForInvalidStructuredParts(t *testing.T) { + input := []byte(`{ + "model": "gpt-4o", + "messages": [ + {"role": "user", "content": "Check tool output."}, + { + "role": "assistant", + "content": null, + "tool_calls": [ + {"id": "call_invalid_parts", "type": "function", "function": {"name": "inspect", "arguments": "{}"}} + ] + }, + { + "role": "tool", + "tool_call_id": "call_invalid_parts", + "content": [ + {"type":"image_url","image_url":{"detail":"low"}}, + {"type":"file","file":{"filename":"orphan.txt"}}, + {"type":"unknown_type","foo":"bar","nested":{"a":1}} + ] + } + ], + "tools": [ + {"type": "function", "function": {"name": "inspect", "description": "Inspect", "parameters": {"type": "object", "properties": {}}}} + ] + }`) + + out := ConvertOpenAIRequestToCodex("gpt-4o", input, true) + result := string(out) + + parts := gjson.Get(result, "input.2.output").Array() + if len(parts) != 3 { + t.Fatalf("expected 3 output parts, got %d: %s", len(parts), gjson.Get(result, "input.2.output").Raw) + } + + expectedFallbacks := []string{ + `{"type":"image_url","image_url":{"detail":"low"}}`, + `{"type":"file","file":{"filename":"orphan.txt"}}`, + `{"type":"unknown_type","foo":"bar","nested":{"a":1}}`, + } + for i, expectedFallback := range expectedFallbacks { + if parts[i].Get("type").String() != "input_text" { + t.Fatalf("part %d: expected input_text fallback, got %s", i, parts[i].Raw) + } + if parts[i].Get("text").String() != expectedFallback { + t.Fatalf("part %d: expected fallback %s, got %s", i, expectedFallback, parts[i].Get("text").String()) + } + } +} + +func TestToolCallOutputWithNonStringJSONContent(t *testing.T) { + tests := []struct { + name string + content string + expectedOutput string + }{ + {name: "null", content: `null`, expectedOutput: `null`}, + {name: "object", content: `{"status":"ok","count":2}`, expectedOutput: `{"status":"ok","count":2}`}, + } + + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + input := []byte(`{ + "model": "gpt-4o", + "messages": [ + {"role": "user", "content": "Check tool output."}, + { + "role": "assistant", + "content": null, + "tool_calls": [ + {"id": "call_json", "type": "function", "function": {"name": "inspect", "arguments": "{}"}} + ] + }, + { + "role": "tool", + "tool_call_id": "call_json", + "content": ` + tt.content + ` + } + ], + "tools": [ + {"type": "function", "function": {"name": "inspect", "description": "Inspect", "parameters": {"type": "object", "properties": {}}}} + ] + }`) + + out := ConvertOpenAIRequestToCodex("gpt-4o", input, true) + result := string(out) + + output := gjson.Get(result, "input.2.output") + if !output.Exists() { + t.Fatalf("expected output field to exist: %s", gjson.Get(result, "input.2").Raw) + } + if output.String() != tt.expectedOutput { + t.Fatalf("expected output %s, got %s", tt.expectedOutput, output.String()) + } + }) + } +} + // Parallel tool calls: assistant invokes 3 tools at once, all call_ids // and outputs must be translated and paired correctly. func TestMultipleToolCalls(t *testing.T) { From c19ae1d5be32537218b61e26ebe7845720966801 Mon Sep 17 00:00:00 2001 From: Kenny Date: Sun, 3 May 2026 15:56:39 -0700 Subject: [PATCH 39/51] Align Codex websocket protocol semantics --- internal/runtime/executor/codex_executor.go | 2 +- .../executor/codex_websockets_executor.go | 153 +++++++++++--- .../codex_websockets_executor_test.go | 189 +++++++++++++++++- 3 files changed, 310 insertions(+), 34 deletions(-) diff --git a/internal/runtime/executor/codex_executor.go b/internal/runtime/executor/codex_executor.go index aa8223f4..5e892ecd 100644 --- a/internal/runtime/executor/codex_executor.go +++ b/internal/runtime/executor/codex_executor.go @@ -31,7 +31,7 @@ import ( const ( codexUserAgent = "codex-tui/0.118.0 (Mac OS 26.3.1; arm64) iTerm.app/3.6.9 (codex-tui; 0.118.0)" - codexOriginator = "codex-tui" + codexOriginator = "codex_cli_rs" codexDefaultImageToolModel = "gpt-image-2" ) diff --git a/internal/runtime/executor/codex_websockets_executor.go b/internal/runtime/executor/codex_websockets_executor.go index 40ba7e92..87ae0efe 100644 --- a/internal/runtime/executor/codex_websockets_executor.go +++ b/internal/runtime/executor/codex_websockets_executor.go @@ -188,7 +188,6 @@ func (e *CodexWebsocketsExecutor) Execute(ctx context.Context, auth *cliproxyaut body = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", body, originalTranslated, requestedModel, requestPath) body, _ = sjson.SetBytes(body, "model", baseModel) body, _ = sjson.SetBytes(body, "stream", true) - body, _ = sjson.DeleteBytes(body, "previous_response_id") body, _ = sjson.DeleteBytes(body, "prompt_cache_retention") body, _ = sjson.DeleteBytes(body, "safety_identifier") body = normalizeCodexInstructions(body) @@ -776,6 +775,11 @@ func buildCodexResponsesWebsocketURL(httpURL string) (string, error) { parsed.Scheme = "ws" case "https": parsed.Scheme = "wss" + default: + return "", fmt.Errorf("codex websockets executor: unsupported responses websocket URL scheme %q", parsed.Scheme) + } + if strings.TrimSpace(parsed.Host) == "" { + return "", fmt.Errorf("codex websockets executor: responses websocket URL host is empty") } return parsed.String(), nil } @@ -809,6 +813,7 @@ func applyCodexPromptCacheHeaders(from sdktranslator.Format, req cliproxyexecuto if cache.ID != "" { rawJSON, _ = sjson.SetBytes(rawJSON, "prompt_cache_key", cache.ID) + setHeaderCasePreserved(headers, "session_id", cache.ID) headers.Set("Conversation_id", cache.ID) } @@ -828,13 +833,19 @@ func applyCodexWebsocketHeaders(ctx context.Context, headers http.Header, auth * ginHeaders = ginCtx.Request.Header.Clone() } - _, cfgBetaFeatures := codexHeaderDefaults(cfg, auth) + isAPIKey := codexAuthUsesAPIKey(auth) + cfgUserAgent, cfgBetaFeatures := codexHeaderDefaults(cfg, auth) ensureHeaderWithPriority(headers, ginHeaders, "x-codex-beta-features", cfgBetaFeatures, "") misc.EnsureHeader(headers, ginHeaders, "x-codex-turn-state", "") misc.EnsureHeader(headers, ginHeaders, "x-codex-turn-metadata", "") misc.EnsureHeader(headers, ginHeaders, "x-client-request-id", "") misc.EnsureHeader(headers, ginHeaders, "x-responsesapi-include-timing-metrics", "") misc.EnsureHeader(headers, ginHeaders, "Version", "") + if isAPIKey { + ensureHeaderWithPriority(headers, ginHeaders, "User-Agent", "", "") + } else { + ensureHeaderWithConfigPrecedence(headers, ginHeaders, "User-Agent", cfgUserAgent, codexUserAgent) + } betaHeader := strings.TrimSpace(headers.Get("OpenAI-Beta")) if betaHeader == "" && ginHeaders != nil { @@ -845,16 +856,9 @@ func applyCodexWebsocketHeaders(ctx context.Context, headers http.Header, auth * } headers.Set("OpenAI-Beta", betaHeader) if strings.Contains(headers.Get("User-Agent"), "Mac OS") { - misc.EnsureHeader(headers, ginHeaders, "Session_id", uuid.NewString()) - } - headers.Del("User-Agent") - - isAPIKey := false - if auth != nil && auth.Attributes != nil { - if v := strings.TrimSpace(auth.Attributes["api_key"]); v != "" { - isAPIKey = true - } + ensureHeaderCasePreserved(headers, ginHeaders, "session_id", "", uuid.NewString()) } + ensureHeaderCasePreserved(headers, ginHeaders, "session_id", "", "") if originator := strings.TrimSpace(ginHeaders.Get("Originator")); originator != "" { headers.Set("Originator", originator) } else if !isAPIKey { @@ -864,7 +868,7 @@ func applyCodexWebsocketHeaders(ctx context.Context, headers http.Header, auth * if auth != nil && auth.Metadata != nil { if accountID, ok := auth.Metadata["account_id"].(string); ok { if trimmed := strings.TrimSpace(accountID); trimmed != "" { - headers.Set("Chatgpt-Account-Id", trimmed) + headers.Set("ChatGPT-Account-ID", trimmed) } } } @@ -879,6 +883,77 @@ func applyCodexWebsocketHeaders(ctx context.Context, headers http.Header, auth * return headers } +func codexAuthUsesAPIKey(auth *cliproxyauth.Auth) bool { + if auth == nil || auth.Attributes == nil { + return false + } + return strings.TrimSpace(auth.Attributes["api_key"]) != "" +} + +func ensureHeaderCasePreserved(target http.Header, source http.Header, key, configValue, fallbackValue string) { + if target == nil { + return + } + if strings.TrimSpace(headerValueCaseInsensitive(target, key)) != "" { + return + } + if source != nil { + if val := strings.TrimSpace(headerValueCaseInsensitive(source, key)); val != "" { + setHeaderCasePreserved(target, key, val) + return + } + } + if val := strings.TrimSpace(configValue); val != "" { + setHeaderCasePreserved(target, key, val) + return + } + if val := strings.TrimSpace(fallbackValue); val != "" { + setHeaderCasePreserved(target, key, val) + } +} + +func setHeaderCasePreserved(headers http.Header, key string, value string) { + if headers == nil { + return + } + key = strings.TrimSpace(key) + value = strings.TrimSpace(value) + if key == "" || value == "" { + return + } + deleteHeaderCaseInsensitive(headers, key) + headers[key] = []string{value} +} + +func headerValueCaseInsensitive(headers http.Header, key string) string { + key = strings.TrimSpace(key) + if headers == nil || key == "" { + return "" + } + if val := strings.TrimSpace(headers.Get(key)); val != "" { + return val + } + for existingKey, values := range headers { + if !strings.EqualFold(existingKey, key) { + continue + } + for _, value := range values { + if trimmed := strings.TrimSpace(value); trimmed != "" { + return trimmed + } + } + } + return "" +} + +func deleteHeaderCaseInsensitive(headers http.Header, key string) { + for existingKey := range headers { + if strings.EqualFold(existingKey, key) { + delete(headers, existingKey) + } + } +} + func codexHeaderDefaults(cfg *config.Config, auth *cliproxyauth.Auth) (string, string) { if cfg == nil || auth == nil { return "", "" @@ -962,25 +1037,53 @@ func parseCodexWebsocketError(payload []byte) (error, bool) { return nil, false } - out := []byte(`{}`) - if errNode := gjson.GetBytes(payload, "error"); errNode.Exists() { - raw := errNode.Raw - if errNode.Type == gjson.String { - raw = errNode.Raw - } - out, _ = sjson.SetRawBytes(out, "error", []byte(raw)) - } else { - out, _ = sjson.SetBytes(out, "error.type", "server_error") - out, _ = sjson.SetBytes(out, "error.message", http.StatusText(status)) - } - + out := buildCodexWebsocketErrorPayload(payload, status) headers := parseCodexWebsocketErrorHeaders(payload) + statusError := statusErr{code: status, msg: string(out)} + if isCodexWebsocketConnectionLimitError(payload) { + retryAfter := time.Duration(0) + statusError.retryAfter = &retryAfter + } return statusErrWithHeaders{ - statusErr: statusErr{code: status, msg: string(out)}, + statusErr: statusError, headers: headers, }, true } +func buildCodexWebsocketErrorPayload(payload []byte, status int) []byte { + out := []byte(`{}`) + out, _ = sjson.SetBytes(out, "status", status) + + if bodyNode := gjson.GetBytes(payload, "body"); bodyNode.Exists() { + out, _ = sjson.SetRawBytes(out, "body", []byte(bodyNode.Raw)) + if bodyErrorNode := bodyNode.Get("error"); bodyErrorNode.Exists() { + out, _ = sjson.SetRawBytes(out, "error", []byte(bodyErrorNode.Raw)) + return out + } + } + + if errNode := gjson.GetBytes(payload, "error"); errNode.Exists() { + out, _ = sjson.SetRawBytes(out, "error", []byte(errNode.Raw)) + return out + } + + out, _ = sjson.SetBytes(out, "error.type", "server_error") + out, _ = sjson.SetBytes(out, "error.message", http.StatusText(status)) + return out +} + +func isCodexWebsocketConnectionLimitError(payload []byte) bool { + if len(payload) == 0 { + return false + } + for _, path := range []string{"error.code", "error.type", "body.error.code", "body.error.type", "code", "error"} { + if strings.TrimSpace(gjson.GetBytes(payload, path).String()) == "websocket_connection_limit_reached" { + return true + } + } + return false +} + func parseCodexWebsocketErrorHeaders(payload []byte) http.Header { headersNode := gjson.GetBytes(payload, "headers") if !headersNode.Exists() || !headersNode.IsObject() { diff --git a/internal/runtime/executor/codex_websockets_executor_test.go b/internal/runtime/executor/codex_websockets_executor_test.go index dec356de..0b7a546e 100644 --- a/internal/runtime/executor/codex_websockets_executor_test.go +++ b/internal/runtime/executor/codex_websockets_executor_test.go @@ -1,15 +1,20 @@ package executor import ( + "bytes" "context" "net/http" "net/http/httptest" "testing" + "time" "github.com/gin-gonic/gin" + "github.com/gorilla/websocket" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" cliproxyauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" + cliproxyexecutor "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/executor" sdkconfig "github.com/router-for-me/CLIProxyAPI/v6/sdk/config" + sdktranslator "github.com/router-for-me/CLIProxyAPI/v6/sdk/translator" "github.com/tidwall/gjson" ) @@ -32,14 +37,71 @@ func TestBuildCodexWebsocketRequestBodyPreservesPreviousResponseID(t *testing.T) } } +func TestCodexWebsocketsExecutePreservesPreviousResponseIDUpstream(t *testing.T) { + upgrader := websocket.Upgrader{CheckOrigin: func(*http.Request) bool { return true }} + capturedPayload := make(chan []byte, 1) + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + if r.URL.Path != "/responses" { + t.Fatalf("request path = %s, want /responses", r.URL.Path) + } + conn, err := upgrader.Upgrade(w, r, nil) + if err != nil { + t.Fatalf("upgrade websocket: %v", err) + } + defer func() { _ = conn.Close() }() + + msgType, payload, err := conn.ReadMessage() + if err != nil { + t.Fatalf("read upstream websocket message: %v", err) + } + if msgType != websocket.TextMessage { + t.Fatalf("message type = %d, want text", msgType) + } + capturedPayload <- bytes.Clone(payload) + + completed := []byte(`{"type":"response.completed","response":{"id":"resp-2","output":[],"usage":{"input_tokens":0,"output_tokens":0,"total_tokens":0}}}`) + if errWrite := conn.WriteMessage(websocket.TextMessage, completed); errWrite != nil { + t.Fatalf("write completed websocket message: %v", errWrite) + } + })) + defer server.Close() + + exec := NewCodexWebsocketsExecutor(&config.Config{SDKConfig: config.SDKConfig{DisableImageGeneration: config.DisableImageGenerationAll}}) + auth := &cliproxyauth.Auth{Attributes: map[string]string{"api_key": "sk-test", "base_url": server.URL}} + req := cliproxyexecutor.Request{ + Model: "gpt-5-codex", + Payload: []byte(`{"model":"gpt-5-codex","previous_response_id":"resp-1","input":[{"type":"message","id":"msg-1"}]}`), + } + opts := cliproxyexecutor.Options{SourceFormat: sdktranslator.FromString("codex")} + + if _, err := exec.Execute(context.Background(), auth, req, opts); err != nil { + t.Fatalf("Execute() error = %v", err) + } + + select { + case payload := <-capturedPayload: + if got := gjson.GetBytes(payload, "type").String(); got != "response.create" { + t.Fatalf("upstream type = %s, want response.create; payload=%s", got, payload) + } + if got := gjson.GetBytes(payload, "previous_response_id").String(); got != "resp-1" { + t.Fatalf("upstream previous_response_id = %s, want resp-1; payload=%s", got, payload) + } + case <-time.After(5 * time.Second): + t.Fatal("timed out waiting for upstream websocket payload") + } +} + func TestApplyCodexWebsocketHeadersDefaultsToCurrentResponsesBeta(t *testing.T) { headers := applyCodexWebsocketHeaders(context.Background(), http.Header{}, nil, "", nil) if got := headers.Get("OpenAI-Beta"); got != codexResponsesWebsocketBetaHeaderValue { t.Fatalf("OpenAI-Beta = %s, want %s", got, codexResponsesWebsocketBetaHeaderValue) } - if got := headers.Get("User-Agent"); got != "" { - t.Fatalf("User-Agent = %s, want empty", got) + if got := headers.Get("User-Agent"); got != codexUserAgent { + t.Fatalf("User-Agent = %s, want %s", got, codexUserAgent) + } + if got := headers.Get("Originator"); got != codexOriginator { + t.Fatalf("Originator = %s, want %s", got, codexOriginator) } if got := headers.Get("Version"); got != "" { t.Fatalf("Version = %q, want empty", got) @@ -62,9 +124,11 @@ func TestApplyCodexWebsocketHeadersPassesThroughClientIdentityHeaders(t *testing } ctx := contextWithGinHeaders(map[string]string{ "Originator": "Codex Desktop", + "User-Agent": "codex_cli_rs/0.1.0", "Version": "0.115.0-alpha.27", "X-Codex-Turn-Metadata": `{"turn_id":"turn-1"}`, "X-Client-Request-Id": "019d2233-e240-7162-992d-38df0a2a0e0d", + "session_id": "sess-client", }) headers := applyCodexWebsocketHeaders(ctx, http.Header{}, auth, "", nil) @@ -72,6 +136,9 @@ func TestApplyCodexWebsocketHeadersPassesThroughClientIdentityHeaders(t *testing if got := headers.Get("Originator"); got != "Codex Desktop" { t.Fatalf("Originator = %s, want %s", got, "Codex Desktop") } + if got := headers.Get("User-Agent"); got != "codex_cli_rs/0.1.0" { + t.Fatalf("User-Agent = %s, want %s", got, "codex_cli_rs/0.1.0") + } if got := headers.Get("Version"); got != "0.115.0-alpha.27" { t.Fatalf("Version = %s, want %s", got, "0.115.0-alpha.27") } @@ -81,6 +148,12 @@ func TestApplyCodexWebsocketHeadersPassesThroughClientIdentityHeaders(t *testing if got := headers.Get("X-Client-Request-Id"); got != "019d2233-e240-7162-992d-38df0a2a0e0d" { t.Fatalf("X-Client-Request-Id = %s, want %s", got, "019d2233-e240-7162-992d-38df0a2a0e0d") } + if got := headerValueCaseInsensitive(headers, "session_id"); got != "sess-client" { + t.Fatalf("session_id = %s, want sess-client", got) + } + if _, ok := headers["session_id"]; !ok { + t.Fatalf("expected lowercase session_id header key, got %#v", headers) + } } func TestApplyCodexWebsocketHeadersUsesConfigDefaultsForOAuth(t *testing.T) { @@ -97,8 +170,8 @@ func TestApplyCodexWebsocketHeadersUsesConfigDefaultsForOAuth(t *testing.T) { headers := applyCodexWebsocketHeaders(context.Background(), http.Header{}, auth, "", cfg) - if got := headers.Get("User-Agent"); got != "" { - t.Fatalf("User-Agent = %s, want empty", got) + if got := headers.Get("User-Agent"); got != "my-codex-client/1.0" { + t.Fatalf("User-Agent = %s, want %s", got, "my-codex-client/1.0") } if got := headers.Get("x-codex-beta-features"); got != "feature-a,feature-b" { t.Fatalf("x-codex-beta-features = %s, want %s", got, "feature-a,feature-b") @@ -129,8 +202,8 @@ func TestApplyCodexWebsocketHeadersPrefersExistingHeadersOverClientAndConfig(t * got := applyCodexWebsocketHeaders(ctx, headers, auth, "", cfg) - if gotVal := got.Get("User-Agent"); gotVal != "" { - t.Fatalf("User-Agent = %s, want empty", gotVal) + if gotVal := got.Get("User-Agent"); gotVal != "existing-ua" { + t.Fatalf("User-Agent = %s, want %s", gotVal, "existing-ua") } if gotVal := got.Get("x-codex-beta-features"); gotVal != "existing-beta" { t.Fatalf("x-codex-beta-features = %s, want %s", gotVal, "existing-beta") @@ -155,8 +228,8 @@ func TestApplyCodexWebsocketHeadersConfigUserAgentOverridesClientHeader(t *testi headers := applyCodexWebsocketHeaders(ctx, http.Header{}, auth, "", cfg) - if got := headers.Get("User-Agent"); got != "" { - t.Fatalf("User-Agent = %s, want empty", got) + if got := headers.Get("User-Agent"); got != "config-ua" { + t.Fatalf("User-Agent = %s, want %s", got, "config-ua") } if got := headers.Get("x-codex-beta-features"); got != "client-beta" { t.Fatalf("x-codex-beta-features = %s, want %s", got, "client-beta") @@ -183,6 +256,106 @@ func TestApplyCodexWebsocketHeadersIgnoresConfigForAPIKeyAuth(t *testing.T) { if got := headers.Get("x-codex-beta-features"); got != "" { t.Fatalf("x-codex-beta-features = %q, want empty", got) } + if got := headers.Get("Originator"); got != "" { + t.Fatalf("Originator = %s, want empty", got) + } +} + +func TestApplyCodexWebsocketHeadersPreservesExplicitAPIKeyUserAgent(t *testing.T) { + auth := &cliproxyauth.Auth{Provider: "codex", Attributes: map[string]string{"api_key": "sk-test"}} + ctx := contextWithGinHeaders(map[string]string{"User-Agent": "api-key-client/1.0", "Originator": "explicit-origin"}) + + headers := applyCodexWebsocketHeaders(ctx, http.Header{}, auth, "sk-test", nil) + + if got := headers.Get("User-Agent"); got != "api-key-client/1.0" { + t.Fatalf("User-Agent = %s, want api-key-client/1.0", got) + } + if got := headers.Get("Originator"); got != "explicit-origin" { + t.Fatalf("Originator = %s, want explicit-origin", got) + } +} + +func TestApplyCodexPromptCacheHeadersSetsLowercaseSessionAndLegacyConversation(t *testing.T) { + req := cliproxyexecutor.Request{Model: "gpt-5-codex", Payload: []byte(`{"prompt_cache_key":"cache-1"}`)} + + _, headers := applyCodexPromptCacheHeaders("openai-response", req, []byte(`{"model":"gpt-5-codex"}`)) + + if got := headerValueCaseInsensitive(headers, "session_id"); got != "cache-1" { + t.Fatalf("session_id = %s, want cache-1", got) + } + if _, ok := headers["session_id"]; !ok { + t.Fatalf("expected lowercase session_id key, got %#v", headers) + } + if got := headers.Get("Conversation_id"); got != "cache-1" { + t.Fatalf("Conversation_id = %s, want cache-1", got) + } +} + +func TestApplyCodexWebsocketHeadersUsesCanonicalAccountHeader(t *testing.T) { + auth := &cliproxyauth.Auth{Provider: "codex", Metadata: map[string]any{"account_id": "acct-1"}} + + headers := applyCodexWebsocketHeaders(context.Background(), http.Header{}, auth, "", nil) + + if got := headers.Get("ChatGPT-Account-ID"); got != "acct-1" { + t.Fatalf("ChatGPT-Account-ID = %s, want acct-1", got) + } +} + +func TestBuildCodexResponsesWebsocketURLRequiresHTTPURL(t *testing.T) { + if got, err := buildCodexResponsesWebsocketURL("https://example.com/backend/responses"); err != nil || got != "wss://example.com/backend/responses" { + t.Fatalf("https URL = %q, %v; want wss URL", got, err) + } + if _, err := buildCodexResponsesWebsocketURL("ftp://example.com/responses"); err == nil { + t.Fatalf("expected unsupported scheme error") + } + if _, err := buildCodexResponsesWebsocketURL("https:///responses"); err == nil { + t.Fatalf("expected empty host error") + } +} + +func TestParseCodexWebsocketErrorMarksConnectionLimitRetryable(t *testing.T) { + err, ok := parseCodexWebsocketError([]byte(`{"type":"error","status":429,"error":{"code":"websocket_connection_limit_reached","message":"too many websockets"},"headers":{"retry-after":"1"}}`)) + if !ok { + t.Fatalf("expected websocket error") + } + status, ok := err.(interface{ StatusCode() int }) + if !ok || status.StatusCode() != http.StatusTooManyRequests { + t.Fatalf("status = %#v, want 429", err) + } + retryable, ok := err.(interface{ RetryAfter() *time.Duration }) + if !ok || retryable.RetryAfter() == nil { + t.Fatalf("expected retryable websocket connection limit error") + } + withHeaders, ok := err.(interface{ Headers() http.Header }) + if !ok || withHeaders.Headers().Get("retry-after") != "1" { + t.Fatalf("headers = %#v, want retry-after", err) + } +} + +func TestParseCodexWebsocketErrorPreservesWrappedBodyAndHeaders(t *testing.T) { + err, ok := parseCodexWebsocketError([]byte(`{"type":"error","status":429,"body":{"error":{"code":"websocket_connection_limit_reached","type":"server_error","message":"too many websocket connections"}},"headers":{"x-request-id":"req-1"}}`)) + if !ok { + t.Fatalf("expected websocket error") + } + + parsed := gjson.Parse(err.Error()) + if got := parsed.Get("status").Int(); got != http.StatusTooManyRequests { + t.Fatalf("wrapped status = %d, want 429; payload=%s", got, err.Error()) + } + if got := parsed.Get("body.error.code").String(); got != "websocket_connection_limit_reached" { + t.Fatalf("wrapped body error code = %s, want websocket_connection_limit_reached; payload=%s", got, err.Error()) + } + if got := parsed.Get("error.code").String(); got != "websocket_connection_limit_reached" { + t.Fatalf("surface error code = %s, want websocket_connection_limit_reached; payload=%s", got, err.Error()) + } + retryable, ok := err.(interface{ RetryAfter() *time.Duration }) + if !ok || retryable.RetryAfter() == nil { + t.Fatalf("expected body.error.code websocket connection limit to be retryable") + } + withHeaders, ok := err.(interface{ Headers() http.Header }) + if !ok || withHeaders.Headers().Get("x-request-id") != "req-1" { + t.Fatalf("headers = %#v, want x-request-id", err) + } } func TestApplyCodexHeadersUsesConfigUserAgentForOAuth(t *testing.T) { From 08b0fe6816380dc76ebe7a3f5442d8c7a0bdb661 Mon Sep 17 00:00:00 2001 From: Kenny Date: Sun, 3 May 2026 19:01:44 -0700 Subject: [PATCH 40/51] Fix Codex websocket retry metadata --- .../executor/codex_websockets_executor.go | 6 +++-- .../codex_websockets_executor_test.go | 27 ++++++++++++++++++- 2 files changed, 30 insertions(+), 3 deletions(-) diff --git a/internal/runtime/executor/codex_websockets_executor.go b/internal/runtime/executor/codex_websockets_executor.go index 87ae0efe..d6f1de86 100644 --- a/internal/runtime/executor/codex_websockets_executor.go +++ b/internal/runtime/executor/codex_websockets_executor.go @@ -868,7 +868,7 @@ func applyCodexWebsocketHeaders(ctx context.Context, headers http.Header, auth * if auth != nil && auth.Metadata != nil { if accountID, ok := auth.Metadata["account_id"].(string); ok { if trimmed := strings.TrimSpace(accountID); trimmed != "" { - headers.Set("ChatGPT-Account-ID", trimmed) + setHeaderCasePreserved(headers, "ChatGPT-Account-ID", trimmed) } } } @@ -1040,7 +1040,9 @@ func parseCodexWebsocketError(payload []byte) (error, bool) { out := buildCodexWebsocketErrorPayload(payload, status) headers := parseCodexWebsocketErrorHeaders(payload) statusError := statusErr{code: status, msg: string(out)} - if isCodexWebsocketConnectionLimitError(payload) { + if retryAfter := parseCodexRetryAfter(status, out, time.Now()); retryAfter != nil { + statusError.retryAfter = retryAfter + } else if isCodexWebsocketConnectionLimitError(payload) { retryAfter := time.Duration(0) statusError.retryAfter = &retryAfter } diff --git a/internal/runtime/executor/codex_websockets_executor_test.go b/internal/runtime/executor/codex_websockets_executor_test.go index 0b7a546e..bf12ef78 100644 --- a/internal/runtime/executor/codex_websockets_executor_test.go +++ b/internal/runtime/executor/codex_websockets_executor_test.go @@ -296,9 +296,16 @@ func TestApplyCodexWebsocketHeadersUsesCanonicalAccountHeader(t *testing.T) { headers := applyCodexWebsocketHeaders(context.Background(), http.Header{}, auth, "", nil) - if got := headers.Get("ChatGPT-Account-ID"); got != "acct-1" { + if got := headerValueCaseInsensitive(headers, "ChatGPT-Account-ID"); got != "acct-1" { t.Fatalf("ChatGPT-Account-ID = %s, want acct-1", got) } + values, ok := headers["ChatGPT-Account-ID"] + if !ok { + t.Fatalf("expected exact ChatGPT-Account-ID key, got %#v", headers) + } + if len(values) != 1 || values[0] != "acct-1" { + t.Fatalf("ChatGPT-Account-ID values = %#v, want [acct-1]", values) + } } func TestBuildCodexResponsesWebsocketURLRequiresHTTPURL(t *testing.T) { @@ -326,12 +333,30 @@ func TestParseCodexWebsocketErrorMarksConnectionLimitRetryable(t *testing.T) { if !ok || retryable.RetryAfter() == nil { t.Fatalf("expected retryable websocket connection limit error") } + if got := *retryable.RetryAfter(); got != 0 { + t.Fatalf("retryAfter = %v, want connection-limit fallback 0", got) + } withHeaders, ok := err.(interface{ Headers() http.Header }) if !ok || withHeaders.Headers().Get("retry-after") != "1" { t.Fatalf("headers = %#v, want retry-after", err) } } +func TestParseCodexWebsocketErrorUsesUsageLimitRetryMetadata(t *testing.T) { + err, ok := parseCodexWebsocketError([]byte(`{"type":"error","status":429,"body":{"error":{"type":"usage_limit_reached","message":"usage limit reached","resets_in_seconds":7}}}`)) + if !ok { + t.Fatalf("expected websocket error") + } + + retryable, ok := err.(interface{ RetryAfter() *time.Duration }) + if !ok || retryable.RetryAfter() == nil { + t.Fatalf("expected retryable usage limit websocket error") + } + if got := *retryable.RetryAfter(); got != 7*time.Second { + t.Fatalf("retryAfter = %v, want 7s", got) + } +} + func TestParseCodexWebsocketErrorPreservesWrappedBodyAndHeaders(t *testing.T) { err, ok := parseCodexWebsocketError([]byte(`{"type":"error","status":429,"body":{"error":{"code":"websocket_connection_limit_reached","type":"server_error","message":"too many websocket connections"}},"headers":{"x-request-id":"req-1"}}`)) if !ok { From 6b4bc0a9a852d38da2e330178b7902a9f7f7db7b Mon Sep 17 00:00:00 2001 From: Kenny Date: Sun, 3 May 2026 21:13:37 -0700 Subject: [PATCH 41/51] Align Codex default identity and docs --- README.md | 2 +- README_CN.md | 2 +- README_JA.md | 2 +- internal/runtime/executor/codex_executor.go | 2 +- .../runtime/executor/codex_websockets_executor_test.go | 10 ++++++++++ 5 files changed, 14 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 2fd6937a..bcadeb87 100644 --- a/README.md +++ b/README.md @@ -53,7 +53,7 @@ VisionCoder is also offering our users a limited-time Date: Mon, 4 May 2026 16:45:25 +0800 Subject: [PATCH 42/51] fix(executor): adjust ApplyThinking order and add payload override test - Moved `ApplyThinking` logic earlier in `openai_compat_executor` to align with configuration application sequence. - Added test to verify payload override precedence over Thinking suffix configuration. --- .../executor/openai_compat_executor.go | 18 ++++---- .../openai_compat_executor_compact_test.go | 44 +++++++++++++++++++ 2 files changed, 54 insertions(+), 8 deletions(-) diff --git a/internal/runtime/executor/openai_compat_executor.go b/internal/runtime/executor/openai_compat_executor.go index 4e44a7ae..63be2d3c 100644 --- a/internal/runtime/executor/openai_compat_executor.go +++ b/internal/runtime/executor/openai_compat_executor.go @@ -96,6 +96,12 @@ func (e *OpenAICompatExecutor) Execute(ctx context.Context, auth *cliproxyauth.A originalPayload := originalPayloadSource originalTranslated := sdktranslator.TranslateRequest(from, to, baseModel, originalPayload, opts.Stream) translated := sdktranslator.TranslateRequest(from, to, baseModel, req.Payload, opts.Stream) + + translated, err = thinking.ApplyThinking(translated, req.Model, from.String(), to.String(), e.Identifier()) + if err != nil { + return resp, err + } + requestedModel := helps.PayloadRequestedModel(opts, req.Model) requestPath := helps.PayloadRequestPath(opts) translated = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", translated, originalTranslated, requestedModel, requestPath) @@ -105,11 +111,6 @@ func (e *OpenAICompatExecutor) Execute(ctx context.Context, auth *cliproxyauth.A } } - translated, err = thinking.ApplyThinking(translated, req.Model, from.String(), to.String(), e.Identifier()) - if err != nil { - return resp, err - } - url := strings.TrimSuffix(baseURL, "/") + endpoint httpReq, err := http.NewRequestWithContext(ctx, http.MethodPost, url, bytes.NewReader(translated)) if err != nil { @@ -199,15 +200,16 @@ func (e *OpenAICompatExecutor) ExecuteStream(ctx context.Context, auth *cliproxy originalPayload := originalPayloadSource originalTranslated := sdktranslator.TranslateRequest(from, to, baseModel, originalPayload, true) translated := sdktranslator.TranslateRequest(from, to, baseModel, req.Payload, true) - requestedModel := helps.PayloadRequestedModel(opts, req.Model) - requestPath := helps.PayloadRequestPath(opts) - translated = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", translated, originalTranslated, requestedModel, requestPath) translated, err = thinking.ApplyThinking(translated, req.Model, from.String(), to.String(), e.Identifier()) if err != nil { return nil, err } + requestedModel := helps.PayloadRequestedModel(opts, req.Model) + requestPath := helps.PayloadRequestPath(opts) + translated = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", translated, originalTranslated, requestedModel, requestPath) + // Request usage data in the final streaming chunk so that token statistics // are captured even when the upstream is an OpenAI-compatible provider. translated, _ = sjson.SetBytes(translated, "stream_options.include_usage", true) diff --git a/internal/runtime/executor/openai_compat_executor_compact_test.go b/internal/runtime/executor/openai_compat_executor_compact_test.go index fe281262..ac9d9b32 100644 --- a/internal/runtime/executor/openai_compat_executor_compact_test.go +++ b/internal/runtime/executor/openai_compat_executor_compact_test.go @@ -56,3 +56,47 @@ func TestOpenAICompatExecutorCompactPassthrough(t *testing.T) { t.Fatalf("payload = %s", string(resp.Payload)) } } + +func TestOpenAICompatExecutorPayloadOverrideWinsOverThinkingSuffix(t *testing.T) { + var gotBody []byte + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + body, _ := io.ReadAll(r.Body) + gotBody = body + w.Header().Set("Content-Type", "application/json") + _, _ = w.Write([]byte(`{"id":"chatcmpl_1","object":"chat.completion","choices":[{"index":0,"message":{"role":"assistant","content":"ok"},"finish_reason":"stop"}],"usage":{"prompt_tokens":1,"completion_tokens":1,"total_tokens":2}}`)) + })) + defer server.Close() + + executor := NewOpenAICompatExecutor("openai-compatibility", &config.Config{ + Payload: config.PayloadConfig{ + Override: []config.PayloadRule{ + { + Models: []config.PayloadModelRule{ + {Name: "custom-openai", Protocol: "openai"}, + }, + Params: map[string]any{ + "reasoning_effort": "low", + }, + }, + }, + }, + }) + auth := &cliproxyauth.Auth{Attributes: map[string]string{ + "base_url": server.URL + "/v1", + "api_key": "test", + }} + payload := []byte(`{"model":"custom-openai(high)","messages":[{"role":"user","content":"hi"}]}`) + _, err := executor.Execute(context.Background(), auth, cliproxyexecutor.Request{ + Model: "custom-openai(high)", + Payload: payload, + }, cliproxyexecutor.Options{ + SourceFormat: sdktranslator.FromString("openai"), + Stream: false, + }) + if err != nil { + t.Fatalf("Execute error: %v", err) + } + if got := gjson.GetBytes(gotBody, "reasoning_effort").String(); got != "low" { + t.Fatalf("reasoning_effort = %q, want %q; body=%s", got, "low", string(gotBody)) + } +} From 85c015065302ed17bac32b0ec05d94b59cf46eb6 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 16:57:50 +0800 Subject: [PATCH 43/51] feat(translator): add token usage tracking and improve usage handling - Introduced `claudeUsageTokens` struct for detailed token usage tracking. - Replaced `calculateClaudeUsageTokens` with `Merge` and `OpenAIUsage` methods for better modularity. - Enhanced integration of usage tokens into response processing, enabling more accurate reporting of token details. Fixed: #2419 --- .../claude_openai_response.go | 58 ++++++++++++++----- .../claude_openai_response_test.go | 58 +++++++++++++++++++ 2 files changed, 103 insertions(+), 13 deletions(-) diff --git a/internal/translator/claude/openai/chat-completions/claude_openai_response.go b/internal/translator/claude/openai/chat-completions/claude_openai_response.go index 1fd3f2ae..99c75238 100644 --- a/internal/translator/claude/openai/chat-completions/claude_openai_response.go +++ b/internal/translator/claude/openai/chat-completions/claude_openai_response.go @@ -25,10 +25,19 @@ type ConvertAnthropicResponseToOpenAIParams struct { CreatedAt int64 ResponseID string FinishReason string + Usage claudeUsageTokens // Tool calls accumulator for streaming ToolCallsAccumulator map[int]*ToolCallAccumulator } +type claudeUsageTokens struct { + InputTokens int64 + OutputTokens int64 + CacheCreationInputTokens int64 + CacheReadInputTokens int64 + HasUsage bool +} + // ToolCallAccumulator holds the state for accumulating tool call data type ToolCallAccumulator struct { ID string @@ -36,15 +45,30 @@ type ToolCallAccumulator struct { Arguments strings.Builder } -func calculateClaudeUsageTokens(usage gjson.Result) (promptTokens, completionTokens, totalTokens, cachedTokens int64) { - inputTokens := usage.Get("input_tokens").Int() - completionTokens = usage.Get("output_tokens").Int() - cachedTokens = usage.Get("cache_read_input_tokens").Int() - cacheCreationInputTokens := usage.Get("cache_creation_input_tokens").Int() +func (u *claudeUsageTokens) Merge(usage gjson.Result) { + if !usage.Exists() { + return + } + u.HasUsage = true + if inputTokens := usage.Get("input_tokens"); inputTokens.Exists() { + u.InputTokens = inputTokens.Int() + } + if outputTokens := usage.Get("output_tokens"); outputTokens.Exists() { + u.OutputTokens = outputTokens.Int() + } + if cacheCreationInputTokens := usage.Get("cache_creation_input_tokens"); cacheCreationInputTokens.Exists() { + u.CacheCreationInputTokens = cacheCreationInputTokens.Int() + } + if cacheReadInputTokens := usage.Get("cache_read_input_tokens"); cacheReadInputTokens.Exists() { + u.CacheReadInputTokens = cacheReadInputTokens.Int() + } +} - promptTokens = inputTokens + cacheCreationInputTokens + cachedTokens +func (u claudeUsageTokens) OpenAIUsage() (promptTokens, completionTokens, totalTokens, cachedTokens int64) { + cachedTokens = u.CacheReadInputTokens + promptTokens = u.InputTokens + u.CacheCreationInputTokens + cachedTokens + completionTokens = u.OutputTokens totalTokens = promptTokens + completionTokens - return promptTokens, completionTokens, totalTokens, cachedTokens } @@ -112,6 +136,7 @@ func ConvertClaudeResponseToOpenAI(_ context.Context, modelName string, original if (*param).(*ConvertAnthropicResponseToOpenAIParams).ToolCallsAccumulator == nil { (*param).(*ConvertAnthropicResponseToOpenAIParams).ToolCallsAccumulator = make(map[int]*ToolCallAccumulator) } + (*param).(*ConvertAnthropicResponseToOpenAIParams).Usage.Merge(message.Get("usage")) } return [][]byte{template} @@ -215,7 +240,8 @@ func ConvertClaudeResponseToOpenAI(_ context.Context, modelName string, original // Handle usage information for token counts if usage := root.Get("usage"); usage.Exists() { - promptTokens, completionTokens, totalTokens, cachedTokens := calculateClaudeUsageTokens(usage) + (*param).(*ConvertAnthropicResponseToOpenAIParams).Usage.Merge(usage) + promptTokens, completionTokens, totalTokens, cachedTokens := (*param).(*ConvertAnthropicResponseToOpenAIParams).Usage.OpenAIUsage() template, _ = sjson.SetBytes(template, "usage.prompt_tokens", promptTokens) template, _ = sjson.SetBytes(template, "usage.completion_tokens", completionTokens) template, _ = sjson.SetBytes(template, "usage.total_tokens", totalTokens) @@ -296,6 +322,7 @@ func ConvertClaudeResponseToOpenAINonStream(_ context.Context, _ string, origina var stopReason string var contentParts []string var reasoningParts []string + usageTokens := claudeUsageTokens{} toolCallsAccumulator := make(map[int]*ToolCallAccumulator) for _, chunk := range chunks { @@ -309,6 +336,7 @@ func ConvertClaudeResponseToOpenAINonStream(_ context.Context, _ string, origina messageID = message.Get("id").String() model = message.Get("model").String() createdAt = time.Now().Unix() + usageTokens.Merge(message.Get("usage")) } case "content_block_start": @@ -371,15 +399,19 @@ func ConvertClaudeResponseToOpenAINonStream(_ context.Context, _ string, origina } } if usage := root.Get("usage"); usage.Exists() { - promptTokens, completionTokens, totalTokens, cachedTokens := calculateClaudeUsageTokens(usage) - out, _ = sjson.SetBytes(out, "usage.prompt_tokens", promptTokens) - out, _ = sjson.SetBytes(out, "usage.completion_tokens", completionTokens) - out, _ = sjson.SetBytes(out, "usage.total_tokens", totalTokens) - out, _ = sjson.SetBytes(out, "usage.prompt_tokens_details.cached_tokens", cachedTokens) + usageTokens.Merge(usage) } } } + if usageTokens.HasUsage { + promptTokens, completionTokens, totalTokens, cachedTokens := usageTokens.OpenAIUsage() + out, _ = sjson.SetBytes(out, "usage.prompt_tokens", promptTokens) + out, _ = sjson.SetBytes(out, "usage.completion_tokens", completionTokens) + out, _ = sjson.SetBytes(out, "usage.total_tokens", totalTokens) + out, _ = sjson.SetBytes(out, "usage.prompt_tokens_details.cached_tokens", cachedTokens) + } + // Set basic response fields including message ID, creation time, and model out, _ = sjson.SetBytes(out, "id", messageID) out, _ = sjson.SetBytes(out, "created", createdAt) diff --git a/internal/translator/claude/openai/chat-completions/claude_openai_response_test.go b/internal/translator/claude/openai/chat-completions/claude_openai_response_test.go index 7bd6eb1f..5a9a6d3a 100644 --- a/internal/translator/claude/openai/chat-completions/claude_openai_response_test.go +++ b/internal/translator/claude/openai/chat-completions/claude_openai_response_test.go @@ -37,6 +37,44 @@ func TestConvertClaudeResponseToOpenAI_StreamUsageIncludesCachedTokens(t *testin } } +func TestConvertClaudeResponseToOpenAI_StreamUsageMergesMessageStartUsage(t *testing.T) { + ctx := context.Background() + var param any + + ConvertClaudeResponseToOpenAI( + ctx, + "claude-opus-4-6", + nil, + nil, + []byte(`data: {"type":"message_start","message":{"id":"msg_123","model":"claude-opus-4-6","usage":{"input_tokens":13,"output_tokens":1,"cache_read_input_tokens":22000,"cache_creation_input_tokens":31}}}`), + ¶m, + ) + out := ConvertClaudeResponseToOpenAI( + ctx, + "claude-opus-4-6", + nil, + nil, + []byte(`data: {"type":"message_delta","delta":{"stop_reason":"end_turn"},"usage":{"output_tokens":4}}`), + ¶m, + ) + if len(out) != 1 { + t.Fatalf("expected 1 chunk, got %d", len(out)) + } + + if gotPromptTokens := gjson.GetBytes(out[0], "usage.prompt_tokens").Int(); gotPromptTokens != 22044 { + t.Fatalf("expected prompt_tokens %d, got %d", 22044, gotPromptTokens) + } + if gotCompletionTokens := gjson.GetBytes(out[0], "usage.completion_tokens").Int(); gotCompletionTokens != 4 { + t.Fatalf("expected completion_tokens %d, got %d", 4, gotCompletionTokens) + } + if gotTotalTokens := gjson.GetBytes(out[0], "usage.total_tokens").Int(); gotTotalTokens != 22048 { + t.Fatalf("expected total_tokens %d, got %d", 22048, gotTotalTokens) + } + if gotCachedTokens := gjson.GetBytes(out[0], "usage.prompt_tokens_details.cached_tokens").Int(); gotCachedTokens != 22000 { + t.Fatalf("expected cached_tokens %d, got %d", 22000, gotCachedTokens) + } +} + func TestConvertClaudeResponseToOpenAINonStream_UsageIncludesCachedTokens(t *testing.T) { rawJSON := []byte("data: {\"type\":\"message_start\",\"message\":{\"id\":\"msg_123\",\"model\":\"claude-opus-4-6\"}}\n" + "data: {\"type\":\"message_delta\",\"delta\":{\"stop_reason\":\"end_turn\"},\"usage\":{\"input_tokens\":13,\"output_tokens\":4,\"cache_read_input_tokens\":22000,\"cache_creation_input_tokens\":31}}\n") @@ -56,3 +94,23 @@ func TestConvertClaudeResponseToOpenAINonStream_UsageIncludesCachedTokens(t *tes t.Fatalf("expected cached_tokens %d, got %d", 22000, gotCachedTokens) } } + +func TestConvertClaudeResponseToOpenAINonStream_UsageMergesMessageStartUsage(t *testing.T) { + rawJSON := []byte("data: {\"type\":\"message_start\",\"message\":{\"id\":\"msg_123\",\"model\":\"claude-opus-4-6\",\"usage\":{\"input_tokens\":13,\"output_tokens\":1,\"cache_read_input_tokens\":22000,\"cache_creation_input_tokens\":31}}}\n" + + "data: {\"type\":\"message_delta\",\"delta\":{\"stop_reason\":\"end_turn\"},\"usage\":{\"output_tokens\":4}}\n") + + out := ConvertClaudeResponseToOpenAINonStream(context.Background(), "", nil, nil, rawJSON, nil) + + if gotPromptTokens := gjson.GetBytes(out, "usage.prompt_tokens").Int(); gotPromptTokens != 22044 { + t.Fatalf("expected prompt_tokens %d, got %d", 22044, gotPromptTokens) + } + if gotCompletionTokens := gjson.GetBytes(out, "usage.completion_tokens").Int(); gotCompletionTokens != 4 { + t.Fatalf("expected completion_tokens %d, got %d", 4, gotCompletionTokens) + } + if gotTotalTokens := gjson.GetBytes(out, "usage.total_tokens").Int(); gotTotalTokens != 22048 { + t.Fatalf("expected total_tokens %d, got %d", 22048, gotTotalTokens) + } + if gotCachedTokens := gjson.GetBytes(out, "usage.prompt_tokens_details.cached_tokens").Int(); gotCachedTokens != 22000 { + t.Fatalf("expected cached_tokens %d, got %d", 22000, gotCachedTokens) + } +} From bf6fa402e203a1048f065ee8fc8f3e821f528b7a Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 17:54:16 +0800 Subject: [PATCH 44/51] fix(executor): strip Vertex OpenAI response tool call IDs for consistency - Integrated `StripVertexOpenAIResponsesToolCallIDs` to remove tool call ID data from request bodies and translated requests. - Ensures uniformity and avoids unnecessary payload data propagation. Fixed: #2549 --- .../executor/gemini_vertex_executor.go | 6 +++ .../executor/helps/vertex_payload_helpers.go | 43 +++++++++++++++++++ 2 files changed, 49 insertions(+) create mode 100644 internal/runtime/executor/helps/vertex_payload_helpers.go diff --git a/internal/runtime/executor/gemini_vertex_executor.go b/internal/runtime/executor/gemini_vertex_executor.go index b147fde9..84a84b3d 100644 --- a/internal/runtime/executor/gemini_vertex_executor.go +++ b/internal/runtime/executor/gemini_vertex_executor.go @@ -338,6 +338,7 @@ func (e *GeminiVertexExecutor) executeWithServiceAccount(ctx context.Context, au requestPath := helps.PayloadRequestPath(opts) body = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", body, originalTranslated, requestedModel, requestPath) body, _ = sjson.SetBytes(body, "model", baseModel) + body = helps.StripVertexOpenAIResponsesToolCallIDs(body, from.String()) } action := getVertexAction(baseModel, false) @@ -459,6 +460,7 @@ func (e *GeminiVertexExecutor) executeWithAPIKey(ctx context.Context, auth *clip requestPath := helps.PayloadRequestPath(opts) body = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", body, originalTranslated, requestedModel, requestPath) body, _ = sjson.SetBytes(body, "model", baseModel) + body = helps.StripVertexOpenAIResponsesToolCallIDs(body, from.String()) action := getVertexAction(baseModel, false) if req.Metadata != nil { @@ -570,6 +572,7 @@ func (e *GeminiVertexExecutor) executeStreamWithServiceAccount(ctx context.Conte requestPath := helps.PayloadRequestPath(opts) body = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", body, originalTranslated, requestedModel, requestPath) body, _ = sjson.SetBytes(body, "model", baseModel) + body = helps.StripVertexOpenAIResponsesToolCallIDs(body, from.String()) action := getVertexAction(baseModel, true) baseURL := vertexBaseURL(location) @@ -700,6 +703,7 @@ func (e *GeminiVertexExecutor) executeStreamWithAPIKey(ctx context.Context, auth requestPath := helps.PayloadRequestPath(opts) body = helps.ApplyPayloadConfigWithRoot(e.cfg, baseModel, to.String(), "", body, originalTranslated, requestedModel, requestPath) body, _ = sjson.SetBytes(body, "model", baseModel) + body = helps.StripVertexOpenAIResponsesToolCallIDs(body, from.String()) action := getVertexAction(baseModel, true) // For API key auth, use simpler URL format without project/location @@ -818,6 +822,7 @@ func (e *GeminiVertexExecutor) countTokensWithServiceAccount(ctx context.Context translatedReq = fixGeminiImageAspectRatio(baseModel, translatedReq) translatedReq, _ = sjson.SetBytes(translatedReq, "model", baseModel) + translatedReq = helps.StripVertexOpenAIResponsesToolCallIDs(translatedReq, from.String()) respCtx := context.WithValue(ctx, "alt", opts.Alt) translatedReq, _ = sjson.DeleteBytes(translatedReq, "tools") translatedReq, _ = sjson.DeleteBytes(translatedReq, "generationConfig") @@ -907,6 +912,7 @@ func (e *GeminiVertexExecutor) countTokensWithAPIKey(ctx context.Context, auth * translatedReq = fixGeminiImageAspectRatio(baseModel, translatedReq) translatedReq, _ = sjson.SetBytes(translatedReq, "model", baseModel) + translatedReq = helps.StripVertexOpenAIResponsesToolCallIDs(translatedReq, from.String()) respCtx := context.WithValue(ctx, "alt", opts.Alt) translatedReq, _ = sjson.DeleteBytes(translatedReq, "tools") translatedReq, _ = sjson.DeleteBytes(translatedReq, "generationConfig") diff --git a/internal/runtime/executor/helps/vertex_payload_helpers.go b/internal/runtime/executor/helps/vertex_payload_helpers.go new file mode 100644 index 00000000..4c84fae4 --- /dev/null +++ b/internal/runtime/executor/helps/vertex_payload_helpers.go @@ -0,0 +1,43 @@ +package helps + +import ( + "fmt" + "strings" + + "github.com/tidwall/gjson" + "github.com/tidwall/sjson" +) + +// StripVertexOpenAIResponsesToolCallIDs removes OpenAI Responses call IDs that +// Vertex rejects in Gemini functionCall/functionResponse payloads. +func StripVertexOpenAIResponsesToolCallIDs(payload []byte, sourceFormat string) []byte { + if !strings.EqualFold(strings.TrimSpace(sourceFormat), "openai-response") { + return payload + } + + contents := gjson.GetBytes(payload, "contents") + if !contents.IsArray() { + return payload + } + + out := payload + for contentIndex, content := range contents.Array() { + parts := content.Get("parts") + if !parts.IsArray() { + continue + } + for partIndex, part := range parts.Array() { + if part.Get("functionCall.id").Exists() { + if updated, errDelete := sjson.DeleteBytes(out, fmt.Sprintf("contents.%d.parts.%d.functionCall.id", contentIndex, partIndex)); errDelete == nil { + out = updated + } + } + if part.Get("functionResponse.id").Exists() { + if updated, errDelete := sjson.DeleteBytes(out, fmt.Sprintf("contents.%d.parts.%d.functionResponse.id", contentIndex, partIndex)); errDelete == nil { + out = updated + } + } + } + } + return out +} From c1caa454b35e9990fe93e67ff3111d69d154d84c Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 21:00:33 +0800 Subject: [PATCH 45/51] fix(translator): handle empty tool function names in OpenAI Claude responses - Added check to prevent processing of empty `function.name` values, ensuring valid data is handled. Fixed: #2557 --- .../openai/claude/openai_claude_response.go | 2 +- .../claude/openai_claude_response_test.go | 41 +++++++++++++++++++ 2 files changed, 42 insertions(+), 1 deletion(-) create mode 100644 internal/translator/openai/claude/openai_claude_response_test.go diff --git a/internal/translator/openai/claude/openai_claude_response.go b/internal/translator/openai/claude/openai_claude_response.go index 46c75898..af49d306 100644 --- a/internal/translator/openai/claude/openai_claude_response.go +++ b/internal/translator/openai/claude/openai_claude_response.go @@ -236,7 +236,7 @@ func convertOpenAIStreamingChunkToAnthropic(rawJSON []byte, param *ConvertOpenAI // Handle function name if function := toolCall.Get("function"); function.Exists() { - if name := function.Get("name"); name.Exists() { + if name := function.Get("name"); name.Exists() && name.String() != "" { accumulator.Name = util.MapToolName(param.ToolNameMap, name.String()) stopThinkingContentBlock(param, &results) diff --git a/internal/translator/openai/claude/openai_claude_response_test.go b/internal/translator/openai/claude/openai_claude_response_test.go new file mode 100644 index 00000000..8c36fc3d --- /dev/null +++ b/internal/translator/openai/claude/openai_claude_response_test.go @@ -0,0 +1,41 @@ +package claude + +import ( + "bytes" + "context" + "testing" +) + +func TestConvertOpenAIResponseToClaude_StreamIgnoresNullToolNameDelta(t *testing.T) { + originalRequest := []byte(`{"stream":true}`) + var param any + + firstChunks := ConvertOpenAIResponseToClaude( + context.Background(), + "test-model", + originalRequest, + nil, + []byte(`data: {"id":"chatcmpl_1","model":"test-model","created":1,"choices":[{"index":0,"delta":{"role":"assistant","tool_calls":[{"index":0,"id":"call_1","type":"function","function":{"name":"read_file","arguments":""}}]},"finish_reason":null}]}`), + ¶m, + ) + firstOutput := bytes.Join(firstChunks, nil) + if !bytes.Contains(firstOutput, []byte(`"name":"read_file"`)) { + t.Fatalf("expected first chunk to start read_file tool block, got %s", string(firstOutput)) + } + + secondChunks := ConvertOpenAIResponseToClaude( + context.Background(), + "test-model", + originalRequest, + nil, + []byte(`data: {"id":"chatcmpl_1","model":"test-model","created":1,"choices":[{"index":0,"delta":{"tool_calls":[{"index":0,"function":{"name":null,"arguments":"{\"path\":\"/tmp/a\"}"}}]},"finish_reason":null}]}`), + ¶m, + ) + secondOutput := bytes.Join(secondChunks, nil) + if bytes.Contains(secondOutput, []byte(`content_block_start`)) { + t.Fatalf("did not expect null tool name delta to start a new content block, got %s", string(secondOutput)) + } + if bytes.Contains(secondOutput, []byte(`"name":""`)) { + t.Fatalf("did not expect null tool name delta to emit an empty tool name, got %s", string(secondOutput)) + } +} From ecf1c2590c1b4772e0e0d4d0f0e602d811f15024 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 21:18:18 +0800 Subject: [PATCH 46/51] fix: preserve Antigravity cancellation errors --- internal/runtime/executor/antigravity_executor.go | 11 ++--------- 1 file changed, 2 insertions(+), 9 deletions(-) diff --git a/internal/runtime/executor/antigravity_executor.go b/internal/runtime/executor/antigravity_executor.go index c07680e8..418ed7b1 100644 --- a/internal/runtime/executor/antigravity_executor.go +++ b/internal/runtime/executor/antigravity_executor.go @@ -894,19 +894,12 @@ attemptLoop: reporter.Publish(ctx, detail) } - select { - case out <- cliproxyexecutor.StreamChunk{Payload: payload}: - case <-ctx.Done(): - return - } + out <- cliproxyexecutor.StreamChunk{Payload: payload} } if errScan := scanner.Err(); errScan != nil { helps.RecordAPIResponseError(ctx, e.cfg, errScan) reporter.PublishFailure(ctx) - select { - case out <- cliproxyexecutor.StreamChunk{Err: errScan}: - case <-ctx.Done(): - } + out <- cliproxyexecutor.StreamChunk{Err: errScan} } else { reporter.EnsurePublished(ctx) } From e4a93c02c584108b981d65691600fc87012b86ca Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Mon, 4 May 2026 23:42:26 +0800 Subject: [PATCH 47/51] fix(executor): enhance parsing of OpenAI stream data lines - Added trimming for stream input lines to prevent processing of unnecessary whitespace. - Improved handling of unsupported prefixes and malformed JSON responses, ensuring errors are recorded and propagated appropriately. Fixed: #2690 --- .../executor/openai_compat_executor.go | 24 ++++-- .../openai_compat_executor_compact_test.go | 79 +++++++++++++++++++ 2 files changed, 98 insertions(+), 5 deletions(-) diff --git a/internal/runtime/executor/openai_compat_executor.go b/internal/runtime/executor/openai_compat_executor.go index e0a7bd88..7e81637c 100644 --- a/internal/runtime/executor/openai_compat_executor.go +++ b/internal/runtime/executor/openai_compat_executor.go @@ -283,17 +283,31 @@ func (e *OpenAICompatExecutor) ExecuteStream(ctx context.Context, auth *cliproxy if detail, ok := helps.ParseOpenAIStreamUsage(line); ok { reporter.Publish(ctx, detail) } - if len(line) == 0 { + trimmedLine := bytes.TrimSpace(line) + if len(trimmedLine) == 0 { continue } - if !bytes.HasPrefix(line, []byte("data:")) { + if !bytes.HasPrefix(trimmedLine, []byte("data:")) { + if bytes.HasPrefix(trimmedLine, []byte(":")) || bytes.HasPrefix(trimmedLine, []byte("event:")) || + bytes.HasPrefix(trimmedLine, []byte("id:")) || bytes.HasPrefix(trimmedLine, []byte("retry:")) { + continue + } + if bytes.HasPrefix(trimmedLine, []byte("{")) || bytes.HasPrefix(trimmedLine, []byte("[")) { + streamErr := statusErr{code: http.StatusBadGateway, msg: string(trimmedLine)} + helps.RecordAPIResponseError(ctx, e.cfg, streamErr) + reporter.PublishFailure(ctx) + select { + case out <- cliproxyexecutor.StreamChunk{Err: streamErr}: + case <-ctx.Done(): + } + return + } continue } - // OpenAI-compatible streams are SSE: lines typically prefixed with "data: ". - // Pass through translator; it yields one or more chunks for the target schema. - chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, bytes.Clone(line), ¶m) + // OpenAI-compatible streams must use SSE data lines. + chunks := sdktranslator.TranslateStream(ctx, to, from, req.Model, opts.OriginalRequest, translated, bytes.Clone(trimmedLine), ¶m) for i := range chunks { select { case out <- cliproxyexecutor.StreamChunk{Payload: chunks[i]}: diff --git a/internal/runtime/executor/openai_compat_executor_compact_test.go b/internal/runtime/executor/openai_compat_executor_compact_test.go index ac9d9b32..49b2cccb 100644 --- a/internal/runtime/executor/openai_compat_executor_compact_test.go +++ b/internal/runtime/executor/openai_compat_executor_compact_test.go @@ -5,6 +5,7 @@ import ( "io" "net/http" "net/http/httptest" + "strings" "testing" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" @@ -100,3 +101,81 @@ func TestOpenAICompatExecutorPayloadOverrideWinsOverThinkingSuffix(t *testing.T) t.Fatalf("reasoning_effort = %q, want %q; body=%s", got, "low", string(gotBody)) } } + +func TestOpenAICompatExecutorStreamRejectsPlainJSONAfterBlankLines(t *testing.T) { + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "text/event-stream") + _, _ = w.Write([]byte("\n\n: openrouter processing\n\nevent: error\n")) + _, _ = w.Write([]byte(`{"error":{"message":"upstream failed","type":"server_error"}}` + "\n")) + })) + defer server.Close() + + executor := NewOpenAICompatExecutor("openai-compatibility", &config.Config{}) + auth := &cliproxyauth.Auth{Attributes: map[string]string{ + "base_url": server.URL + "/v1", + "api_key": "test", + }} + result, err := executor.ExecuteStream(context.Background(), auth, cliproxyexecutor.Request{ + Model: "openrouter-model", + Payload: []byte(`{"model":"openrouter-model","messages":[{"role":"user","content":"hi"}],"stream":true}`), + }, cliproxyexecutor.Options{ + SourceFormat: sdktranslator.FromString("openai"), + Stream: true, + }) + if err != nil { + t.Fatalf("ExecuteStream error: %v", err) + } + + var gotErr error + for chunk := range result.Chunks { + if chunk.Err != nil { + gotErr = chunk.Err + break + } + } + if gotErr == nil { + t.Fatalf("expected plain JSON stream error") + } + if status, ok := gotErr.(interface{ StatusCode() int }); !ok || status.StatusCode() != http.StatusBadGateway { + t.Fatalf("stream error status = %v, want %d", gotErr, http.StatusBadGateway) + } + if !strings.Contains(gotErr.Error(), "upstream failed") { + t.Fatalf("stream error = %v", gotErr) + } +} + +func TestOpenAICompatExecutorStreamSkipsKeepAliveUntilDataLine(t *testing.T) { + server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + w.Header().Set("Content-Type", "text/event-stream") + _, _ = w.Write([]byte("\n\n: openrouter processing\n\nevent: ping\nid: 1\nretry: 1000\n")) + _, _ = w.Write([]byte(`data: {"id":"chatcmpl_1","object":"chat.completion.chunk","choices":[{"index":0,"delta":{"content":"hello"},"finish_reason":null}]}` + "\n")) + })) + defer server.Close() + + executor := NewOpenAICompatExecutor("openai-compatibility", &config.Config{}) + auth := &cliproxyauth.Auth{Attributes: map[string]string{ + "base_url": server.URL + "/v1", + "api_key": "test", + }} + result, err := executor.ExecuteStream(context.Background(), auth, cliproxyexecutor.Request{ + Model: "openrouter-model", + Payload: []byte(`{"model":"openrouter-model","messages":[{"role":"user","content":"hi"}],"stream":true}`), + }, cliproxyexecutor.Options{ + SourceFormat: sdktranslator.FromString("openai"), + Stream: true, + }) + if err != nil { + t.Fatalf("ExecuteStream error: %v", err) + } + + var got strings.Builder + for chunk := range result.Chunks { + if chunk.Err != nil { + t.Fatalf("unexpected stream error: %v", chunk.Err) + } + got.Write(chunk.Payload) + } + if gjson.Get(got.String(), "choices.0.delta.content").String() != "hello" { + t.Fatalf("stream payload = %s", got.String()) + } +} From ba5d8ca7336e78ca2b7c6134638a4054b589c330 Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Tue, 5 May 2026 01:47:53 +0800 Subject: [PATCH 48/51] feat(usage): add support for requested model alias handling - Introduced methods for setting and retrieving model aliases in execution and usage contexts. - Enhanced `UsageReporter` and related structures to include client-requested aliases. - Updated tests to validate alias propagation and ensure correct usage reporting. - Adjusted metadata handling in CLIProxyAPI executors to address alias integration. --- internal/redisqueue/plugin.go | 6 ++++ internal/redisqueue/plugin_test.go | 6 ++++ .../runtime/executor/helps/usage_helpers.go | 7 ++++ .../executor/helps/usage_helpers_test.go | 14 ++++++++ sdk/api/handlers/handlers.go | 6 ++-- sdk/cliproxy/auth/conductor.go | 35 +++++++++++++++++++ .../conductor_oauth_alias_suspension_test.go | 25 +++++++++++-- sdk/cliproxy/usage/manager.go | 32 +++++++++++++++++ 8 files changed, 125 insertions(+), 6 deletions(-) diff --git a/internal/redisqueue/plugin.go b/internal/redisqueue/plugin.go index 97168419..b33bc8fd 100644 --- a/internal/redisqueue/plugin.go +++ b/internal/redisqueue/plugin.go @@ -33,6 +33,10 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec if modelName == "" { modelName = "unknown" } + aliasName := strings.TrimSpace(record.Alias) + if aliasName == "" { + aliasName = modelName + } provider := strings.TrimSpace(record.Provider) if provider == "" { provider = "unknown" @@ -76,6 +80,7 @@ func (p *usageQueuePlugin) HandleUsage(ctx context.Context, record coreusage.Rec requestDetail: detail, Provider: provider, Model: modelName, + Alias: aliasName, Endpoint: resolveEndpoint(ctx), AuthType: authType, APIKey: apiKey, @@ -91,6 +96,7 @@ type queuedUsageDetail struct { requestDetail Provider string `json:"provider"` Model string `json:"model"` + Alias string `json:"alias"` Endpoint string `json:"endpoint"` AuthType string `json:"auth_type"` APIKey string `json:"api_key"` diff --git a/internal/redisqueue/plugin_test.go b/internal/redisqueue/plugin_test.go index 0cc8b9b9..8dcade90 100644 --- a/internal/redisqueue/plugin_test.go +++ b/internal/redisqueue/plugin_test.go @@ -24,6 +24,7 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndSuccess(t *testing.T) { plugin.HandleUsage(ctx, coreusage.Record{ Provider: "openai", Model: "gpt-5.4", + Alias: "client-gpt", APIKey: "test-key", AuthIndex: "0", AuthType: "apikey", @@ -40,6 +41,7 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndSuccess(t *testing.T) { payload := popSinglePayload(t) requireStringField(t, payload, "provider", "openai") requireStringField(t, payload, "model", "gpt-5.4") + requireStringField(t, payload, "alias", "client-gpt") requireStringField(t, payload, "endpoint", "POST /v1/chat/completions") requireStringField(t, payload, "auth_type", "apikey") requireStringField(t, payload, "request_id", "ctx-request-id") @@ -58,6 +60,7 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndFailureAndGinRequestID(t plugin.HandleUsage(ctx, coreusage.Record{ Provider: "openai", Model: "gpt-5.4-mini", + Alias: "client-mini", APIKey: "test-key", AuthIndex: "0", AuthType: "apikey", @@ -74,6 +77,7 @@ func TestUsageQueuePluginPayloadIncludesStableFieldsAndFailureAndGinRequestID(t payload := popSinglePayload(t) requireStringField(t, payload, "provider", "openai") requireStringField(t, payload, "model", "gpt-5.4-mini") + requireStringField(t, payload, "alias", "client-mini") requireStringField(t, payload, "endpoint", "GET /v1/responses") requireStringField(t, payload, "auth_type", "apikey") requireStringField(t, payload, "request_id", "gin-request-id") @@ -102,6 +106,7 @@ func TestUsageQueuePluginAsyncIgnoresRecycledGinContext(t *testing.T) { mgr.Publish(ctx, coreusage.Record{ Provider: "openai", Model: "gpt-5.4", + Alias: "client-gpt", APIKey: "test-key", AuthIndex: "0", AuthType: "apikey", @@ -117,6 +122,7 @@ func TestUsageQueuePluginAsyncIgnoresRecycledGinContext(t *testing.T) { payload := waitForSinglePayload(t, 2*time.Second) requireStringField(t, payload, "endpoint", "POST /v1/chat/completions") + requireStringField(t, payload, "alias", "client-gpt") requireStringField(t, payload, "request_id", "ctx-request-id") requireBoolField(t, payload, "failed", true) }) diff --git a/internal/runtime/executor/helps/usage_helpers.go b/internal/runtime/executor/helps/usage_helpers.go index c5e258c8..312a1d35 100644 --- a/internal/runtime/executor/helps/usage_helpers.go +++ b/internal/runtime/executor/helps/usage_helpers.go @@ -18,6 +18,7 @@ import ( type UsageReporter struct { provider string model string + alias string authID string authIndex string authType string @@ -29,9 +30,14 @@ type UsageReporter struct { func NewUsageReporter(ctx context.Context, provider, model string, auth *cliproxyauth.Auth) *UsageReporter { apiKey := APIKeyFromContext(ctx) + alias := usage.RequestedModelAliasFromContext(ctx) + if alias == "" { + alias = model + } reporter := &UsageReporter{ provider: provider, model: model, + alias: strings.TrimSpace(alias), requestedAt: time.Now(), apiKey: apiKey, source: resolveUsageSource(auth, apiKey), @@ -139,6 +145,7 @@ func (r *UsageReporter) buildRecordForModel(model string, detail usage.Detail, f return usage.Record{ Provider: r.provider, Model: model, + Alias: r.alias, Source: r.source, APIKey: r.apiKey, AuthID: r.authID, diff --git a/internal/runtime/executor/helps/usage_helpers_test.go b/internal/runtime/executor/helps/usage_helpers_test.go index c77335fd..ef2c7de5 100644 --- a/internal/runtime/executor/helps/usage_helpers_test.go +++ b/internal/runtime/executor/helps/usage_helpers_test.go @@ -1,6 +1,7 @@ package helps import ( + "context" "testing" "time" @@ -107,6 +108,19 @@ func TestUsageReporterBuildRecordIncludesLatency(t *testing.T) { } } +func TestUsageReporterBuildRecordIncludesRequestedModelAlias(t *testing.T) { + ctx := usage.WithRequestedModelAlias(context.Background(), "client-gpt") + reporter := NewUsageReporter(ctx, "openai", "gpt-5.4", nil) + + record := reporter.buildRecord(usage.Detail{TotalTokens: 3}, false) + if record.Model != "gpt-5.4" { + t.Fatalf("model = %q, want %q", record.Model, "gpt-5.4") + } + if record.Alias != "client-gpt" { + t.Fatalf("alias = %q, want %q", record.Alias, "client-gpt") + } +} + func TestUsageReporterBuildAdditionalModelRecordSkipsZeroTokens(t *testing.T) { reporter := &UsageReporter{ provider: "codex", diff --git a/sdk/api/handlers/handlers.go b/sdk/api/handlers/handlers.go index 52b2a4fd..e89227aa 100644 --- a/sdk/api/handlers/handlers.go +++ b/sdk/api/handlers/handlers.go @@ -539,7 +539,7 @@ func (h *BaseAPIHandler) ExecuteWithAuthManager(ctx context.Context, handlerType return nil, nil, errMsg } reqMeta := requestExecutionMetadata(ctx) - reqMeta[coreexecutor.RequestedModelMetadataKey] = normalizedModel + reqMeta[coreexecutor.RequestedModelMetadataKey] = modelName payload := rawJSON if len(payload) == 0 { payload = nil @@ -587,7 +587,7 @@ func (h *BaseAPIHandler) ExecuteCountWithAuthManager(ctx context.Context, handle return nil, nil, errMsg } reqMeta := requestExecutionMetadata(ctx) - reqMeta[coreexecutor.RequestedModelMetadataKey] = normalizedModel + reqMeta[coreexecutor.RequestedModelMetadataKey] = modelName payload := rawJSON if len(payload) == 0 { payload = nil @@ -639,7 +639,7 @@ func (h *BaseAPIHandler) ExecuteStreamWithAuthManager(ctx context.Context, handl return nil, nil, errChan } reqMeta := requestExecutionMetadata(ctx) - reqMeta[coreexecutor.RequestedModelMetadataKey] = normalizedModel + reqMeta[coreexecutor.RequestedModelMetadataKey] = modelName payload := rawJSON if len(payload) == 0 { payload = nil diff --git a/sdk/cliproxy/auth/conductor.go b/sdk/cliproxy/auth/conductor.go index d2a3db18..ab3eca49 100644 --- a/sdk/cliproxy/auth/conductor.go +++ b/sdk/cliproxy/auth/conductor.go @@ -22,6 +22,7 @@ import ( "github.com/router-for-me/CLIProxyAPI/v6/internal/thinking" "github.com/router-for-me/CLIProxyAPI/v6/internal/util" cliproxyexecutor "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/executor" + coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" log "github.com/sirupsen/logrus" ) @@ -827,6 +828,7 @@ func (m *Manager) executeStreamWithModelPool(ctx context.Context, executor Provi if executor == nil { return nil, &Error{Code: "executor_not_found", Message: "executor not registered"} } + ctx = contextWithRequestedModelAlias(ctx, opts, routeModel) var lastErr error for idx, execModel := range execModels { resultModel := m.stateModelForExecution(auth, routeModel, execModel, pooled) @@ -1319,6 +1321,7 @@ func (m *Manager) executeMixedOnce(ctx context.Context, providers []string, req execCtx = context.WithValue(execCtx, roundTripperContextKey{}, rt) execCtx = context.WithValue(execCtx, "cliproxy.roundtripper", rt) } + execCtx = contextWithRequestedModelAlias(execCtx, opts, routeModel) models, pooled := m.preparedExecutionModels(auth, routeModel) if len(models) == 0 { @@ -1397,6 +1400,7 @@ func (m *Manager) executeCountMixedOnce(ctx context.Context, providers []string, execCtx = context.WithValue(execCtx, roundTripperContextKey{}, rt) execCtx = context.WithValue(execCtx, "cliproxy.roundtripper", rt) } + execCtx = contextWithRequestedModelAlias(execCtx, opts, routeModel) models, pooled := m.preparedExecutionModels(auth, routeModel) if len(models) == 0 { @@ -1534,6 +1538,36 @@ func hasRequestedModelMetadata(meta map[string]any) bool { } } +func contextWithRequestedModelAlias(ctx context.Context, opts cliproxyexecutor.Options, fallback string) context.Context { + alias := requestedModelAliasFromOptions(opts, fallback) + return coreusage.WithRequestedModelAlias(ctx, alias) +} + +func requestedModelAliasFromOptions(opts cliproxyexecutor.Options, fallback string) string { + fallback = strings.TrimSpace(fallback) + if len(opts.Metadata) == 0 { + return fallback + } + raw, ok := opts.Metadata[cliproxyexecutor.RequestedModelMetadataKey] + if !ok || raw == nil { + return fallback + } + switch value := raw.(type) { + case string: + if strings.TrimSpace(value) == "" { + return fallback + } + return strings.TrimSpace(value) + case []byte: + if len(value) == 0 { + return fallback + } + return strings.TrimSpace(string(value)) + default: + return fallback + } +} + func pinnedAuthIDFromMetadata(meta map[string]any) string { if len(meta) == 0 { return "" @@ -3096,6 +3130,7 @@ func (m *Manager) tryAntigravityCreditsExecute(ctx context.Context, req cliproxy creditsCtx = context.WithValue(creditsCtx, "cliproxy.roundtripper", rt) } creditsOpts := ensureRequestedModelMetadata(opts, routeModel) + creditsCtx = contextWithRequestedModelAlias(creditsCtx, creditsOpts, routeModel) publishSelectedAuthMetadata(creditsOpts.Metadata, c.auth.ID) models := m.executionModelCandidates(c.auth, routeModel) if len(models) == 0 { diff --git a/sdk/cliproxy/auth/conductor_oauth_alias_suspension_test.go b/sdk/cliproxy/auth/conductor_oauth_alias_suspension_test.go index 8bc779e5..b4b72204 100644 --- a/sdk/cliproxy/auth/conductor_oauth_alias_suspension_test.go +++ b/sdk/cliproxy/auth/conductor_oauth_alias_suspension_test.go @@ -10,20 +10,23 @@ import ( internalconfig "github.com/router-for-me/CLIProxyAPI/v6/internal/config" "github.com/router-for-me/CLIProxyAPI/v6/internal/registry" cliproxyexecutor "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/executor" + coreusage "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/usage" ) type aliasRoutingExecutor struct { id string - mu sync.Mutex - executeModels []string + mu sync.Mutex + executeModels []string + executeAliases []string } func (e *aliasRoutingExecutor) Identifier() string { return e.id } -func (e *aliasRoutingExecutor) Execute(_ context.Context, _ *Auth, req cliproxyexecutor.Request, _ cliproxyexecutor.Options) (cliproxyexecutor.Response, error) { +func (e *aliasRoutingExecutor) Execute(ctx context.Context, _ *Auth, req cliproxyexecutor.Request, _ cliproxyexecutor.Options) (cliproxyexecutor.Response, error) { e.mu.Lock() e.executeModels = append(e.executeModels, req.Model) + e.executeAliases = append(e.executeAliases, coreusage.RequestedModelAliasFromContext(ctx)) e.mu.Unlock() return cliproxyexecutor.Response{Payload: []byte(req.Model)}, nil } @@ -52,6 +55,14 @@ func (e *aliasRoutingExecutor) ExecuteModels() []string { return out } +func (e *aliasRoutingExecutor) ExecuteAliases() []string { + e.mu.Lock() + defer e.mu.Unlock() + out := make([]string, len(e.executeAliases)) + copy(out, e.executeAliases) + return out +} + func TestManagerExecute_OAuthAliasBypassesBlockedRouteModel(t *testing.T) { const ( provider = "antigravity" @@ -108,4 +119,12 @@ func TestManagerExecute_OAuthAliasBypassesBlockedRouteModel(t *testing.T) { if gotModels[0] != targetModel { t.Fatalf("execute model = %q, want %q", gotModels[0], targetModel) } + + gotAliases := executor.ExecuteAliases() + if len(gotAliases) != 1 { + t.Fatalf("execute aliases len = %d, want 1", len(gotAliases)) + } + if gotAliases[0] != routeModel { + t.Fatalf("execute alias = %q, want %q", gotAliases[0], routeModel) + } } diff --git a/sdk/cliproxy/usage/manager.go b/sdk/cliproxy/usage/manager.go index c3d95f66..72405d75 100644 --- a/sdk/cliproxy/usage/manager.go +++ b/sdk/cliproxy/usage/manager.go @@ -2,6 +2,7 @@ package usage import ( "context" + "strings" "sync" "time" @@ -12,6 +13,7 @@ import ( type Record struct { Provider string Model string + Alias string APIKey string AuthID string AuthIndex string @@ -32,6 +34,36 @@ type Detail struct { TotalTokens int64 } +type requestedModelAliasContextKey struct{} + +// WithRequestedModelAlias stores the client-requested model name for usage sinks. +func WithRequestedModelAlias(ctx context.Context, alias string) context.Context { + if ctx == nil { + ctx = context.Background() + } + alias = strings.TrimSpace(alias) + if alias == "" { + return ctx + } + return context.WithValue(ctx, requestedModelAliasContextKey{}, alias) +} + +// RequestedModelAliasFromContext returns the client-requested model name stored in ctx. +func RequestedModelAliasFromContext(ctx context.Context) string { + if ctx == nil { + return "" + } + raw := ctx.Value(requestedModelAliasContextKey{}) + switch value := raw.(type) { + case string: + return strings.TrimSpace(value) + case []byte: + return strings.TrimSpace(string(value)) + default: + return "" + } +} + // Plugin consumes usage records emitted by the proxy runtime. type Plugin interface { HandleUsage(ctx context.Context, record Record) From 61b39d49bd8cad26c8d74eb0bd0f6b8fda16ab2c Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Tue, 5 May 2026 02:53:04 +0800 Subject: [PATCH 49/51] feat(management): add usage record retrieval endpoint - Implemented `/v0/management/usage` endpoint for fetching queued usage records from Redis. - Included validation for `count` parameter to ensure positive integers. - Added unit tests for queue retrieval and validation, with authentication validation in integration tests. - Updated management routing to include the new endpoint. --- internal/api/handlers/management/usage.go | 55 +++++++++++ .../api/handlers/management/usage_test.go | 98 +++++++++++++++++++ internal/api/server.go | 1 + internal/api/server_test.go | 55 +++++++++++ 4 files changed, 209 insertions(+) create mode 100644 internal/api/handlers/management/usage.go create mode 100644 internal/api/handlers/management/usage_test.go diff --git a/internal/api/handlers/management/usage.go b/internal/api/handlers/management/usage.go new file mode 100644 index 00000000..8cb175eb --- /dev/null +++ b/internal/api/handlers/management/usage.go @@ -0,0 +1,55 @@ +package management + +import ( + "encoding/json" + "errors" + "net/http" + "strconv" + "strings" + + "github.com/gin-gonic/gin" + "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" +) + +type usageQueueRecord []byte + +func (r usageQueueRecord) MarshalJSON() ([]byte, error) { + if json.Valid(r) { + return append([]byte(nil), r...), nil + } + return json.Marshal(string(r)) +} + +// GetUsage pops queued usage records from the Redis-compatible usage queue. +func (h *Handler) GetUsage(c *gin.Context) { + if h == nil { + c.JSON(http.StatusInternalServerError, gin.H{"error": "handler unavailable"}) + return + } + + count, errCount := parseUsageQueueCount(c.Query("count")) + if errCount != nil { + c.JSON(http.StatusBadRequest, gin.H{"error": errCount.Error()}) + return + } + + items := redisqueue.PopOldest(count) + records := make([]usageQueueRecord, 0, len(items)) + for _, item := range items { + records = append(records, usageQueueRecord(append([]byte(nil), item...))) + } + + c.JSON(http.StatusOK, records) +} + +func parseUsageQueueCount(value string) (int, error) { + value = strings.TrimSpace(value) + if value == "" { + return 1, nil + } + count, errCount := strconv.Atoi(value) + if errCount != nil || count <= 0 { + return 0, errors.New("count must be a positive integer") + } + return count, nil +} diff --git a/internal/api/handlers/management/usage_test.go b/internal/api/handlers/management/usage_test.go new file mode 100644 index 00000000..5c5f5c69 --- /dev/null +++ b/internal/api/handlers/management/usage_test.go @@ -0,0 +1,98 @@ +package management + +import ( + "encoding/json" + "net/http" + "net/http/httptest" + "testing" + + "github.com/gin-gonic/gin" + "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" +) + +func TestGetUsagePopsRequestedRecords(t *testing.T) { + gin.SetMode(gin.TestMode) + withManagementUsageQueue(t, func() { + redisqueue.Enqueue([]byte(`{"id":1}`)) + redisqueue.Enqueue([]byte(`{"id":2}`)) + redisqueue.Enqueue([]byte(`{"id":3}`)) + + rec := httptest.NewRecorder() + ginCtx, _ := gin.CreateTestContext(rec) + ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + + h := &Handler{} + h.GetUsage(ginCtx) + + if rec.Code != http.StatusOK { + t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusOK, rec.Body.String()) + } + + var payload []json.RawMessage + if errUnmarshal := json.Unmarshal(rec.Body.Bytes(), &payload); errUnmarshal != nil { + t.Fatalf("unmarshal response: %v", errUnmarshal) + } + if len(payload) != 2 { + t.Fatalf("response records = %d, want 2", len(payload)) + } + requireRecordID(t, payload[0], 1) + requireRecordID(t, payload[1], 2) + + remaining := redisqueue.PopOldest(10) + if len(remaining) != 1 || string(remaining[0]) != `{"id":3}` { + t.Fatalf("remaining queue = %q, want third item only", remaining) + } + }) +} + +func TestGetUsageInvalidCountDoesNotPop(t *testing.T) { + gin.SetMode(gin.TestMode) + withManagementUsageQueue(t, func() { + redisqueue.Enqueue([]byte(`{"id":1}`)) + + rec := httptest.NewRecorder() + ginCtx, _ := gin.CreateTestContext(rec) + ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=0", nil) + + h := &Handler{} + h.GetUsage(ginCtx) + + if rec.Code != http.StatusBadRequest { + t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusBadRequest, rec.Body.String()) + } + + remaining := redisqueue.PopOldest(10) + if len(remaining) != 1 || string(remaining[0]) != `{"id":1}` { + t.Fatalf("remaining queue = %q, want original item", remaining) + } + }) +} + +func withManagementUsageQueue(t *testing.T, fn func()) { + t.Helper() + + prevQueueEnabled := redisqueue.Enabled() + redisqueue.SetEnabled(false) + redisqueue.SetEnabled(true) + + defer func() { + redisqueue.SetEnabled(false) + redisqueue.SetEnabled(prevQueueEnabled) + }() + + fn() +} + +func requireRecordID(t *testing.T, raw json.RawMessage, want int) { + t.Helper() + + var payload struct { + ID int `json:"id"` + } + if errUnmarshal := json.Unmarshal(raw, &payload); errUnmarshal != nil { + t.Fatalf("unmarshal record: %v", errUnmarshal) + } + if payload.ID != want { + t.Fatalf("record id = %d, want %d", payload.ID, want) + } +} diff --git a/internal/api/server.go b/internal/api/server.go index 2e89ac5a..5c43db48 100644 --- a/internal/api/server.go +++ b/internal/api/server.go @@ -551,6 +551,7 @@ func (s *Server) registerManagementRoutes() { mgmt.PATCH("/api-keys", s.mgmt.PatchAPIKeys) mgmt.DELETE("/api-keys", s.mgmt.DeleteAPIKeys) mgmt.GET("/api-key-usage", s.mgmt.GetAPIKeyUsage) + mgmt.GET("/usage", s.mgmt.GetUsage) mgmt.GET("/gemini-api-key", s.mgmt.GetGeminiKeys) mgmt.PUT("/gemini-api-key", s.mgmt.PutGeminiKeys) diff --git a/internal/api/server_test.go b/internal/api/server_test.go index db1ef27d..d5718091 100644 --- a/internal/api/server_test.go +++ b/internal/api/server_test.go @@ -13,6 +13,7 @@ import ( gin "github.com/gin-gonic/gin" proxyconfig "github.com/router-for-me/CLIProxyAPI/v6/internal/config" internallogging "github.com/router-for-me/CLIProxyAPI/v6/internal/logging" + "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" sdkaccess "github.com/router-for-me/CLIProxyAPI/v6/sdk/access" "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth" sdkconfig "github.com/router-for-me/CLIProxyAPI/v6/sdk/config" @@ -84,6 +85,60 @@ func TestHealthz(t *testing.T) { }) } +func TestManagementUsageRequiresManagementAuthAndPopsArray(t *testing.T) { + t.Setenv("MANAGEMENT_PASSWORD", "test-management-key") + + prevQueueEnabled := redisqueue.Enabled() + redisqueue.SetEnabled(false) + t.Cleanup(func() { + redisqueue.SetEnabled(false) + redisqueue.SetEnabled(prevQueueEnabled) + }) + + server := newTestServer(t) + + redisqueue.Enqueue([]byte(`{"id":1}`)) + redisqueue.Enqueue([]byte(`{"id":2}`)) + + missingKeyReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + missingKeyRR := httptest.NewRecorder() + server.engine.ServeHTTP(missingKeyRR, missingKeyReq) + if missingKeyRR.Code != http.StatusUnauthorized { + t.Fatalf("missing key status = %d, want %d body=%s", missingKeyRR.Code, http.StatusUnauthorized, missingKeyRR.Body.String()) + } + + authReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + authReq.Header.Set("Authorization", "Bearer test-management-key") + authRR := httptest.NewRecorder() + server.engine.ServeHTTP(authRR, authReq) + if authRR.Code != http.StatusOK { + t.Fatalf("authenticated status = %d, want %d body=%s", authRR.Code, http.StatusOK, authRR.Body.String()) + } + + var payload []json.RawMessage + if errUnmarshal := json.Unmarshal(authRR.Body.Bytes(), &payload); errUnmarshal != nil { + t.Fatalf("unmarshal response: %v body=%s", errUnmarshal, authRR.Body.String()) + } + if len(payload) != 2 { + t.Fatalf("response records = %d, want 2", len(payload)) + } + for i, raw := range payload { + var record struct { + ID int `json:"id"` + } + if errUnmarshal := json.Unmarshal(raw, &record); errUnmarshal != nil { + t.Fatalf("unmarshal record %d: %v", i, errUnmarshal) + } + if record.ID != i+1 { + t.Fatalf("record %d id = %d, want %d", i, record.ID, i+1) + } + } + + if remaining := redisqueue.PopOldest(1); len(remaining) != 0 { + t.Fatalf("remaining queue = %q, want empty", remaining) + } +} + func TestAmpProviderModelRoutes(t *testing.T) { testCases := []struct { name string From da6c599efd8da34d23d3668371fbb5ac70399e9d Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Tue, 5 May 2026 03:02:25 +0800 Subject: [PATCH 50/51] refactor(management): rename `GetUsage` to `GetUsageQueue` and update routes/tests - Renamed handler and test methods for better clarity on functionality. - Updated route from `/v0/management/usage` to `/v0/management/usage-queue`. - Adjusted integration and unit tests to reflect new naming and routes. --- internal/api/handlers/management/usage.go | 4 ++-- internal/api/handlers/management/usage_test.go | 12 ++++++------ internal/api/server.go | 2 +- internal/api/server_test.go | 12 ++++++++++-- 4 files changed, 19 insertions(+), 11 deletions(-) diff --git a/internal/api/handlers/management/usage.go b/internal/api/handlers/management/usage.go index 8cb175eb..dfddf503 100644 --- a/internal/api/handlers/management/usage.go +++ b/internal/api/handlers/management/usage.go @@ -20,8 +20,8 @@ func (r usageQueueRecord) MarshalJSON() ([]byte, error) { return json.Marshal(string(r)) } -// GetUsage pops queued usage records from the Redis-compatible usage queue. -func (h *Handler) GetUsage(c *gin.Context) { +// GetUsageQueue pops queued usage records from the usage queue. +func (h *Handler) GetUsageQueue(c *gin.Context) { if h == nil { c.JSON(http.StatusInternalServerError, gin.H{"error": "handler unavailable"}) return diff --git a/internal/api/handlers/management/usage_test.go b/internal/api/handlers/management/usage_test.go index 5c5f5c69..ca46d976 100644 --- a/internal/api/handlers/management/usage_test.go +++ b/internal/api/handlers/management/usage_test.go @@ -10,7 +10,7 @@ import ( "github.com/router-for-me/CLIProxyAPI/v6/internal/redisqueue" ) -func TestGetUsagePopsRequestedRecords(t *testing.T) { +func TestGetUsageQueuePopsRequestedRecords(t *testing.T) { gin.SetMode(gin.TestMode) withManagementUsageQueue(t, func() { redisqueue.Enqueue([]byte(`{"id":1}`)) @@ -19,10 +19,10 @@ func TestGetUsagePopsRequestedRecords(t *testing.T) { rec := httptest.NewRecorder() ginCtx, _ := gin.CreateTestContext(rec) - ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage-queue?count=2", nil) h := &Handler{} - h.GetUsage(ginCtx) + h.GetUsageQueue(ginCtx) if rec.Code != http.StatusOK { t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusOK, rec.Body.String()) @@ -45,17 +45,17 @@ func TestGetUsagePopsRequestedRecords(t *testing.T) { }) } -func TestGetUsageInvalidCountDoesNotPop(t *testing.T) { +func TestGetUsageQueueInvalidCountDoesNotPop(t *testing.T) { gin.SetMode(gin.TestMode) withManagementUsageQueue(t, func() { redisqueue.Enqueue([]byte(`{"id":1}`)) rec := httptest.NewRecorder() ginCtx, _ := gin.CreateTestContext(rec) - ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=0", nil) + ginCtx.Request = httptest.NewRequest(http.MethodGet, "/v0/management/usage-queue?count=0", nil) h := &Handler{} - h.GetUsage(ginCtx) + h.GetUsageQueue(ginCtx) if rec.Code != http.StatusBadRequest { t.Fatalf("status = %d, want %d body=%s", rec.Code, http.StatusBadRequest, rec.Body.String()) diff --git a/internal/api/server.go b/internal/api/server.go index 5c43db48..487ea571 100644 --- a/internal/api/server.go +++ b/internal/api/server.go @@ -551,7 +551,7 @@ func (s *Server) registerManagementRoutes() { mgmt.PATCH("/api-keys", s.mgmt.PatchAPIKeys) mgmt.DELETE("/api-keys", s.mgmt.DeleteAPIKeys) mgmt.GET("/api-key-usage", s.mgmt.GetAPIKeyUsage) - mgmt.GET("/usage", s.mgmt.GetUsage) + mgmt.GET("/usage-queue", s.mgmt.GetUsageQueue) mgmt.GET("/gemini-api-key", s.mgmt.GetGeminiKeys) mgmt.PUT("/gemini-api-key", s.mgmt.PutGeminiKeys) diff --git a/internal/api/server_test.go b/internal/api/server_test.go index d5718091..fe37cb72 100644 --- a/internal/api/server_test.go +++ b/internal/api/server_test.go @@ -100,14 +100,22 @@ func TestManagementUsageRequiresManagementAuthAndPopsArray(t *testing.T) { redisqueue.Enqueue([]byte(`{"id":1}`)) redisqueue.Enqueue([]byte(`{"id":2}`)) - missingKeyReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + missingKeyReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage-queue?count=2", nil) missingKeyRR := httptest.NewRecorder() server.engine.ServeHTTP(missingKeyRR, missingKeyReq) if missingKeyRR.Code != http.StatusUnauthorized { t.Fatalf("missing key status = %d, want %d body=%s", missingKeyRR.Code, http.StatusUnauthorized, missingKeyRR.Body.String()) } - authReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + legacyReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage?count=2", nil) + legacyReq.Header.Set("Authorization", "Bearer test-management-key") + legacyRR := httptest.NewRecorder() + server.engine.ServeHTTP(legacyRR, legacyReq) + if legacyRR.Code != http.StatusNotFound { + t.Fatalf("legacy usage status = %d, want %d body=%s", legacyRR.Code, http.StatusNotFound, legacyRR.Body.String()) + } + + authReq := httptest.NewRequest(http.MethodGet, "/v0/management/usage-queue?count=2", nil) authReq.Header.Set("Authorization", "Bearer test-management-key") authRR := httptest.NewRecorder() server.engine.ServeHTTP(authRR, authReq) From ed1458aa6d3430ba59538aeb980b8934f0e80c1f Mon Sep 17 00:00:00 2001 From: Luis Pater Date: Wed, 6 May 2026 00:41:50 +0800 Subject: [PATCH 51/51] chore(docs): update sponsor details in README - Replaced sponsor `z.ai` with `PackyCode` and updated related descriptions, images, and links in `README.md`, `README_CN.md`, and `README_JA.md`. - Removed outdated sponsor entries for `Poixe AI` in all README files. - Added new image assets for PackyCode (`packycode-cn.png` and `packycode-en.png`). --- README.md | 16 ++++------------ README_CN.md | 16 ++++------------ README_JA.md | 16 ++++------------ assets/packycode-cn.png | Bin 0 -> 173559 bytes assets/packycode-en.png | Bin 0 -> 410370 bytes 5 files changed, 12 insertions(+), 36 deletions(-) create mode 100644 assets/packycode-cn.png create mode 100644 assets/packycode-en.png diff --git a/README.md b/README.md index bcadeb87..b1ddb9c0 100644 --- a/README.md +++ b/README.md @@ -10,23 +10,19 @@ So you can use local or multi-account CLI access with OpenAI(include Responses)/ ## Sponsor -[![z.ai](https://assets.router-for.me/english-5-0.jpg)](https://z.ai/subscribe?ic=8JVLJQFSKB) +[![https://www.packyapi.com/register?aff=cliproxyapi](./assets/packycode-en.png)](https://www.packyapi.com/register?aff=cliproxyapi) -This project is sponsored by Z.ai, supporting us with their GLM CODING PLAN. +Thanks to PackyCode for sponsoring this project! -GLM CODING PLAN is a subscription service designed for AI coding, starting at just $10/month. It provides access to their flagship GLM-4.7 & (GLM-5 Only Available for Pro Users)model across 10+ popular AI coding tools (Claude Code, Cline, Roo Code, etc.), offering developers top-tier, fast, and stable coding experiences. +PackyCode is a reliable and efficient API relay service provider, offering relay services for Claude Code, Codex, Gemini, and more. -Get 10% OFF GLM CODING PLAN:https://z.ai/subscribe?ic=8JVLJQFSKB +PackyCode provides special discounts for our software users: register using this link and enter the "cliproxyapi" promo code during recharge to get 10% off. --- - - - - @@ -35,10 +31,6 @@ Get 10% OFF GLM CODING PLAN:https://z.ai/subscribe?ic=8JVLJQFSKB - - - -
PackyCodeThanks to PackyCode for sponsoring this project! PackyCode is a reliable and efficient API relay service provider, offering relay services for Claude Code, Codex, Gemini, and more. PackyCode provides special discounts for our software users: register using this link and enter the "cliproxyapi" promo code during recharge to get 10% off.
AICodeMirror Thanks to AICodeMirror for sponsoring this project! AICodeMirror provides official high-stability relay services for Claude Code / Codex / Gemini CLI, with enterprise-grade concurrency, fast invoicing, and 24/7 dedicated technical support. Claude Code / Codex / Gemini official channels at 38% / 2% / 9% of original price, with extra discounts on top-ups! AICodeMirror offers special benefits for CLIProxyAPI users: register via this link to enjoy 20% off your first top-up, and enterprise customers can get up to 25% off!
Huge thanks to BmoPlus for sponsoring this project! BmoPlus is a highly reliable AI account provider built strictly for heavy AI users and developers. They offer rock-solid, ready-to-use accounts and official top-up services for ChatGPT Plus / ChatGPT Pro (Full Warranty) / Claude Pro / Super Grok / Gemini Pro. By registering and ordering through BmoPlus - Premium AI Accounts & Top-ups, users can unlock the mind-blowing rate of 10% of the official GPT subscription price (90% OFF)!
PoixeAIThanks to Poixe AI for sponsoring this project! Poixe AI provides reliable LLM API services. You can leverage the platform's API endpoints to seamlessly build AI-powered products. Additionally, you can become a vendor by providing AI API resources to the platform and earn revenue. Register through the exclusive CLIProxyAPI referral link and receive a bonus of $5 USD on your first top-up.
VisionCoder Thanks to VisionCoder for supporting this project. VisionCoder Developer Platform is a reliable and efficient API relay service provider, offering access to mainstream AI models such as Claude Code, Codex, and Gemini. It helps developers and teams integrate AI capabilities more easily and improve productivity.

diff --git a/README_CN.md b/README_CN.md index 26602584..e7fa7878 100644 --- a/README_CN.md +++ b/README_CN.md @@ -10,23 +10,19 @@ ## 赞助商 -[![bigmodel.cn](https://assets.router-for.me/chinese-5-0.jpg)](https://www.bigmodel.cn/claude-code?ic=RRVJPB5SII) +[![https://www.packyapi.com/register?aff=cliproxyapi](./assets/packycode-cn.png)](https://www.packyapi.com/register?aff=cliproxyapi) -本项目由 Z智谱 提供赞助, 他们通过 GLM CODING PLAN 对本项目提供技术支持。 +感谢 PackyCode 对本项目的赞助! -GLM CODING PLAN 是专为AI编码打造的订阅套餐,每月最低仅需20元,即可在十余款主流AI编码工具如 Claude Code、Cline、Roo Code 中畅享智谱旗舰模型GLM-4.7(受限于算力,目前仅限Pro用户开放),为开发者提供顶尖的编码体验。 +PackyCode 是一家可靠高效的 API 中转服务商,提供 Claude Code、Codex、Gemini 等多种服务的中转。 -智谱AI为本产品提供了特别优惠,使用以下链接购买可以享受九折优惠:https://www.bigmodel.cn/claude-code?ic=RRVJPB5SII +PackyCode 为本软件用户提供了特别优惠:使用此链接注册,并在充值时输入 "cliproxyapi" 优惠码即可享受九折优惠。 --- - - - - @@ -35,10 +31,6 @@ GLM CODING PLAN 是专为AI编码打造的订阅套餐,每月最低仅需20元 - - - -
PackyCode感谢 PackyCode 对本项目的赞助!PackyCode 是一家可靠高效的 API 中转服务商,提供 Claude Code、Codex、Gemini 等多种服务的中转。PackyCode 为本软件用户提供了特别优惠:使用此链接注册,并在充值时输入 "cliproxyapi" 优惠码即可享受九折优惠。
AICodeMirror 感谢 AICodeMirror 赞助了本项目!AICodeMirror 提供 Claude Code / Codex / Gemini CLI 官方高稳定中转服务,支持企业级高并发、极速开票、7×24 专属技术支持。 Claude Code / Codex / Gemini 官方渠道低至 3.8 / 0.2 / 0.9 折,充值更有折上折!AICodeMirror 为 CLIProxyAPI 的用户提供了特别福利,通过此链接注册的用户,可享受首充8折,企业客户最高可享 7.5 折!
感谢 BmoPlus 赞助了本项目!BmoPlus 是一家专为AI订阅重度用户打造的可靠 AI 账号代充服务商,提供稳定的 ChatGPT Plus / ChatGPT Pro(全程质保) / Claude Pro / Super Grok / Gemini Pro 的官方代充&成品账号。 通过BmoPlus AI成品号专卖/代充注册下单的用户,可享GPT 官网订阅一折 的震撼价格!
PoixeAI感谢 Poixe AI 对本项目的赞助!Poixe AI 提供可靠的 AI 模型接口服务,您可以使用平台提供的 LLM API 接口轻松构建 AI 产品,同时也可以成为供应商,为平台提供大模型资源以赚取收益。通过 CLIProxyAPI 专属链接注册,充值额外赠送 $5 美金
VisionCoder 感谢 VisionCoder 对本项目的支持。VisionCoder 开发平台 是一个可靠高效的 API 中继服务提供商,提供 Claude Code、Codex、Gemini 等主流 AI 模型,帮助开发者和团队更轻松地集成 AI 功能,提升工作效率。

diff --git a/README_JA.md b/README_JA.md index a1eaf1bd..debe4ae5 100644 --- a/README_JA.md +++ b/README_JA.md @@ -10,23 +10,19 @@ OAuth経由でOpenAI Codex(GPTモデル)およびClaude Codeもサポート ## スポンサー -[![z.ai](https://assets.router-for.me/english-5-0.jpg)](https://z.ai/subscribe?ic=8JVLJQFSKB) +[![https://www.packyapi.com/register?aff=cliproxyapi](./assets/packycode-en.png)](https://www.packyapi.com/register?aff=cliproxyapi) -本プロジェクトはZ.aiにスポンサーされており、GLM CODING PLANの提供を受けています。 +PackyCodeのスポンサーシップに感謝します! -GLM CODING PLANはAIコーディング向けに設計されたサブスクリプションサービスで、月額わずか$10から利用可能です。フラッグシップのGLM-4.7および(GLM-5はProユーザーのみ利用可能)モデルを10以上の人気AIコーディングツール(Claude Code、Cline、Roo Codeなど)で利用でき、開発者にトップクラスの高速かつ安定したコーディング体験を提供します。 +PackyCodeは信頼性が高く効率的なAPIリレーサービスプロバイダーで、Claude Code、Codex、Geminiなどのリレーサービスを提供しています。 -GLM CODING PLANを10%割引で取得:https://z.ai/subscribe?ic=8JVLJQFSKB +PackyCodeは当ソフトウェアのユーザーに特別割引を提供しています:こちらのリンクから登録し、チャージ時にプロモーションコード「cliproxyapi」を入力すると10%割引になります。 --- - - - - @@ -35,10 +31,6 @@ GLM CODING PLANを10%割引で取得:https://z.ai/subscribe?ic=8JVLJQFSKB - - - - diff --git a/assets/packycode-cn.png b/assets/packycode-cn.png new file mode 100644 index 0000000000000000000000000000000000000000..3e34d6caed0c2e4b8c8ae19a7f71253c91fd63c3 GIT binary patch literal 173559 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 literal 0 HcmV?d00001 diff --git a/assets/packycode-en.png b/assets/packycode-en.png new file mode 100644 index 0000000000000000000000000000000000000000..90f716e2a443c48fd3f99aaffa9d0fec6103d593 GIT binary patch literal 410370 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%0DP8XYT;x%U5Qvv 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

njg)f! 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 VspTIZo>?P5qNih|U7_U|{Xb&gT_pej literal 0 HcmV?d00001

PackyCodePackyCodeのスポンサーシップに感謝します!PackyCodeは信頼性が高く効率的なAPIリレーサービスプロバイダーで、Claude Code、Codex、Geminiなどのリレーサービスを提供しています。PackyCodeは当ソフトウェアのユーザーに特別割引を提供しています:こちらのリンクから登録し、チャージ時にプロモーションコード「cliproxyapi」を入力すると10%割引になります。
AICodeMirror AICodeMirrorのスポンサーシップに感謝します!AICodeMirrorはClaude Code / Codex / Gemini CLI向けの公式高安定性リレーサービスを提供しており、エンタープライズグレードの同時接続、迅速な請求書発行、24時間365日の専任技術サポートを備えています。Claude Code / Codex / Geminiの公式チャネルが元の価格の38% / 2% / 9%で利用でき、チャージ時にはさらに割引があります!CLIProxyAPIユーザー向けの特別特典:こちらのリンクから登録すると、初回チャージが20%割引になり、エンタープライズのお客様は最大25%割引を受けられます!
本プロジェクトにご支援いただいた BmoPlus に感謝いたします!BmoPlusは、AIサブスクリプションのヘビーユーザー向けに特化した信頼性の高いAIアカウントサービスプロバイダーであり、安定した ChatGPT Plus / ChatGPT Pro (完全保証) / Claude Pro / Super Grok / Gemini Pro の公式代行チャージおよび即納アカウントを提供しています。こちらのBmoPlus AIアカウント専門店/代行チャージ経由でご登録・ご注文いただいたユーザー様は、GPTを 公式サイト価格の約1割(90% OFF) という驚異的な価格でご利用いただけます!
PoixeAIPoixe AIのスポンサーシップに感謝します!Poixe AIは信頼できるAIモデルAPIサービスを提供しており、プラットフォームが提供するLLM APIを使って簡単にAI製品を構築できます。また、サプライヤーとしてプラットフォームに大規模モデルのリソースを提供し、収益を得ることも可能です。CLIProxyAPIの専用リンクから登録すると、チャージ時に追加で$5が付与されます。
VisionCoder VisionCoderのご支援に感謝します!VisionCoder 開発プラットフォーム は、信頼性が高く効率的なAPIリレーサービスプロバイダーで、Claude Code、Codex、Geminiなどの主要AIモデルを提供し、開発者やチームがより簡単にAI機能を統合して生産性を向上できるよう支援します。さらに、VisionCoderはユーザー向けに Token Plan の期間限定キャンペーン(1か月購入で1か月分プレゼント)も提供しています。