mirror of
https://github.com/router-for-me/CLIProxyAPI.git
synced 2026-02-03 04:50:52 +08:00
Updated Antigravity, Gemini, and Gemini-CLI translators to process `systemResult` of type `string` for system instructions. Ensures properly formatted JSON with dynamic content assignment.
181 lines
7.0 KiB
Go
181 lines
7.0 KiB
Go
// Package claude provides request translation functionality for Claude API.
|
|
// It handles parsing and transforming Claude API requests into the internal client format,
|
|
// extracting model information, system instructions, message contents, and tool declarations.
|
|
// The package also performs JSON data cleaning and transformation to ensure compatibility
|
|
// between Claude API format and the internal client's expected format.
|
|
package claude
|
|
|
|
import (
|
|
"bytes"
|
|
"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 geminiClaudeThoughtSignature = "skip_thought_signature_validator"
|
|
|
|
// ConvertClaudeRequestToGemini parses a Claude API request and returns a complete
|
|
// Gemini CLI request body (as JSON bytes) ready to be sent via SendRawMessageStream.
|
|
// All JSON transformations are performed using gjson/sjson.
|
|
//
|
|
// Parameters:
|
|
// - modelName: The name of the model.
|
|
// - rawJSON: The raw JSON request from the Claude API.
|
|
// - stream: A boolean indicating if the request is for a streaming response.
|
|
//
|
|
// Returns:
|
|
// - []byte: The transformed request in Gemini CLI format.
|
|
func ConvertClaudeRequestToGemini(modelName string, inputRawJSON []byte, _ bool) []byte {
|
|
rawJSON := bytes.Clone(inputRawJSON)
|
|
rawJSON = bytes.Replace(rawJSON, []byte(`"url":{"type":"string","format":"uri",`), []byte(`"url":{"type":"string",`), -1)
|
|
|
|
// Build output Gemini CLI request JSON
|
|
out := `{"contents":[]}`
|
|
out, _ = sjson.Set(out, "model", modelName)
|
|
|
|
// system instruction
|
|
if systemResult := gjson.GetBytes(rawJSON, "system"); systemResult.IsArray() {
|
|
systemInstruction := `{"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 := `{"text":""}`
|
|
part, _ = sjson.Set(part, "text", textResult.String())
|
|
systemInstruction, _ = sjson.SetRaw(systemInstruction, "parts.-1", part)
|
|
hasSystemParts = true
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
if hasSystemParts {
|
|
out, _ = sjson.SetRaw(out, "system_instruction", systemInstruction)
|
|
}
|
|
} else if systemResult.Type == gjson.String {
|
|
out, _ = sjson.Set(out, "request.system_instruction.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 := `{"role":"","parts":[]}`
|
|
contentJSON, _ = sjson.Set(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 := `{"text":""}`
|
|
part, _ = sjson.Set(part, "text", contentResult.Get("text").String())
|
|
contentJSON, _ = sjson.SetRaw(contentJSON, "parts.-1", part)
|
|
|
|
case "tool_use":
|
|
functionName := contentResult.Get("name").String()
|
|
functionArgs := contentResult.Get("input").String()
|
|
argsResult := gjson.Parse(functionArgs)
|
|
if argsResult.IsObject() && gjson.Valid(functionArgs) {
|
|
part := `{"thoughtSignature":"","functionCall":{"name":"","args":{}}}`
|
|
part, _ = sjson.Set(part, "thoughtSignature", geminiClaudeThoughtSignature)
|
|
part, _ = sjson.Set(part, "functionCall.name", functionName)
|
|
part, _ = sjson.SetRaw(part, "functionCall.args", functionArgs)
|
|
contentJSON, _ = sjson.SetRaw(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 := `{"functionResponse":{"name":"","response":{"result":""}}}`
|
|
part, _ = sjson.Set(part, "functionResponse.name", funcName)
|
|
part, _ = sjson.Set(part, "functionResponse.response.result", responseData)
|
|
contentJSON, _ = sjson.SetRaw(contentJSON, "parts.-1", part)
|
|
}
|
|
return true
|
|
})
|
|
out, _ = sjson.SetRaw(out, "contents.-1", contentJSON)
|
|
} else if contentsResult.Type == gjson.String {
|
|
part := `{"text":""}`
|
|
part, _ = sjson.Set(part, "text", contentsResult.String())
|
|
contentJSON, _ = sjson.SetRaw(contentJSON, "parts.-1", part)
|
|
out, _ = sjson.SetRaw(out, "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 := inputSchemaResult.Raw
|
|
tool, _ := sjson.Delete(toolResult.Raw, "input_schema")
|
|
tool, _ = sjson.SetRaw(tool, "parametersJsonSchema", inputSchema)
|
|
tool, _ = sjson.Delete(tool, "strict")
|
|
tool, _ = sjson.Delete(tool, "input_examples")
|
|
tool, _ = sjson.Delete(tool, "type")
|
|
tool, _ = sjson.Delete(tool, "cache_control")
|
|
if gjson.Valid(tool) && gjson.Parse(tool).IsObject() {
|
|
if !hasTools {
|
|
out, _ = sjson.SetRaw(out, "tools", `[{"functionDeclarations":[]}]`)
|
|
hasTools = true
|
|
}
|
|
out, _ = sjson.SetRaw(out, "tools.0.functionDeclarations.-1", tool)
|
|
}
|
|
}
|
|
return true
|
|
})
|
|
if !hasTools {
|
|
out, _ = sjson.Delete(out, "tools")
|
|
}
|
|
}
|
|
|
|
// Map Anthropic thinking -> Gemini thinkingBudget/include_thoughts when enabled
|
|
// Only apply for models that use numeric budgets, not discrete levels.
|
|
if t := gjson.GetBytes(rawJSON, "thinking"); t.Exists() && t.IsObject() && util.ModelSupportsThinking(modelName) && !util.ModelUsesThinkingLevels(modelName) {
|
|
if t.Get("type").String() == "enabled" {
|
|
if b := t.Get("budget_tokens"); b.Exists() && b.Type == gjson.Number {
|
|
budget := int(b.Int())
|
|
out, _ = sjson.Set(out, "generationConfig.thinkingConfig.thinkingBudget", budget)
|
|
out, _ = sjson.Set(out, "generationConfig.thinkingConfig.include_thoughts", true)
|
|
}
|
|
}
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "temperature"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.Set(out, "generationConfig.temperature", v.Num)
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "top_p"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.Set(out, "generationConfig.topP", v.Num)
|
|
}
|
|
if v := gjson.GetBytes(rawJSON, "top_k"); v.Exists() && v.Type == gjson.Number {
|
|
out, _ = sjson.Set(out, "generationConfig.topK", v.Num)
|
|
}
|
|
|
|
result := []byte(out)
|
|
result = common.AttachDefaultSafetySettings(result, "safetySettings")
|
|
|
|
return result
|
|
}
|