Merge pull request #3554 from sususu98/fix/gemini-cli-request-schema-cleanup

fix: clean Gemini CLI request schemas
This commit is contained in:
sususu98
2026-05-26 16:37:15 +08:00
committed by GitHub
2 changed files with 137 additions and 0 deletions

View File

@@ -141,6 +141,7 @@ func (e *GeminiCLIExecutor) Execute(ctx context.Context, auth *cliproxyauth.Auth
requestedModel := helps.PayloadRequestedModel(opts, req.Model)
requestPath := helps.PayloadRequestPath(opts)
basePayload = helps.ApplyPayloadConfigWithRequest(e.cfg, baseModel, "gemini", from.String(), "request", basePayload, originalTranslated, requestedModel, requestPath, opts.Headers)
basePayload = cleanGeminiCLIRequestSchemas(basePayload)
action := "generateContent"
if req.Metadata != nil {
@@ -297,6 +298,7 @@ func (e *GeminiCLIExecutor) ExecuteStream(ctx context.Context, auth *cliproxyaut
requestedModel := helps.PayloadRequestedModel(opts, req.Model)
requestPath := helps.PayloadRequestPath(opts)
basePayload = helps.ApplyPayloadConfigWithRequest(e.cfg, baseModel, "gemini", from.String(), "request", basePayload, originalTranslated, requestedModel, requestPath, opts.Headers)
basePayload = cleanGeminiCLIRequestSchemas(basePayload)
projectID := resolveGeminiProjectID(auth)
@@ -530,6 +532,7 @@ func (e *GeminiCLIExecutor) CountTokens(ctx context.Context, auth *cliproxyauth.
payload = deleteJSONField(payload, "model")
payload = deleteJSONField(payload, "request.safetySettings")
payload = fixGeminiCLIImageAspectRatio(baseModel, payload)
payload = cleanGeminiCLIRequestSchemas(payload)
tok, errTok := tokenSource.Token()
if errTok != nil {
@@ -859,6 +862,65 @@ func deleteJSONField(body []byte, key string) []byte {
return updated
}
func cleanGeminiCLIRequestSchemas(body []byte) []byte {
if len(body) == 0 {
return body
}
hasTools := gjson.GetBytes(body, "request.tools.0").Exists()
hasResponseSchema := gjson.GetBytes(body, "request.generationConfig.responseSchema").Exists()
hasResponseJSONSchema := gjson.GetBytes(body, "request.generationConfig.responseJsonSchema").Exists()
if !hasTools && !hasResponseSchema && !hasResponseJSONSchema {
return body
}
tools := gjson.GetBytes(body, "request.tools")
if tools.IsArray() {
for i, tool := range tools.Array() {
for _, declarationsKey := range []string{"function_declarations", "functionDeclarations"} {
funcDecls := tool.Get(declarationsKey)
if !funcDecls.IsArray() {
continue
}
for j, decl := range funcDecls.Array() {
for _, schemaKey := range []string{"parameters", "parametersJsonSchema"} {
params := decl.Get(schemaKey)
if !params.Exists() || !params.IsObject() {
continue
}
cleaned := util.CleanJSONSchemaForGemini(params.Raw)
path := fmt.Sprintf("request.tools.%d.%s.%d.%s", i, declarationsKey, j, schemaKey)
updated, errSet := sjson.SetRawBytes(body, path, []byte(cleaned))
if errSet != nil {
log.Errorf("gemini cli executor: failed to set cleaned schema at %s: %v", path, errSet)
continue
}
body = updated
}
}
}
}
}
for _, schemaPath := range []string{
"request.generationConfig.responseSchema",
"request.generationConfig.responseJsonSchema",
} {
responseSchema := gjson.GetBytes(body, schemaPath)
if !responseSchema.IsObject() {
continue
}
cleaned := util.CleanJSONSchemaForGemini(responseSchema.Raw)
updated, errSet := sjson.SetRawBytes(body, schemaPath, []byte(cleaned))
if errSet != nil {
log.Errorf("gemini cli executor: failed to set cleaned response schema at %s: %v", schemaPath, errSet)
continue
}
body = updated
}
return body
}
func fixGeminiCLIImageAspectRatio(modelName string, rawJSON []byte) []byte {
if modelName == "gemini-2.5-flash-image-preview" {
aspectRatioResult := gjson.GetBytes(rawJSON, "request.generationConfig.imageConfig.aspectRatio")

View File

@@ -0,0 +1,75 @@
package executor
import (
"strings"
"testing"
"github.com/tidwall/gjson"
)
func TestCleanGeminiCLIRequestSchemasFlattensFunctionDeclarationTypeArray(t *testing.T) {
input := []byte(`{
"request": {
"tools": [
{
"function_declarations": [
{
"name": "wecom_mcp",
"parameters": {
"type": "object",
"properties": {
"args": {
"description": "call args",
"type": ["string", "object"]
}
}
}
}
]
},
{
"functionDeclarations": [
{
"name": "camel_tool",
"parametersJsonSchema": {
"type": "object",
"properties": {
"value": {
"type": ["integer", "string"]
}
}
}
}
]
}
],
"nonSchema": {
"type": ["string", "object"]
}
}
}`)
out := cleanGeminiCLIRequestSchemas(input)
argsType := gjson.GetBytes(out, "request.tools.0.function_declarations.0.parameters.properties.args.type")
if argsType.String() != "string" {
t.Fatalf("args.type = %s, want string; body=%s", argsType.Raw, string(out))
}
argsDesc := gjson.GetBytes(out, "request.tools.0.function_declarations.0.parameters.properties.args.description").String()
if !strings.Contains(argsDesc, "Accepts: string | object") {
t.Fatalf("args.description = %q, want accepted type hint", argsDesc)
}
valueType := gjson.GetBytes(out, "request.tools.1.functionDeclarations.0.parametersJsonSchema.properties.value.type")
if valueType.String() != "integer" {
t.Fatalf("value.type = %s, want integer; body=%s", valueType.Raw, string(out))
}
valueDesc := gjson.GetBytes(out, "request.tools.1.functionDeclarations.0.parametersJsonSchema.properties.value.description").String()
if !strings.Contains(valueDesc, "Accepts: integer | string") {
t.Fatalf("value.description = %q, want accepted type hint", valueDesc)
}
if nonSchema := gjson.GetBytes(out, "request.nonSchema.type"); !nonSchema.IsArray() {
t.Fatalf("request.nonSchema.type should be preserved outside schema paths, got %s", nonSchema.Raw)
}
}