mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 04:50:52 +08:00
Gemini handler extracts model from URL path, not JSON body, so rewriting the request body alone wasn't sufficient for model mapping. - Add MappedModelContextKey constant for context passing - Update routes.go to use NewFallbackHandlerWithMapper - Add check for valid mapping before routing to local handler - Add tests for gemini bridge model mapping
121 lines
3.4 KiB
Go
121 lines
3.4 KiB
Go
package amp
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/router-for-me/CLIProxyAPI/v6/sdk/api/handlers"
|
|
"github.com/router-for-me/CLIProxyAPI/v6/sdk/api/handlers/gemini"
|
|
)
|
|
|
|
func TestCreateGeminiBridgeHandler_ActionParameterExtraction(t *testing.T) {
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
tests := []struct {
|
|
name string
|
|
path string
|
|
mappedModel string // empty string means no mapping
|
|
expectedAction string
|
|
}{
|
|
{
|
|
name: "no_mapping_uses_url_model",
|
|
path: "/publishers/google/models/gemini-pro:generateContent",
|
|
mappedModel: "",
|
|
expectedAction: "gemini-pro:generateContent",
|
|
},
|
|
{
|
|
name: "mapped_model_replaces_url_model",
|
|
path: "/publishers/google/models/gemini-exp:generateContent",
|
|
mappedModel: "gemini-2.0-flash",
|
|
expectedAction: "gemini-2.0-flash:generateContent",
|
|
},
|
|
{
|
|
name: "mapping_preserves_method",
|
|
path: "/publishers/google/models/gemini-2.5-preview:streamGenerateContent",
|
|
mappedModel: "gemini-flash",
|
|
expectedAction: "gemini-flash:streamGenerateContent",
|
|
},
|
|
{
|
|
name: "empty_mapped_model_ignored",
|
|
path: "/publishers/google/models/gemini-pro:generateContent",
|
|
mappedModel: "",
|
|
expectedAction: "gemini-pro:generateContent",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
var capturedAction string
|
|
|
|
mockGeminiHandler := func(c *gin.Context) {
|
|
capturedAction = c.Param("action")
|
|
c.JSON(http.StatusOK, gin.H{"captured": capturedAction})
|
|
}
|
|
|
|
// Mirror the bridge logic from gemini_bridge.go
|
|
bridgeHandler := func(c *gin.Context) {
|
|
path := c.Param("path")
|
|
if idx := strings.Index(path, "/models/"); idx >= 0 {
|
|
actionPart := path[idx+8:]
|
|
|
|
if mappedModel, exists := c.Get(MappedModelContextKey); exists {
|
|
if strModel, ok := mappedModel.(string); ok && strModel != "" {
|
|
if colonIdx := strings.Index(actionPart, ":"); colonIdx > 0 {
|
|
method := actionPart[colonIdx:]
|
|
actionPart = strModel + method
|
|
}
|
|
}
|
|
}
|
|
|
|
c.Params = append(c.Params, gin.Param{Key: "action", Value: actionPart})
|
|
mockGeminiHandler(c)
|
|
return
|
|
}
|
|
c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid path"})
|
|
}
|
|
|
|
r := gin.New()
|
|
if tt.mappedModel != "" {
|
|
r.Use(func(c *gin.Context) {
|
|
c.Set(MappedModelContextKey, tt.mappedModel)
|
|
c.Next()
|
|
})
|
|
}
|
|
r.POST("/api/provider/google/v1beta1/*path", bridgeHandler)
|
|
|
|
req := httptest.NewRequest(http.MethodPost, "/api/provider/google/v1beta1"+tt.path, nil)
|
|
w := httptest.NewRecorder()
|
|
r.ServeHTTP(w, req)
|
|
|
|
if w.Code != http.StatusOK {
|
|
t.Fatalf("Expected status 200, got %d", w.Code)
|
|
}
|
|
if capturedAction != tt.expectedAction {
|
|
t.Errorf("Expected action '%s', got '%s'", tt.expectedAction, capturedAction)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestCreateGeminiBridgeHandler_InvalidPath(t *testing.T) {
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
base := &handlers.BaseAPIHandler{}
|
|
geminiHandlers := gemini.NewGeminiAPIHandler(base)
|
|
bridgeHandler := createGeminiBridgeHandler(geminiHandlers)
|
|
|
|
r := gin.New()
|
|
r.POST("/api/provider/google/v1beta1/*path", bridgeHandler)
|
|
|
|
req := httptest.NewRequest(http.MethodPost, "/api/provider/google/v1beta1/invalid/path", nil)
|
|
w := httptest.NewRecorder()
|
|
r.ServeHTTP(w, req)
|
|
|
|
if w.Code != http.StatusBadRequest {
|
|
t.Errorf("Expected status 400 for invalid path, got %d", w.Code)
|
|
}
|
|
}
|