8c2f1a80d3
- 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.
90 lines
2.9 KiB
Go
90 lines
2.9 KiB
Go
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",
|
|
"base_url": "https://codex.example.com",
|
|
},
|
|
}); 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",
|
|
"base_url": "https://claude.example.com",
|
|
},
|
|
}); 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"]["https://codex.example.com|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"]["https://claude.example.com|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)
|
|
}
|
|
}
|