mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 21:10:51 +08:00
AMP CLI sends Gemini requests to non-standard paths that were being directly proxied to ampcode.com without checking for local OAuth. This fix adds: - GeminiBridge handler to transform AMP CLI paths to standard format - Enhanced model extraction from AMP's /publishers/google/models/* paths - FallbackHandler wrapper to check for local OAuth before proxying Flow: - If user has local Google OAuth → use it (free tier) - If no local OAuth → fallback to ampcode.com (charges credits) Fixes issue where gemini-3-pro-preview requests always charged AMP credits even when user had valid Google Cloud OAuth configured.
130 lines
4.2 KiB
Go
130 lines
4.2 KiB
Go
package amp
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"io"
|
|
"net/http/httputil"
|
|
"strings"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/util"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
// FallbackHandler wraps a standard handler with fallback logic to ampcode.com
|
|
// when the model's provider is not available in CLIProxyAPI
|
|
type FallbackHandler struct {
|
|
getProxy func() *httputil.ReverseProxy
|
|
}
|
|
|
|
// NewFallbackHandler creates a new fallback handler wrapper
|
|
// The getProxy function allows lazy evaluation of the proxy (useful when proxy is created after routes)
|
|
func NewFallbackHandler(getProxy func() *httputil.ReverseProxy) *FallbackHandler {
|
|
return &FallbackHandler{
|
|
getProxy: getProxy,
|
|
}
|
|
}
|
|
|
|
// WrapHandler wraps a gin.HandlerFunc with fallback logic
|
|
// If the model's provider is not configured in CLIProxyAPI, it forwards to ampcode.com
|
|
func (fh *FallbackHandler) WrapHandler(handler gin.HandlerFunc) gin.HandlerFunc {
|
|
return func(c *gin.Context) {
|
|
// Read the request body to extract the model name
|
|
bodyBytes, err := io.ReadAll(c.Request.Body)
|
|
if err != nil {
|
|
log.Errorf("amp fallback: failed to read request body: %v", err)
|
|
handler(c)
|
|
return
|
|
}
|
|
|
|
// Restore the body for the handler to read
|
|
c.Request.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
// Try to extract model from request body or URL path (for Gemini)
|
|
modelName := extractModelFromRequest(bodyBytes, c)
|
|
if modelName == "" {
|
|
// Can't determine model, proceed with normal handler
|
|
handler(c)
|
|
return
|
|
}
|
|
|
|
// Normalize model (handles Gemini thinking suffixes)
|
|
normalizedModel, _ := util.NormalizeGeminiThinkingModel(modelName)
|
|
|
|
// Check if we have providers for this model
|
|
providers := util.GetProviderName(normalizedModel)
|
|
|
|
if len(providers) == 0 {
|
|
// No providers configured - check if we have a proxy for fallback
|
|
proxy := fh.getProxy()
|
|
if proxy != nil {
|
|
// Fallback to ampcode.com
|
|
log.Infof("amp fallback: model %s has no configured provider, forwarding to ampcode.com", modelName)
|
|
|
|
// Restore body again for the proxy
|
|
c.Request.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
|
|
// Forward to ampcode.com
|
|
proxy.ServeHTTP(c.Writer, c.Request)
|
|
return
|
|
}
|
|
|
|
// No proxy available, let the normal handler return the error
|
|
log.Debugf("amp fallback: model %s has no configured provider and no proxy available", modelName)
|
|
}
|
|
|
|
// Providers available or no proxy for fallback, restore body and use normal handler
|
|
// Filter Anthropic-Beta header to remove features requiring special subscription
|
|
// This is needed when using local providers (bypassing the Amp proxy)
|
|
if betaHeader := c.Request.Header.Get("Anthropic-Beta"); betaHeader != "" {
|
|
filtered := filterBetaFeatures(betaHeader, "context-1m-2025-08-07")
|
|
if filtered != "" {
|
|
c.Request.Header.Set("Anthropic-Beta", filtered)
|
|
} else {
|
|
c.Request.Header.Del("Anthropic-Beta")
|
|
}
|
|
}
|
|
|
|
c.Request.Body = io.NopCloser(bytes.NewReader(bodyBytes))
|
|
handler(c)
|
|
}
|
|
}
|
|
|
|
// extractModelFromRequest attempts to extract the model name from various request formats
|
|
func extractModelFromRequest(body []byte, c *gin.Context) string {
|
|
// First try to parse from JSON body (OpenAI, Claude, etc.)
|
|
var payload map[string]interface{}
|
|
if err := json.Unmarshal(body, &payload); err == nil {
|
|
// Check common model field names
|
|
if model, ok := payload["model"].(string); ok {
|
|
return model
|
|
}
|
|
}
|
|
|
|
// For Gemini requests, model is in the URL path
|
|
// Standard format: /models/{model}:generateContent -> :action parameter
|
|
if action := c.Param("action"); action != "" {
|
|
// Split by colon to get model name (e.g., "gemini-pro:generateContent" -> "gemini-pro")
|
|
parts := strings.Split(action, ":")
|
|
if len(parts) > 0 && parts[0] != "" {
|
|
return parts[0]
|
|
}
|
|
}
|
|
|
|
// AMP CLI format: /publishers/google/models/{model}:method -> *path parameter
|
|
// Example: /publishers/google/models/gemini-3-pro-preview:streamGenerateContent
|
|
if path := c.Param("path"); path != "" {
|
|
// Look for /models/{model}:method pattern
|
|
if idx := strings.Index(path, "/models/"); idx >= 0 {
|
|
modelPart := path[idx+8:] // Skip "/models/"
|
|
// Split by colon to get model name
|
|
if colonIdx := strings.Index(modelPart, ":"); colonIdx > 0 {
|
|
return modelPart[:colonIdx]
|
|
}
|
|
}
|
|
}
|
|
|
|
return ""
|
|
}
|