package management import ( "encoding/json" "fmt" "strings" "github.com/gin-gonic/gin" "github.com/router-for-me/CLIProxyAPI/v6/internal/config" ) // Generic helpers for list[string] func (h *Handler) putStringList(c *gin.Context, set func([]string), after func()) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []string if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []string `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } set(arr) if after != nil { after() } h.persist(c) } func (h *Handler) patchStringList(c *gin.Context, target *[]string, after func()) { var body struct { Old *string `json:"old"` New *string `json:"new"` Index *int `json:"index"` Value *string `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } if body.Index != nil && body.Value != nil && *body.Index >= 0 && *body.Index < len(*target) { (*target)[*body.Index] = *body.Value if after != nil { after() } h.persist(c) return } if body.Old != nil && body.New != nil { for i := range *target { if (*target)[i] == *body.Old { (*target)[i] = *body.New if after != nil { after() } h.persist(c) return } } *target = append(*target, *body.New) if after != nil { after() } h.persist(c) return } c.JSON(400, gin.H{"error": "missing fields"}) } func (h *Handler) deleteFromStringList(c *gin.Context, target *[]string, after func()) { if idxStr := c.Query("index"); idxStr != "" { var idx int _, err := fmt.Sscanf(idxStr, "%d", &idx) if err == nil && idx >= 0 && idx < len(*target) { *target = append((*target)[:idx], (*target)[idx+1:]...) if after != nil { after() } h.persist(c) return } } if val := strings.TrimSpace(c.Query("value")); val != "" { out := make([]string, 0, len(*target)) for _, v := range *target { if strings.TrimSpace(v) != val { out = append(out, v) } } *target = out if after != nil { after() } h.persist(c) return } c.JSON(400, gin.H{"error": "missing index or value"}) } // api-keys func (h *Handler) GetAPIKeys(c *gin.Context) { c.JSON(200, gin.H{"api-keys": h.cfg.APIKeys}) } func (h *Handler) PutAPIKeys(c *gin.Context) { h.putStringList(c, func(v []string) { h.cfg.APIKeys = append([]string(nil), v...) h.cfg.Access.Providers = nil }, nil) } func (h *Handler) PatchAPIKeys(c *gin.Context) { h.patchStringList(c, &h.cfg.APIKeys, func() { h.cfg.Access.Providers = nil }) } func (h *Handler) DeleteAPIKeys(c *gin.Context) { h.deleteFromStringList(c, &h.cfg.APIKeys, func() { h.cfg.Access.Providers = nil }) } // gemini-api-key: []GeminiKey func (h *Handler) GetGeminiKeys(c *gin.Context) { c.JSON(200, gin.H{"gemini-api-key": h.cfg.GeminiKey}) } func (h *Handler) PutGeminiKeys(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []config.GeminiKey if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []config.GeminiKey `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } h.cfg.GeminiKey = append([]config.GeminiKey(nil), arr...) h.cfg.SanitizeGeminiKeys() h.persist(c) } func (h *Handler) PatchGeminiKey(c *gin.Context) { type geminiKeyPatch struct { APIKey *string `json:"api-key"` Prefix *string `json:"prefix"` BaseURL *string `json:"base-url"` ProxyURL *string `json:"proxy-url"` Headers *map[string]string `json:"headers"` ExcludedModels *[]string `json:"excluded-models"` } var body struct { Index *int `json:"index"` Match *string `json:"match"` Value *geminiKeyPatch `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil || body.Value == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } targetIndex := -1 if body.Index != nil && *body.Index >= 0 && *body.Index < len(h.cfg.GeminiKey) { targetIndex = *body.Index } if targetIndex == -1 && body.Match != nil { match := strings.TrimSpace(*body.Match) if match != "" { for i := range h.cfg.GeminiKey { if h.cfg.GeminiKey[i].APIKey == match { targetIndex = i break } } } } if targetIndex == -1 { c.JSON(404, gin.H{"error": "item not found"}) return } entry := h.cfg.GeminiKey[targetIndex] if body.Value.APIKey != nil { trimmed := strings.TrimSpace(*body.Value.APIKey) if trimmed == "" { h.cfg.GeminiKey = append(h.cfg.GeminiKey[:targetIndex], h.cfg.GeminiKey[targetIndex+1:]...) h.cfg.SanitizeGeminiKeys() h.persist(c) return } entry.APIKey = trimmed } if body.Value.Prefix != nil { entry.Prefix = strings.TrimSpace(*body.Value.Prefix) } if body.Value.BaseURL != nil { entry.BaseURL = strings.TrimSpace(*body.Value.BaseURL) } if body.Value.ProxyURL != nil { entry.ProxyURL = strings.TrimSpace(*body.Value.ProxyURL) } if body.Value.Headers != nil { entry.Headers = config.NormalizeHeaders(*body.Value.Headers) } if body.Value.ExcludedModels != nil { entry.ExcludedModels = config.NormalizeExcludedModels(*body.Value.ExcludedModels) } h.cfg.GeminiKey[targetIndex] = entry h.cfg.SanitizeGeminiKeys() h.persist(c) } func (h *Handler) DeleteGeminiKey(c *gin.Context) { if val := strings.TrimSpace(c.Query("api-key")); val != "" { out := make([]config.GeminiKey, 0, len(h.cfg.GeminiKey)) for _, v := range h.cfg.GeminiKey { if v.APIKey != val { out = append(out, v) } } if len(out) != len(h.cfg.GeminiKey) { h.cfg.GeminiKey = out h.cfg.SanitizeGeminiKeys() h.persist(c) } else { c.JSON(404, gin.H{"error": "item not found"}) } return } if idxStr := c.Query("index"); idxStr != "" { var idx int if _, err := fmt.Sscanf(idxStr, "%d", &idx); err == nil && idx >= 0 && idx < len(h.cfg.GeminiKey) { h.cfg.GeminiKey = append(h.cfg.GeminiKey[:idx], h.cfg.GeminiKey[idx+1:]...) h.cfg.SanitizeGeminiKeys() h.persist(c) return } } c.JSON(400, gin.H{"error": "missing api-key or index"}) } // claude-api-key: []ClaudeKey func (h *Handler) GetClaudeKeys(c *gin.Context) { c.JSON(200, gin.H{"claude-api-key": h.cfg.ClaudeKey}) } func (h *Handler) PutClaudeKeys(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []config.ClaudeKey if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []config.ClaudeKey `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } for i := range arr { normalizeClaudeKey(&arr[i]) } h.cfg.ClaudeKey = arr h.cfg.SanitizeClaudeKeys() h.persist(c) } func (h *Handler) PatchClaudeKey(c *gin.Context) { type claudeKeyPatch struct { APIKey *string `json:"api-key"` Prefix *string `json:"prefix"` BaseURL *string `json:"base-url"` ProxyURL *string `json:"proxy-url"` Models *[]config.ClaudeModel `json:"models"` Headers *map[string]string `json:"headers"` ExcludedModels *[]string `json:"excluded-models"` } var body struct { Index *int `json:"index"` Match *string `json:"match"` Value *claudeKeyPatch `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil || body.Value == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } targetIndex := -1 if body.Index != nil && *body.Index >= 0 && *body.Index < len(h.cfg.ClaudeKey) { targetIndex = *body.Index } if targetIndex == -1 && body.Match != nil { match := strings.TrimSpace(*body.Match) for i := range h.cfg.ClaudeKey { if h.cfg.ClaudeKey[i].APIKey == match { targetIndex = i break } } } if targetIndex == -1 { c.JSON(404, gin.H{"error": "item not found"}) return } entry := h.cfg.ClaudeKey[targetIndex] if body.Value.APIKey != nil { entry.APIKey = strings.TrimSpace(*body.Value.APIKey) } if body.Value.Prefix != nil { entry.Prefix = strings.TrimSpace(*body.Value.Prefix) } if body.Value.BaseURL != nil { entry.BaseURL = strings.TrimSpace(*body.Value.BaseURL) } if body.Value.ProxyURL != nil { entry.ProxyURL = strings.TrimSpace(*body.Value.ProxyURL) } if body.Value.Models != nil { entry.Models = append([]config.ClaudeModel(nil), (*body.Value.Models)...) } if body.Value.Headers != nil { entry.Headers = config.NormalizeHeaders(*body.Value.Headers) } if body.Value.ExcludedModels != nil { entry.ExcludedModels = config.NormalizeExcludedModels(*body.Value.ExcludedModels) } normalizeClaudeKey(&entry) h.cfg.ClaudeKey[targetIndex] = entry h.cfg.SanitizeClaudeKeys() h.persist(c) } func (h *Handler) DeleteClaudeKey(c *gin.Context) { if val := c.Query("api-key"); val != "" { out := make([]config.ClaudeKey, 0, len(h.cfg.ClaudeKey)) for _, v := range h.cfg.ClaudeKey { if v.APIKey != val { out = append(out, v) } } h.cfg.ClaudeKey = out h.cfg.SanitizeClaudeKeys() h.persist(c) return } if idxStr := c.Query("index"); idxStr != "" { var idx int _, err := fmt.Sscanf(idxStr, "%d", &idx) if err == nil && idx >= 0 && idx < len(h.cfg.ClaudeKey) { h.cfg.ClaudeKey = append(h.cfg.ClaudeKey[:idx], h.cfg.ClaudeKey[idx+1:]...) h.cfg.SanitizeClaudeKeys() h.persist(c) return } } c.JSON(400, gin.H{"error": "missing api-key or index"}) } // openai-compatibility: []OpenAICompatibility func (h *Handler) GetOpenAICompat(c *gin.Context) { c.JSON(200, gin.H{"openai-compatibility": normalizedOpenAICompatibilityEntries(h.cfg.OpenAICompatibility)}) } func (h *Handler) PutOpenAICompat(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []config.OpenAICompatibility if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []config.OpenAICompatibility `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } filtered := make([]config.OpenAICompatibility, 0, len(arr)) for i := range arr { normalizeOpenAICompatibilityEntry(&arr[i]) if strings.TrimSpace(arr[i].BaseURL) != "" { filtered = append(filtered, arr[i]) } } h.cfg.OpenAICompatibility = filtered h.cfg.SanitizeOpenAICompatibility() h.persist(c) } func (h *Handler) PatchOpenAICompat(c *gin.Context) { type openAICompatPatch struct { Name *string `json:"name"` Prefix *string `json:"prefix"` BaseURL *string `json:"base-url"` APIKeyEntries *[]config.OpenAICompatibilityAPIKey `json:"api-key-entries"` Models *[]config.OpenAICompatibilityModel `json:"models"` Headers *map[string]string `json:"headers"` } var body struct { Name *string `json:"name"` Index *int `json:"index"` Value *openAICompatPatch `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil || body.Value == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } targetIndex := -1 if body.Index != nil && *body.Index >= 0 && *body.Index < len(h.cfg.OpenAICompatibility) { targetIndex = *body.Index } if targetIndex == -1 && body.Name != nil { match := strings.TrimSpace(*body.Name) for i := range h.cfg.OpenAICompatibility { if h.cfg.OpenAICompatibility[i].Name == match { targetIndex = i break } } } if targetIndex == -1 { c.JSON(404, gin.H{"error": "item not found"}) return } entry := h.cfg.OpenAICompatibility[targetIndex] if body.Value.Name != nil { entry.Name = strings.TrimSpace(*body.Value.Name) } if body.Value.Prefix != nil { entry.Prefix = strings.TrimSpace(*body.Value.Prefix) } if body.Value.BaseURL != nil { trimmed := strings.TrimSpace(*body.Value.BaseURL) if trimmed == "" { h.cfg.OpenAICompatibility = append(h.cfg.OpenAICompatibility[:targetIndex], h.cfg.OpenAICompatibility[targetIndex+1:]...) h.cfg.SanitizeOpenAICompatibility() h.persist(c) return } entry.BaseURL = trimmed } if body.Value.APIKeyEntries != nil { entry.APIKeyEntries = append([]config.OpenAICompatibilityAPIKey(nil), (*body.Value.APIKeyEntries)...) } if body.Value.Models != nil { entry.Models = append([]config.OpenAICompatibilityModel(nil), (*body.Value.Models)...) } if body.Value.Headers != nil { entry.Headers = config.NormalizeHeaders(*body.Value.Headers) } normalizeOpenAICompatibilityEntry(&entry) h.cfg.OpenAICompatibility[targetIndex] = entry h.cfg.SanitizeOpenAICompatibility() h.persist(c) } func (h *Handler) DeleteOpenAICompat(c *gin.Context) { if name := c.Query("name"); name != "" { out := make([]config.OpenAICompatibility, 0, len(h.cfg.OpenAICompatibility)) for _, v := range h.cfg.OpenAICompatibility { if v.Name != name { out = append(out, v) } } h.cfg.OpenAICompatibility = out h.cfg.SanitizeOpenAICompatibility() h.persist(c) return } if idxStr := c.Query("index"); idxStr != "" { var idx int _, err := fmt.Sscanf(idxStr, "%d", &idx) if err == nil && idx >= 0 && idx < len(h.cfg.OpenAICompatibility) { h.cfg.OpenAICompatibility = append(h.cfg.OpenAICompatibility[:idx], h.cfg.OpenAICompatibility[idx+1:]...) h.cfg.SanitizeOpenAICompatibility() h.persist(c) return } } c.JSON(400, gin.H{"error": "missing name or index"}) } // vertex-api-key: []VertexCompatKey func (h *Handler) GetVertexCompatKeys(c *gin.Context) { c.JSON(200, gin.H{"vertex-api-key": h.cfg.VertexCompatAPIKey}) } func (h *Handler) PutVertexCompatKeys(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []config.VertexCompatKey if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []config.VertexCompatKey `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } for i := range arr { normalizeVertexCompatKey(&arr[i]) } h.cfg.VertexCompatAPIKey = arr h.cfg.SanitizeVertexCompatKeys() h.persist(c) } func (h *Handler) PatchVertexCompatKey(c *gin.Context) { type vertexCompatPatch struct { APIKey *string `json:"api-key"` Prefix *string `json:"prefix"` BaseURL *string `json:"base-url"` ProxyURL *string `json:"proxy-url"` Headers *map[string]string `json:"headers"` Models *[]config.VertexCompatModel `json:"models"` } var body struct { Index *int `json:"index"` Match *string `json:"match"` Value *vertexCompatPatch `json:"value"` } if errBindJSON := c.ShouldBindJSON(&body); errBindJSON != nil || body.Value == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } targetIndex := -1 if body.Index != nil && *body.Index >= 0 && *body.Index < len(h.cfg.VertexCompatAPIKey) { targetIndex = *body.Index } if targetIndex == -1 && body.Match != nil { match := strings.TrimSpace(*body.Match) if match != "" { for i := range h.cfg.VertexCompatAPIKey { if h.cfg.VertexCompatAPIKey[i].APIKey == match { targetIndex = i break } } } } if targetIndex == -1 { c.JSON(404, gin.H{"error": "item not found"}) return } entry := h.cfg.VertexCompatAPIKey[targetIndex] if body.Value.APIKey != nil { trimmed := strings.TrimSpace(*body.Value.APIKey) if trimmed == "" { h.cfg.VertexCompatAPIKey = append(h.cfg.VertexCompatAPIKey[:targetIndex], h.cfg.VertexCompatAPIKey[targetIndex+1:]...) h.cfg.SanitizeVertexCompatKeys() h.persist(c) return } entry.APIKey = trimmed } if body.Value.Prefix != nil { entry.Prefix = strings.TrimSpace(*body.Value.Prefix) } if body.Value.BaseURL != nil { trimmed := strings.TrimSpace(*body.Value.BaseURL) if trimmed == "" { h.cfg.VertexCompatAPIKey = append(h.cfg.VertexCompatAPIKey[:targetIndex], h.cfg.VertexCompatAPIKey[targetIndex+1:]...) h.cfg.SanitizeVertexCompatKeys() h.persist(c) return } entry.BaseURL = trimmed } if body.Value.ProxyURL != nil { entry.ProxyURL = strings.TrimSpace(*body.Value.ProxyURL) } if body.Value.Headers != nil { entry.Headers = config.NormalizeHeaders(*body.Value.Headers) } if body.Value.Models != nil { entry.Models = append([]config.VertexCompatModel(nil), (*body.Value.Models)...) } normalizeVertexCompatKey(&entry) h.cfg.VertexCompatAPIKey[targetIndex] = entry h.cfg.SanitizeVertexCompatKeys() h.persist(c) } func (h *Handler) DeleteVertexCompatKey(c *gin.Context) { if val := strings.TrimSpace(c.Query("api-key")); val != "" { out := make([]config.VertexCompatKey, 0, len(h.cfg.VertexCompatAPIKey)) for _, v := range h.cfg.VertexCompatAPIKey { if v.APIKey != val { out = append(out, v) } } h.cfg.VertexCompatAPIKey = out h.cfg.SanitizeVertexCompatKeys() h.persist(c) return } if idxStr := c.Query("index"); idxStr != "" { var idx int _, errScan := fmt.Sscanf(idxStr, "%d", &idx) if errScan == nil && idx >= 0 && idx < len(h.cfg.VertexCompatAPIKey) { h.cfg.VertexCompatAPIKey = append(h.cfg.VertexCompatAPIKey[:idx], h.cfg.VertexCompatAPIKey[idx+1:]...) h.cfg.SanitizeVertexCompatKeys() h.persist(c) return } } c.JSON(400, gin.H{"error": "missing api-key or index"}) } // oauth-excluded-models: map[string][]string func (h *Handler) GetOAuthExcludedModels(c *gin.Context) { c.JSON(200, gin.H{"oauth-excluded-models": config.NormalizeOAuthExcludedModels(h.cfg.OAuthExcludedModels)}) } func (h *Handler) PutOAuthExcludedModels(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var entries map[string][]string if err = json.Unmarshal(data, &entries); err != nil { var wrapper struct { Items map[string][]string `json:"items"` } if err2 := json.Unmarshal(data, &wrapper); err2 != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } entries = wrapper.Items } h.cfg.OAuthExcludedModels = config.NormalizeOAuthExcludedModels(entries) h.persist(c) } func (h *Handler) PatchOAuthExcludedModels(c *gin.Context) { var body struct { Provider *string `json:"provider"` Models []string `json:"models"` } if err := c.ShouldBindJSON(&body); err != nil || body.Provider == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } provider := strings.ToLower(strings.TrimSpace(*body.Provider)) if provider == "" { c.JSON(400, gin.H{"error": "invalid provider"}) return } normalized := config.NormalizeExcludedModels(body.Models) if len(normalized) == 0 { if h.cfg.OAuthExcludedModels == nil { c.JSON(404, gin.H{"error": "provider not found"}) return } if _, ok := h.cfg.OAuthExcludedModels[provider]; !ok { c.JSON(404, gin.H{"error": "provider not found"}) return } delete(h.cfg.OAuthExcludedModels, provider) if len(h.cfg.OAuthExcludedModels) == 0 { h.cfg.OAuthExcludedModels = nil } h.persist(c) return } if h.cfg.OAuthExcludedModels == nil { h.cfg.OAuthExcludedModels = make(map[string][]string) } h.cfg.OAuthExcludedModels[provider] = normalized h.persist(c) } func (h *Handler) DeleteOAuthExcludedModels(c *gin.Context) { provider := strings.ToLower(strings.TrimSpace(c.Query("provider"))) if provider == "" { c.JSON(400, gin.H{"error": "missing provider"}) return } if h.cfg.OAuthExcludedModels == nil { c.JSON(404, gin.H{"error": "provider not found"}) return } if _, ok := h.cfg.OAuthExcludedModels[provider]; !ok { c.JSON(404, gin.H{"error": "provider not found"}) return } delete(h.cfg.OAuthExcludedModels, provider) if len(h.cfg.OAuthExcludedModels) == 0 { h.cfg.OAuthExcludedModels = nil } h.persist(c) } // oauth-model-alias: map[string][]OAuthModelAlias func (h *Handler) GetOAuthModelAlias(c *gin.Context) { c.JSON(200, gin.H{"oauth-model-alias": sanitizedOAuthModelAlias(h.cfg.OAuthModelAlias)}) } func (h *Handler) PutOAuthModelAlias(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var entries map[string][]config.OAuthModelAlias if err = json.Unmarshal(data, &entries); err != nil { var wrapper struct { Items map[string][]config.OAuthModelAlias `json:"items"` } if err2 := json.Unmarshal(data, &wrapper); err2 != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } entries = wrapper.Items } h.cfg.OAuthModelAlias = sanitizedOAuthModelAlias(entries) h.persist(c) } func (h *Handler) PatchOAuthModelAlias(c *gin.Context) { var body struct { Provider *string `json:"provider"` Channel *string `json:"channel"` Aliases []config.OAuthModelAlias `json:"aliases"` } if errBindJSON := c.ShouldBindJSON(&body); errBindJSON != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } channelRaw := "" if body.Channel != nil { channelRaw = *body.Channel } else if body.Provider != nil { channelRaw = *body.Provider } channel := strings.ToLower(strings.TrimSpace(channelRaw)) if channel == "" { c.JSON(400, gin.H{"error": "invalid channel"}) return } normalizedMap := sanitizedOAuthModelAlias(map[string][]config.OAuthModelAlias{channel: body.Aliases}) normalized := normalizedMap[channel] if len(normalized) == 0 { if h.cfg.OAuthModelAlias == nil { c.JSON(404, gin.H{"error": "channel not found"}) return } if _, ok := h.cfg.OAuthModelAlias[channel]; !ok { c.JSON(404, gin.H{"error": "channel not found"}) return } delete(h.cfg.OAuthModelAlias, channel) if len(h.cfg.OAuthModelAlias) == 0 { h.cfg.OAuthModelAlias = nil } h.persist(c) return } if h.cfg.OAuthModelAlias == nil { h.cfg.OAuthModelAlias = make(map[string][]config.OAuthModelAlias) } h.cfg.OAuthModelAlias[channel] = normalized h.persist(c) } func (h *Handler) DeleteOAuthModelAlias(c *gin.Context) { channel := strings.ToLower(strings.TrimSpace(c.Query("channel"))) if channel == "" { channel = strings.ToLower(strings.TrimSpace(c.Query("provider"))) } if channel == "" { c.JSON(400, gin.H{"error": "missing channel"}) return } if h.cfg.OAuthModelAlias == nil { c.JSON(404, gin.H{"error": "channel not found"}) return } if _, ok := h.cfg.OAuthModelAlias[channel]; !ok { c.JSON(404, gin.H{"error": "channel not found"}) return } delete(h.cfg.OAuthModelAlias, channel) if len(h.cfg.OAuthModelAlias) == 0 { h.cfg.OAuthModelAlias = nil } h.persist(c) } // codex-api-key: []CodexKey func (h *Handler) GetCodexKeys(c *gin.Context) { c.JSON(200, gin.H{"codex-api-key": h.cfg.CodexKey}) } func (h *Handler) PutCodexKeys(c *gin.Context) { data, err := c.GetRawData() if err != nil { c.JSON(400, gin.H{"error": "failed to read body"}) return } var arr []config.CodexKey if err = json.Unmarshal(data, &arr); err != nil { var obj struct { Items []config.CodexKey `json:"items"` } if err2 := json.Unmarshal(data, &obj); err2 != nil || len(obj.Items) == 0 { c.JSON(400, gin.H{"error": "invalid body"}) return } arr = obj.Items } // Filter out codex entries with empty base-url (treat as removed) filtered := make([]config.CodexKey, 0, len(arr)) for i := range arr { entry := arr[i] normalizeCodexKey(&entry) if entry.BaseURL == "" { continue } filtered = append(filtered, entry) } h.cfg.CodexKey = filtered h.cfg.SanitizeCodexKeys() h.persist(c) } func (h *Handler) PatchCodexKey(c *gin.Context) { type codexKeyPatch struct { APIKey *string `json:"api-key"` Prefix *string `json:"prefix"` BaseURL *string `json:"base-url"` ProxyURL *string `json:"proxy-url"` Models *[]config.CodexModel `json:"models"` Headers *map[string]string `json:"headers"` ExcludedModels *[]string `json:"excluded-models"` } var body struct { Index *int `json:"index"` Match *string `json:"match"` Value *codexKeyPatch `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil || body.Value == nil { c.JSON(400, gin.H{"error": "invalid body"}) return } targetIndex := -1 if body.Index != nil && *body.Index >= 0 && *body.Index < len(h.cfg.CodexKey) { targetIndex = *body.Index } if targetIndex == -1 && body.Match != nil { match := strings.TrimSpace(*body.Match) for i := range h.cfg.CodexKey { if h.cfg.CodexKey[i].APIKey == match { targetIndex = i break } } } if targetIndex == -1 { c.JSON(404, gin.H{"error": "item not found"}) return } entry := h.cfg.CodexKey[targetIndex] if body.Value.APIKey != nil { entry.APIKey = strings.TrimSpace(*body.Value.APIKey) } if body.Value.Prefix != nil { entry.Prefix = strings.TrimSpace(*body.Value.Prefix) } if body.Value.BaseURL != nil { trimmed := strings.TrimSpace(*body.Value.BaseURL) if trimmed == "" { h.cfg.CodexKey = append(h.cfg.CodexKey[:targetIndex], h.cfg.CodexKey[targetIndex+1:]...) h.cfg.SanitizeCodexKeys() h.persist(c) return } entry.BaseURL = trimmed } if body.Value.ProxyURL != nil { entry.ProxyURL = strings.TrimSpace(*body.Value.ProxyURL) } if body.Value.Models != nil { entry.Models = append([]config.CodexModel(nil), (*body.Value.Models)...) } if body.Value.Headers != nil { entry.Headers = config.NormalizeHeaders(*body.Value.Headers) } if body.Value.ExcludedModels != nil { entry.ExcludedModels = config.NormalizeExcludedModels(*body.Value.ExcludedModels) } normalizeCodexKey(&entry) h.cfg.CodexKey[targetIndex] = entry h.cfg.SanitizeCodexKeys() h.persist(c) } func (h *Handler) DeleteCodexKey(c *gin.Context) { if val := c.Query("api-key"); val != "" { out := make([]config.CodexKey, 0, len(h.cfg.CodexKey)) for _, v := range h.cfg.CodexKey { if v.APIKey != val { out = append(out, v) } } h.cfg.CodexKey = out h.cfg.SanitizeCodexKeys() h.persist(c) return } if idxStr := c.Query("index"); idxStr != "" { var idx int _, err := fmt.Sscanf(idxStr, "%d", &idx) if err == nil && idx >= 0 && idx < len(h.cfg.CodexKey) { h.cfg.CodexKey = append(h.cfg.CodexKey[:idx], h.cfg.CodexKey[idx+1:]...) h.cfg.SanitizeCodexKeys() h.persist(c) return } } c.JSON(400, gin.H{"error": "missing api-key or index"}) } func normalizeOpenAICompatibilityEntry(entry *config.OpenAICompatibility) { if entry == nil { return } // Trim base-url; empty base-url indicates provider should be removed by sanitization entry.BaseURL = strings.TrimSpace(entry.BaseURL) entry.Headers = config.NormalizeHeaders(entry.Headers) existing := make(map[string]struct{}, len(entry.APIKeyEntries)) for i := range entry.APIKeyEntries { trimmed := strings.TrimSpace(entry.APIKeyEntries[i].APIKey) entry.APIKeyEntries[i].APIKey = trimmed if trimmed != "" { existing[trimmed] = struct{}{} } } } func normalizedOpenAICompatibilityEntries(entries []config.OpenAICompatibility) []config.OpenAICompatibility { if len(entries) == 0 { return nil } out := make([]config.OpenAICompatibility, len(entries)) for i := range entries { copyEntry := entries[i] if len(copyEntry.APIKeyEntries) > 0 { copyEntry.APIKeyEntries = append([]config.OpenAICompatibilityAPIKey(nil), copyEntry.APIKeyEntries...) } normalizeOpenAICompatibilityEntry(©Entry) out[i] = copyEntry } return out } func normalizeClaudeKey(entry *config.ClaudeKey) { if entry == nil { return } entry.APIKey = strings.TrimSpace(entry.APIKey) entry.BaseURL = strings.TrimSpace(entry.BaseURL) entry.ProxyURL = strings.TrimSpace(entry.ProxyURL) entry.Headers = config.NormalizeHeaders(entry.Headers) entry.ExcludedModels = config.NormalizeExcludedModels(entry.ExcludedModels) if len(entry.Models) == 0 { return } normalized := make([]config.ClaudeModel, 0, len(entry.Models)) for i := range entry.Models { model := entry.Models[i] model.Name = strings.TrimSpace(model.Name) model.Alias = strings.TrimSpace(model.Alias) if model.Name == "" && model.Alias == "" { continue } normalized = append(normalized, model) } entry.Models = normalized } func normalizeCodexKey(entry *config.CodexKey) { if entry == nil { return } entry.APIKey = strings.TrimSpace(entry.APIKey) entry.Prefix = strings.TrimSpace(entry.Prefix) entry.BaseURL = strings.TrimSpace(entry.BaseURL) entry.ProxyURL = strings.TrimSpace(entry.ProxyURL) entry.Headers = config.NormalizeHeaders(entry.Headers) entry.ExcludedModels = config.NormalizeExcludedModels(entry.ExcludedModels) if len(entry.Models) == 0 { return } normalized := make([]config.CodexModel, 0, len(entry.Models)) for i := range entry.Models { model := entry.Models[i] model.Name = strings.TrimSpace(model.Name) model.Alias = strings.TrimSpace(model.Alias) if model.Name == "" && model.Alias == "" { continue } normalized = append(normalized, model) } entry.Models = normalized } func normalizeVertexCompatKey(entry *config.VertexCompatKey) { if entry == nil { return } entry.APIKey = strings.TrimSpace(entry.APIKey) entry.Prefix = strings.TrimSpace(entry.Prefix) entry.BaseURL = strings.TrimSpace(entry.BaseURL) entry.ProxyURL = strings.TrimSpace(entry.ProxyURL) entry.Headers = config.NormalizeHeaders(entry.Headers) if len(entry.Models) == 0 { return } normalized := make([]config.VertexCompatModel, 0, len(entry.Models)) for i := range entry.Models { model := entry.Models[i] model.Name = strings.TrimSpace(model.Name) model.Alias = strings.TrimSpace(model.Alias) if model.Name == "" || model.Alias == "" { continue } normalized = append(normalized, model) } entry.Models = normalized } func sanitizedOAuthModelAlias(entries map[string][]config.OAuthModelAlias) map[string][]config.OAuthModelAlias { if len(entries) == 0 { return nil } copied := make(map[string][]config.OAuthModelAlias, len(entries)) for channel, aliases := range entries { if len(aliases) == 0 { continue } copied[channel] = append([]config.OAuthModelAlias(nil), aliases...) } if len(copied) == 0 { return nil } cfg := config.Config{OAuthModelAlias: copied} cfg.SanitizeOAuthModelAlias() if len(cfg.OAuthModelAlias) == 0 { return nil } return cfg.OAuthModelAlias } // GetAmpCode returns the complete ampcode configuration. func (h *Handler) GetAmpCode(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"ampcode": config.AmpCode{}}) return } c.JSON(200, gin.H{"ampcode": h.cfg.AmpCode}) } // GetAmpUpstreamURL returns the ampcode upstream URL. func (h *Handler) GetAmpUpstreamURL(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"upstream-url": ""}) return } c.JSON(200, gin.H{"upstream-url": h.cfg.AmpCode.UpstreamURL}) } // PutAmpUpstreamURL updates the ampcode upstream URL. func (h *Handler) PutAmpUpstreamURL(c *gin.Context) { h.updateStringField(c, func(v string) { h.cfg.AmpCode.UpstreamURL = strings.TrimSpace(v) }) } // DeleteAmpUpstreamURL clears the ampcode upstream URL. func (h *Handler) DeleteAmpUpstreamURL(c *gin.Context) { h.cfg.AmpCode.UpstreamURL = "" h.persist(c) } // GetAmpUpstreamAPIKey returns the ampcode upstream API key. func (h *Handler) GetAmpUpstreamAPIKey(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"upstream-api-key": ""}) return } c.JSON(200, gin.H{"upstream-api-key": h.cfg.AmpCode.UpstreamAPIKey}) } // PutAmpUpstreamAPIKey updates the ampcode upstream API key. func (h *Handler) PutAmpUpstreamAPIKey(c *gin.Context) { h.updateStringField(c, func(v string) { h.cfg.AmpCode.UpstreamAPIKey = strings.TrimSpace(v) }) } // DeleteAmpUpstreamAPIKey clears the ampcode upstream API key. func (h *Handler) DeleteAmpUpstreamAPIKey(c *gin.Context) { h.cfg.AmpCode.UpstreamAPIKey = "" h.persist(c) } // GetAmpRestrictManagementToLocalhost returns the localhost restriction setting. func (h *Handler) GetAmpRestrictManagementToLocalhost(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"restrict-management-to-localhost": true}) return } c.JSON(200, gin.H{"restrict-management-to-localhost": h.cfg.AmpCode.RestrictManagementToLocalhost}) } // PutAmpRestrictManagementToLocalhost updates the localhost restriction setting. func (h *Handler) PutAmpRestrictManagementToLocalhost(c *gin.Context) { h.updateBoolField(c, func(v bool) { h.cfg.AmpCode.RestrictManagementToLocalhost = v }) } // GetAmpModelMappings returns the ampcode model mappings. func (h *Handler) GetAmpModelMappings(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"model-mappings": []config.AmpModelMapping{}}) return } c.JSON(200, gin.H{"model-mappings": h.cfg.AmpCode.ModelMappings}) } // PutAmpModelMappings replaces all ampcode model mappings. func (h *Handler) PutAmpModelMappings(c *gin.Context) { var body struct { Value []config.AmpModelMapping `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } h.cfg.AmpCode.ModelMappings = body.Value h.persist(c) } // PatchAmpModelMappings adds or updates model mappings. func (h *Handler) PatchAmpModelMappings(c *gin.Context) { var body struct { Value []config.AmpModelMapping `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } existing := make(map[string]int) for i, m := range h.cfg.AmpCode.ModelMappings { existing[strings.TrimSpace(m.From)] = i } for _, newMapping := range body.Value { from := strings.TrimSpace(newMapping.From) if idx, ok := existing[from]; ok { h.cfg.AmpCode.ModelMappings[idx] = newMapping } else { h.cfg.AmpCode.ModelMappings = append(h.cfg.AmpCode.ModelMappings, newMapping) existing[from] = len(h.cfg.AmpCode.ModelMappings) - 1 } } h.persist(c) } // DeleteAmpModelMappings removes specified model mappings by "from" field. func (h *Handler) DeleteAmpModelMappings(c *gin.Context) { var body struct { Value []string `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil || len(body.Value) == 0 { h.cfg.AmpCode.ModelMappings = nil h.persist(c) return } toRemove := make(map[string]bool) for _, from := range body.Value { toRemove[strings.TrimSpace(from)] = true } newMappings := make([]config.AmpModelMapping, 0, len(h.cfg.AmpCode.ModelMappings)) for _, m := range h.cfg.AmpCode.ModelMappings { if !toRemove[strings.TrimSpace(m.From)] { newMappings = append(newMappings, m) } } h.cfg.AmpCode.ModelMappings = newMappings h.persist(c) } // GetAmpForceModelMappings returns whether model mappings are forced. func (h *Handler) GetAmpForceModelMappings(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"force-model-mappings": false}) return } c.JSON(200, gin.H{"force-model-mappings": h.cfg.AmpCode.ForceModelMappings}) } // PutAmpForceModelMappings updates the force model mappings setting. func (h *Handler) PutAmpForceModelMappings(c *gin.Context) { h.updateBoolField(c, func(v bool) { h.cfg.AmpCode.ForceModelMappings = v }) } // GetAmpUpstreamAPIKeys returns the ampcode upstream API keys mapping. func (h *Handler) GetAmpUpstreamAPIKeys(c *gin.Context) { if h == nil || h.cfg == nil { c.JSON(200, gin.H{"upstream-api-keys": []config.AmpUpstreamAPIKeyEntry{}}) return } c.JSON(200, gin.H{"upstream-api-keys": h.cfg.AmpCode.UpstreamAPIKeys}) } // PutAmpUpstreamAPIKeys replaces all ampcode upstream API keys mappings. func (h *Handler) PutAmpUpstreamAPIKeys(c *gin.Context) { var body struct { Value []config.AmpUpstreamAPIKeyEntry `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } // Normalize entries: trim whitespace, filter empty normalized := normalizeAmpUpstreamAPIKeyEntries(body.Value) h.cfg.AmpCode.UpstreamAPIKeys = normalized h.persist(c) } // PatchAmpUpstreamAPIKeys adds or updates upstream API keys entries. // Matching is done by upstream-api-key value. func (h *Handler) PatchAmpUpstreamAPIKeys(c *gin.Context) { var body struct { Value []config.AmpUpstreamAPIKeyEntry `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } existing := make(map[string]int) for i, entry := range h.cfg.AmpCode.UpstreamAPIKeys { existing[strings.TrimSpace(entry.UpstreamAPIKey)] = i } for _, newEntry := range body.Value { upstreamKey := strings.TrimSpace(newEntry.UpstreamAPIKey) if upstreamKey == "" { continue } normalizedEntry := config.AmpUpstreamAPIKeyEntry{ UpstreamAPIKey: upstreamKey, APIKeys: normalizeAPIKeysList(newEntry.APIKeys), } if idx, ok := existing[upstreamKey]; ok { h.cfg.AmpCode.UpstreamAPIKeys[idx] = normalizedEntry } else { h.cfg.AmpCode.UpstreamAPIKeys = append(h.cfg.AmpCode.UpstreamAPIKeys, normalizedEntry) existing[upstreamKey] = len(h.cfg.AmpCode.UpstreamAPIKeys) - 1 } } h.persist(c) } // DeleteAmpUpstreamAPIKeys removes specified upstream API keys entries. // Body must be JSON: {"value": ["", ...]}. // If "value" is an empty array, clears all entries. // If JSON is invalid or "value" is missing/null, returns 400 and does not persist any change. func (h *Handler) DeleteAmpUpstreamAPIKeys(c *gin.Context) { var body struct { Value []string `json:"value"` } if err := c.ShouldBindJSON(&body); err != nil { c.JSON(400, gin.H{"error": "invalid body"}) return } if body.Value == nil { c.JSON(400, gin.H{"error": "missing value"}) return } // Empty array means clear all if len(body.Value) == 0 { h.cfg.AmpCode.UpstreamAPIKeys = nil h.persist(c) return } toRemove := make(map[string]bool) for _, key := range body.Value { trimmed := strings.TrimSpace(key) if trimmed == "" { continue } toRemove[trimmed] = true } if len(toRemove) == 0 { c.JSON(400, gin.H{"error": "empty value"}) return } newEntries := make([]config.AmpUpstreamAPIKeyEntry, 0, len(h.cfg.AmpCode.UpstreamAPIKeys)) for _, entry := range h.cfg.AmpCode.UpstreamAPIKeys { if !toRemove[strings.TrimSpace(entry.UpstreamAPIKey)] { newEntries = append(newEntries, entry) } } h.cfg.AmpCode.UpstreamAPIKeys = newEntries h.persist(c) } // normalizeAmpUpstreamAPIKeyEntries normalizes a list of upstream API key entries. func normalizeAmpUpstreamAPIKeyEntries(entries []config.AmpUpstreamAPIKeyEntry) []config.AmpUpstreamAPIKeyEntry { if len(entries) == 0 { return nil } out := make([]config.AmpUpstreamAPIKeyEntry, 0, len(entries)) for _, entry := range entries { upstreamKey := strings.TrimSpace(entry.UpstreamAPIKey) if upstreamKey == "" { continue } apiKeys := normalizeAPIKeysList(entry.APIKeys) out = append(out, config.AmpUpstreamAPIKeyEntry{ UpstreamAPIKey: upstreamKey, APIKeys: apiKeys, }) } if len(out) == 0 { return nil } return out } // normalizeAPIKeysList trims and filters empty strings from a list of API keys. func normalizeAPIKeysList(keys []string) []string { if len(keys) == 0 { return nil } out := make([]string, 0, len(keys)) for _, k := range keys { trimmed := strings.TrimSpace(k) if trimmed != "" { out = append(out, trimmed) } } if len(out) == 0 { return nil } return out }