mirror of
https://github.com/router-for-me/CLIProxyAPIPlus.git
synced 2026-04-18 12:23:44 +00:00
Claude Code and MCP clients may send tool names containing characters invalid for Gemini's function_declarations (e.g. '/', '@', spaces). Sanitize on request via SanitizeFunctionName and restore original names on response for both antigravity/claude and gemini-cli/claude translators.
243 lines
10 KiB
Go
243 lines
10 KiB
Go
// Package claude provides request translation functionality for Claude Code API compatibility.
|
|
// This package handles the conversion of Claude Code API requests into Gemini CLI-compatible
|
|
// JSON format, transforming message contents, system instructions, and tool declarations
|
|
// into the format expected by Gemini CLI API clients. It performs JSON data transformation
|
|
// to ensure compatibility between Claude Code API format and Gemini CLI API's expected format.
|
|
package claude
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/translator/gemini/common"
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/util"
|
|
"github.com/tidwall/gjson"
|
|
"github.com/tidwall/sjson"
|
|
)
|
|
|
|
const geminiCLIClaudeThoughtSignature = "skip_thought_signature_validator"
|
|
|
|
// ConvertClaudeRequestToCLI parses and transforms a Claude Code API request into Gemini CLI API format.
|
|
// It extracts the model name, system instruction, message contents, and tool declarations
|
|
// from the raw JSON request and returns them in the format expected by the Gemini CLI API.
|
|
// The function performs the following transformations:
|
|
// 1. Extracts the model information from the request
|
|
// 2. Restructures the JSON to match Gemini CLI API format
|
|
// 3. Converts system instructions to the expected format
|
|
// 4. Maps message contents with proper role transformations
|
|
// 5. Handles tool declarations and tool choices
|
|
// 6. Maps generation configuration parameters
|
|
//
|
|
// Parameters:
|
|
// - modelName: The name of the model to use for the request
|
|
// - rawJSON: The raw JSON request data from the Claude Code API
|
|
// - stream: A boolean indicating if the request is for a streaming response (unused in current implementation)
|
|
//
|
|
// Returns:
|
|
// - []byte: The transformed request data in Gemini CLI API format
|
|
func ConvertClaudeRequestToCLI(modelName string, inputRawJSON []byte, _ bool) []byte {
|
|
rawJSON := inputRawJSON
|
|
|
|
// Build output Gemini CLI request JSON
|
|
out := []byte(`{"model":"","request":{"contents":[]}}`)
|
|
out, _ = sjson.SetBytes(out, "model", modelName)
|
|
|
|
// system instruction
|
|
if systemResult := gjson.GetBytes(rawJSON, "system"); systemResult.IsArray() {
|
|
systemInstruction := []byte(`{"role":"user","parts":[]}`)
|
|
hasSystemParts := false
|
|
systemResult.ForEach(func(_, systemPromptResult gjson.Result) bool {
|
|
if systemPromptResult.Get("type").String() == "text" {
|
|
textResult := systemPromptResult.Get("text")
|
|
if textResult.Type == gjson.String {
|
|
part := []byte(`{"text":""}`)
|
|
part, _ = sjson.SetBytes(part, "text", textResult.String())
|
|
systemInstruction, _ = sjson.SetRawBytes(systemInstruction, "parts.-1", part)
|
|
hasSystemParts = true
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
if hasSystemParts {
|
|
out, _ = sjson.SetRawBytes(out, "request.systemInstruction", systemInstruction)
|
|
}
|
|
} else if systemResult.Type == gjson.String {
|
|
out, _ = sjson.SetBytes(out, "request.systemInstruction.parts.-1.text", systemResult.String())
|
|
}
|
|
|
|
// contents
|
|
if messagesResult := gjson.GetBytes(rawJSON, "messages"); messagesResult.IsArray() {
|
|
messagesResult.ForEach(func(_, messageResult gjson.Result) bool {
|
|
roleResult := messageResult.Get("role")
|
|
if roleResult.Type != gjson.String {
|
|
return true
|
|
}
|
|
role := roleResult.String()
|
|
if role == "assistant" {
|
|
role = "model"
|
|
}
|
|
|
|
contentJSON := []byte(`{"role":"","parts":[]}`)
|
|
contentJSON, _ = sjson.SetBytes(contentJSON, "role", role)
|
|
|
|
contentsResult := messageResult.Get("content")
|
|
if contentsResult.IsArray() {
|
|
contentsResult.ForEach(func(_, contentResult gjson.Result) bool {
|
|
switch contentResult.Get("type").String() {
|
|
case "text":
|
|
part := []byte(`{"text":""}`)
|
|
part, _ = sjson.SetBytes(part, "text", contentResult.Get("text").String())
|
|
contentJSON, _ = sjson.SetRawBytes(contentJSON, "parts.-1", part)
|
|
|
|
case "tool_use":
|
|
functionName := util.SanitizeFunctionName(contentResult.Get("name").String())
|
|
functionArgs := contentResult.Get("input").String()
|
|
argsResult := gjson.Parse(functionArgs)
|
|
if argsResult.IsObject() && gjson.Valid(functionArgs) {
|
|
part := []byte(`{"thoughtSignature":"","functionCall":{"name":"","args":{}}}`)
|
|
part, _ = sjson.SetBytes(part, "thoughtSignature", geminiCLIClaudeThoughtSignature)
|
|
part, _ = sjson.SetBytes(part, "functionCall.name", functionName)
|
|
part, _ = sjson.SetRawBytes(part, "functionCall.args", []byte(functionArgs))
|
|
contentJSON, _ = sjson.SetRawBytes(contentJSON, "parts.-1", part)
|
|
}
|
|
|
|
case "tool_result":
|
|
toolCallID := contentResult.Get("tool_use_id").String()
|
|
if toolCallID == "" {
|
|
return true
|
|
}
|
|
funcName := toolCallID
|
|
toolCallIDs := strings.Split(toolCallID, "-")
|
|
if len(toolCallIDs) > 1 {
|
|
funcName = strings.Join(toolCallIDs[0:len(toolCallIDs)-1], "-")
|
|
}
|
|
responseData := contentResult.Get("content").Raw
|
|
part := []byte(`{"functionResponse":{"name":"","response":{"result":""}}}`)
|
|
part, _ = sjson.SetBytes(part, "functionResponse.name", util.SanitizeFunctionName(funcName))
|
|
part, _ = sjson.SetBytes(part, "functionResponse.response.result", responseData)
|
|
contentJSON, _ = sjson.SetRawBytes(contentJSON, "parts.-1", part)
|
|
|
|
case "image":
|
|
source := contentResult.Get("source")
|
|
if source.Get("type").String() == "base64" {
|
|
mimeType := source.Get("media_type").String()
|
|
data := source.Get("data").String()
|
|
if mimeType != "" && data != "" {
|
|
part := []byte(`{"inlineData":{"mime_type":"","data":""}}`)
|
|
part, _ = sjson.SetBytes(part, "inlineData.mime_type", mimeType)
|
|
part, _ = sjson.SetBytes(part, "inlineData.data", data)
|
|
contentJSON, _ = sjson.SetRawBytes(contentJSON, "parts.-1", part)
|
|
}
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
out, _ = sjson.SetRawBytes(out, "request.contents.-1", contentJSON)
|
|
} else if contentsResult.Type == gjson.String {
|
|
part := []byte(`{"text":""}`)
|
|
part, _ = sjson.SetBytes(part, "text", contentsResult.String())
|
|
contentJSON, _ = sjson.SetRawBytes(contentJSON, "parts.-1", part)
|
|
out, _ = sjson.SetRawBytes(out, "request.contents.-1", contentJSON)
|
|
}
|
|
return true
|
|
})
|
|
}
|
|
|
|
// tools
|
|
if toolsResult := gjson.GetBytes(rawJSON, "tools"); toolsResult.IsArray() {
|
|
hasTools := false
|
|
toolsResult.ForEach(func(_, toolResult gjson.Result) bool {
|
|
inputSchemaResult := toolResult.Get("input_schema")
|
|
if inputSchemaResult.Exists() && inputSchemaResult.IsObject() {
|
|
inputSchema := util.CleanJSONSchemaForGemini(inputSchemaResult.Raw)
|
|
tool, _ := sjson.DeleteBytes([]byte(toolResult.Raw), "input_schema")
|
|
tool, _ = sjson.SetRawBytes(tool, "parametersJsonSchema", []byte(inputSchema))
|
|
tool, _ = sjson.SetBytes(tool, "name", util.SanitizeFunctionName(gjson.GetBytes(tool, "name").String()))
|
|
tool, _ = sjson.DeleteBytes(tool, "strict")
|
|
tool, _ = sjson.DeleteBytes(tool, "input_examples")
|
|
tool, _ = sjson.DeleteBytes(tool, "type")
|
|
tool, _ = sjson.DeleteBytes(tool, "cache_control")
|
|
tool, _ = sjson.DeleteBytes(tool, "defer_loading")
|
|
tool, _ = sjson.DeleteBytes(tool, "eager_input_streaming")
|
|
if gjson.ValidBytes(tool) && gjson.ParseBytes(tool).IsObject() {
|
|
if !hasTools {
|
|
out, _ = sjson.SetRawBytes(out, "request.tools", []byte(`[{"functionDeclarations":[]}]`))
|
|
hasTools = true
|
|
}
|
|
out, _ = sjson.SetRawBytes(out, "request.tools.0.functionDeclarations.-1", tool)
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
if !hasTools {
|
|
out, _ = sjson.DeleteBytes(out, "request.tools")
|
|
}
|
|
}
|
|
|
|
// tool_choice
|
|
toolChoiceResult := gjson.GetBytes(rawJSON, "tool_choice")
|
|
if toolChoiceResult.Exists() {
|
|
toolChoiceType := ""
|
|
toolChoiceName := ""
|
|
if toolChoiceResult.IsObject() {
|
|
toolChoiceType = toolChoiceResult.Get("type").String()
|
|
toolChoiceName = toolChoiceResult.Get("name").String()
|
|
} else if toolChoiceResult.Type == gjson.String {
|
|
toolChoiceType = toolChoiceResult.String()
|
|
}
|
|
|
|
switch toolChoiceType {
|
|
case "auto":
|
|
out, _ = sjson.SetBytes(out, "request.toolConfig.functionCallingConfig.mode", "AUTO")
|
|
case "none":
|
|
out, _ = sjson.SetBytes(out, "request.toolConfig.functionCallingConfig.mode", "NONE")
|
|
case "any":
|
|
out, _ = sjson.SetBytes(out, "request.toolConfig.functionCallingConfig.mode", "ANY")
|
|
case "tool":
|
|
out, _ = sjson.SetBytes(out, "request.toolConfig.functionCallingConfig.mode", "ANY")
|
|
if toolChoiceName != "" {
|
|
out, _ = sjson.SetBytes(out, "request.toolConfig.functionCallingConfig.allowedFunctionNames", []string{util.SanitizeFunctionName(toolChoiceName)})
|
|
}
|
|
}
|
|
}
|
|
|
|
// Map Anthropic thinking -> Gemini CLI thinkingConfig when enabled
|
|
// Translator only does format conversion, ApplyThinking handles model capability validation.
|
|
if t := gjson.GetBytes(rawJSON, "thinking"); t.Exists() && t.IsObject() {
|
|
switch t.Get("type").String() {
|
|
case "enabled":
|
|
if b := t.Get("budget_tokens"); b.Exists() && b.Type == gjson.Number {
|
|
budget := int(b.Int())
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.thinkingConfig.thinkingBudget", budget)
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.thinkingConfig.includeThoughts", true)
|
|
}
|
|
case "adaptive", "auto":
|
|
// For adaptive thinking:
|
|
// - If output_config.effort is explicitly present, pass through as thinkingLevel.
|
|
// - Otherwise, treat it as "enabled with target-model maximum" and emit high.
|
|
// ApplyThinking handles clamping to target model's supported levels.
|
|
effort := ""
|
|
if v := gjson.GetBytes(rawJSON, "output_config.effort"); v.Exists() && v.Type == gjson.String {
|
|
effort = strings.ToLower(strings.TrimSpace(v.String()))
|
|
}
|
|
if effort != "" {
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.thinkingConfig.thinkingLevel", effort)
|
|
} else {
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.thinkingConfig.thinkingLevel", "high")
|
|
}
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.thinkingConfig.includeThoughts", true)
|
|
}
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "temperature"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.temperature", v.Num)
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "top_p"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.topP", v.Num)
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "top_k"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.SetBytes(out, "request.generationConfig.topK", v.Num)
|
|
}
|
|
|
|
out = common.AttachDefaultSafetySettings(out, "request.safetySettings")
|
|
return out
|
|
}
|