| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971 |
- package claude
- import (
- "encoding/json"
- "fmt"
- "io"
- "net/http"
- "strings"
- "github.com/QuantumNous/new-api/common"
- "github.com/QuantumNous/new-api/constant"
- "github.com/QuantumNous/new-api/dto"
- "github.com/QuantumNous/new-api/logger"
- "github.com/QuantumNous/new-api/relay/channel/openrouter"
- relaycommon "github.com/QuantumNous/new-api/relay/common"
- "github.com/QuantumNous/new-api/relay/helper"
- "github.com/QuantumNous/new-api/relay/reasonmap"
- "github.com/QuantumNous/new-api/service"
- "github.com/QuantumNous/new-api/setting/model_setting"
- "github.com/QuantumNous/new-api/setting/reasoning"
- "github.com/QuantumNous/new-api/types"
- "github.com/gin-gonic/gin"
- "github.com/tidwall/gjson"
- "github.com/tidwall/sjson"
- )
- const (
- WebSearchMaxUsesLow = 1
- WebSearchMaxUsesMedium = 5
- WebSearchMaxUsesHigh = 10
- )
- func stopReasonClaude2OpenAI(reason string) string {
- return reasonmap.ClaudeStopReasonToOpenAIFinishReason(reason)
- }
- func maybeMarkClaudeRefusal(c *gin.Context, stopReason string) {
- if c == nil {
- return
- }
- if strings.EqualFold(stopReason, "refusal") {
- common.SetContextKey(c, constant.ContextKeyAdminRejectReason, "claude_stop_reason=refusal")
- }
- }
- func RequestOpenAI2ClaudeMessage(c *gin.Context, textRequest dto.GeneralOpenAIRequest) (*dto.ClaudeRequest, error) {
- claudeTools := make([]any, 0, len(textRequest.Tools))
- for _, tool := range textRequest.Tools {
- if params, ok := tool.Function.Parameters.(map[string]any); ok {
- claudeTool := dto.Tool{
- Name: tool.Function.Name,
- Description: tool.Function.Description,
- }
- claudeTool.InputSchema = make(map[string]interface{})
- if params["type"] != nil {
- claudeTool.InputSchema["type"] = params["type"].(string)
- }
- claudeTool.InputSchema["properties"] = params["properties"]
- claudeTool.InputSchema["required"] = params["required"]
- for s, a := range params {
- if s == "type" || s == "properties" || s == "required" {
- continue
- }
- claudeTool.InputSchema[s] = a
- }
- claudeTools = append(claudeTools, &claudeTool)
- }
- }
- // Web search tool
- // https://docs.anthropic.com/en/docs/agents-and-tools/tool-use/web-search-tool
- if textRequest.WebSearchOptions != nil {
- webSearchTool := dto.ClaudeWebSearchTool{
- Type: "web_search_20250305",
- Name: "web_search",
- }
- // 处理 user_location
- if textRequest.WebSearchOptions.UserLocation != nil {
- anthropicUserLocation := &dto.ClaudeWebSearchUserLocation{
- Type: "approximate", // 固定为 "approximate"
- }
- // 解析 UserLocation JSON
- var userLocationMap map[string]interface{}
- if err := json.Unmarshal(textRequest.WebSearchOptions.UserLocation, &userLocationMap); err == nil {
- // 检查是否有 approximate 字段
- if approximateData, ok := userLocationMap["approximate"].(map[string]interface{}); ok {
- if timezone, ok := approximateData["timezone"].(string); ok && timezone != "" {
- anthropicUserLocation.Timezone = timezone
- }
- if country, ok := approximateData["country"].(string); ok && country != "" {
- anthropicUserLocation.Country = country
- }
- if region, ok := approximateData["region"].(string); ok && region != "" {
- anthropicUserLocation.Region = region
- }
- if city, ok := approximateData["city"].(string); ok && city != "" {
- anthropicUserLocation.City = city
- }
- }
- }
- webSearchTool.UserLocation = anthropicUserLocation
- }
- // 处理 search_context_size 转换为 max_uses
- if textRequest.WebSearchOptions.SearchContextSize != "" {
- switch textRequest.WebSearchOptions.SearchContextSize {
- case "low":
- webSearchTool.MaxUses = WebSearchMaxUsesLow
- case "medium":
- webSearchTool.MaxUses = WebSearchMaxUsesMedium
- case "high":
- webSearchTool.MaxUses = WebSearchMaxUsesHigh
- }
- }
- claudeTools = append(claudeTools, &webSearchTool)
- }
- claudeRequest := dto.ClaudeRequest{
- Model: textRequest.Model,
- StopSequences: nil,
- Temperature: textRequest.Temperature,
- Tools: claudeTools,
- }
- if maxTokens := textRequest.GetMaxTokens(); maxTokens > 0 {
- claudeRequest.MaxTokens = common.GetPointer(maxTokens)
- }
- if textRequest.TopP != nil {
- claudeRequest.TopP = common.GetPointer(*textRequest.TopP)
- }
- if textRequest.TopK != nil {
- claudeRequest.TopK = common.GetPointer(*textRequest.TopK)
- }
- if textRequest.IsStream(nil) {
- claudeRequest.Stream = common.GetPointer(true)
- }
- // 处理 tool_choice 和 parallel_tool_calls
- if textRequest.ToolChoice != nil || textRequest.ParallelTooCalls != nil {
- claudeToolChoice := mapToolChoice(textRequest.ToolChoice, textRequest.ParallelTooCalls)
- if claudeToolChoice != nil {
- claudeRequest.ToolChoice = claudeToolChoice
- }
- }
- if claudeRequest.MaxTokens == nil || *claudeRequest.MaxTokens == 0 {
- defaultMaxTokens := uint(model_setting.GetClaudeSettings().GetDefaultMaxTokens(textRequest.Model))
- claudeRequest.MaxTokens = &defaultMaxTokens
- }
- if baseModel, effortLevel, ok := reasoning.TrimEffortSuffix(textRequest.Model); ok && effortLevel != "" &&
- strings.HasPrefix(textRequest.Model, "claude-opus-4-6") {
- claudeRequest.Model = baseModel
- claudeRequest.Thinking = &dto.Thinking{
- Type: "adaptive",
- }
- claudeRequest.OutputConfig = json.RawMessage(fmt.Sprintf(`{"effort":"%s"}`, effortLevel))
- claudeRequest.TopP = common.GetPointer[float64](0)
- claudeRequest.Temperature = common.GetPointer[float64](1.0)
- } else if model_setting.GetClaudeSettings().ThinkingAdapterEnabled &&
- strings.HasSuffix(textRequest.Model, "-thinking") {
- // 因为BudgetTokens 必须大于1024
- if claudeRequest.MaxTokens == nil || *claudeRequest.MaxTokens < 1280 {
- claudeRequest.MaxTokens = common.GetPointer[uint](1280)
- }
- // BudgetTokens 为 max_tokens 的 80%
- claudeRequest.Thinking = &dto.Thinking{
- Type: "enabled",
- BudgetTokens: common.GetPointer[int](int(float64(*claudeRequest.MaxTokens) * model_setting.GetClaudeSettings().ThinkingAdapterBudgetTokensPercentage)),
- }
- // TODO: 临时处理
- // https://docs.anthropic.com/en/docs/build-with-claude/extended-thinking#important-considerations-when-using-extended-thinking
- claudeRequest.TopP = common.GetPointer[float64](0)
- claudeRequest.Temperature = common.GetPointer[float64](1.0)
- if !model_setting.ShouldPreserveThinkingSuffix(textRequest.Model) {
- claudeRequest.Model = strings.TrimSuffix(textRequest.Model, "-thinking")
- }
- }
- if textRequest.ReasoningEffort != "" {
- switch textRequest.ReasoningEffort {
- case "low":
- claudeRequest.Thinking = &dto.Thinking{
- Type: "enabled",
- BudgetTokens: common.GetPointer[int](1280),
- }
- case "medium":
- claudeRequest.Thinking = &dto.Thinking{
- Type: "enabled",
- BudgetTokens: common.GetPointer[int](2048),
- }
- case "high":
- claudeRequest.Thinking = &dto.Thinking{
- Type: "enabled",
- BudgetTokens: common.GetPointer[int](4096),
- }
- }
- }
- // 指定了 reasoning 参数,覆盖 budgetTokens
- if textRequest.Reasoning != nil {
- var reasoning openrouter.RequestReasoning
- if err := common.Unmarshal(textRequest.Reasoning, &reasoning); err != nil {
- return nil, err
- }
- budgetTokens := reasoning.MaxTokens
- if budgetTokens > 0 {
- claudeRequest.Thinking = &dto.Thinking{
- Type: "enabled",
- BudgetTokens: &budgetTokens,
- }
- }
- }
- if textRequest.Stop != nil {
- // stop maybe string/array string, convert to array string
- switch textRequest.Stop.(type) {
- case string:
- claudeRequest.StopSequences = []string{textRequest.Stop.(string)}
- case []interface{}:
- stopSequences := make([]string, 0)
- for _, stop := range textRequest.Stop.([]interface{}) {
- stopSequences = append(stopSequences, stop.(string))
- }
- claudeRequest.StopSequences = stopSequences
- }
- }
- formatMessages := make([]dto.Message, 0)
- lastMessage := dto.Message{
- Role: "tool",
- }
- for i, message := range textRequest.Messages {
- if message.Role == "" {
- textRequest.Messages[i].Role = "user"
- }
- fmtMessage := dto.Message{
- Role: message.Role,
- Content: message.Content,
- }
- if message.Role == "tool" {
- fmtMessage.ToolCallId = message.ToolCallId
- }
- if message.Role == "assistant" && message.ToolCalls != nil {
- fmtMessage.ToolCalls = message.ToolCalls
- if message.IsStringContent() && message.StringContent() == "" {
- fmtMessage.SetNullContent()
- }
- }
- if lastMessage.Role == message.Role && lastMessage.Role != "tool" {
- if lastMessage.IsStringContent() && message.IsStringContent() {
- fmtMessage.SetStringContent(strings.Trim(fmt.Sprintf("%s %s", lastMessage.StringContent(), message.StringContent()), "\""))
- // delete last message
- formatMessages = formatMessages[:len(formatMessages)-1]
- }
- }
- if fmtMessage.Content == nil && !(message.Role == "assistant" && message.ToolCalls != nil) {
- fmtMessage.SetStringContent("...")
- }
- formatMessages = append(formatMessages, fmtMessage)
- lastMessage = fmtMessage
- }
- claudeMessages := make([]dto.ClaudeMessage, 0)
- isFirstMessage := true
- // 初始化system消息数组,用于累积多个system消息
- var systemMessages []dto.ClaudeMediaMessage
- for _, message := range formatMessages {
- if message.Role == "system" {
- // 根据Claude API规范,system字段使用数组格式更有通用性
- if message.IsStringContent() {
- systemMessages = append(systemMessages, dto.ClaudeMediaMessage{
- Type: "text",
- Text: common.GetPointer[string](message.StringContent()),
- })
- } else {
- // 支持复合内容的system消息(虽然不常见,但需要考虑完整性)
- for _, ctx := range message.ParseContent() {
- if ctx.Type == "text" {
- systemMessages = append(systemMessages, dto.ClaudeMediaMessage{
- Type: "text",
- Text: common.GetPointer[string](ctx.Text),
- })
- }
- // 未来可以在这里扩展对图片等其他类型的支持
- }
- }
- } else {
- if isFirstMessage {
- isFirstMessage = false
- if message.Role != "user" {
- // fix: first message is assistant, add user message
- claudeMessage := dto.ClaudeMessage{
- Role: "user",
- Content: []dto.ClaudeMediaMessage{
- {
- Type: "text",
- Text: common.GetPointer[string]("..."),
- },
- },
- }
- claudeMessages = append(claudeMessages, claudeMessage)
- }
- }
- claudeMessage := dto.ClaudeMessage{
- Role: message.Role,
- }
- if message.Role == "tool" {
- if len(claudeMessages) > 0 && claudeMessages[len(claudeMessages)-1].Role == "user" {
- lastMessage := claudeMessages[len(claudeMessages)-1]
- if content, ok := lastMessage.Content.(string); ok {
- lastMessage.Content = []dto.ClaudeMediaMessage{
- {
- Type: "text",
- Text: common.GetPointer[string](content),
- },
- }
- }
- lastMessage.Content = append(lastMessage.Content.([]dto.ClaudeMediaMessage), dto.ClaudeMediaMessage{
- Type: "tool_result",
- ToolUseId: message.ToolCallId,
- Content: message.Content,
- })
- claudeMessages[len(claudeMessages)-1] = lastMessage
- continue
- } else {
- claudeMessage.Role = "user"
- claudeMessage.Content = []dto.ClaudeMediaMessage{
- {
- Type: "tool_result",
- ToolUseId: message.ToolCallId,
- Content: message.Content,
- },
- }
- }
- } else if message.IsStringContent() && message.ToolCalls == nil {
- claudeMessage.Content = message.StringContent()
- } else {
- claudeMediaMessages := make([]dto.ClaudeMediaMessage, 0)
- for _, mediaMessage := range message.ParseContent() {
- claudeMediaMessage := dto.ClaudeMediaMessage{
- Type: mediaMessage.Type,
- }
- if mediaMessage.Type == "text" {
- claudeMediaMessage.Text = common.GetPointer[string](mediaMessage.Text)
- } else {
- imageUrl := mediaMessage.GetImageMedia()
- claudeMediaMessage.Type = "image"
- claudeMediaMessage.Source = &dto.ClaudeMessageSource{
- Type: "base64",
- }
- // 使用统一的文件服务获取图片数据
- var source *types.FileSource
- if strings.HasPrefix(imageUrl.Url, "http") {
- source = types.NewURLFileSource(imageUrl.Url)
- } else {
- source = types.NewBase64FileSource(imageUrl.Url, "")
- }
- base64Data, mimeType, err := service.GetBase64Data(c, source, "formatting image for Claude")
- if err != nil {
- return nil, fmt.Errorf("get file data failed: %s", err.Error())
- }
- claudeMediaMessage.Source.MediaType = mimeType
- claudeMediaMessage.Source.Data = base64Data
- }
- claudeMediaMessages = append(claudeMediaMessages, claudeMediaMessage)
- }
- if message.ToolCalls != nil {
- for _, toolCall := range message.ParseToolCalls() {
- inputObj := make(map[string]any)
- if err := common.UnmarshalJsonStr(toolCall.Function.Arguments, &inputObj); err != nil {
- common.SysLog("tool call function arguments is not a map[string]any: " + fmt.Sprintf("%v", toolCall.Function.Arguments))
- inputObj = map[string]any{}
- }
- claudeMediaMessages = append(claudeMediaMessages, dto.ClaudeMediaMessage{
- Type: "tool_use",
- Id: toolCall.ID,
- Name: toolCall.Function.Name,
- Input: inputObj,
- })
- }
- }
- claudeMessage.Content = claudeMediaMessages
- }
- claudeMessages = append(claudeMessages, claudeMessage)
- }
- }
- // 设置累积的system消息
- if len(systemMessages) > 0 {
- claudeRequest.System = systemMessages
- }
- claudeRequest.Prompt = ""
- claudeRequest.Messages = claudeMessages
- return &claudeRequest, nil
- }
- func StreamResponseClaude2OpenAI(claudeResponse *dto.ClaudeResponse, claudeInfo *ClaudeResponseInfo) *dto.ChatCompletionsStreamResponse {
- var response dto.ChatCompletionsStreamResponse
- response.Object = "chat.completion.chunk"
- response.Model = claudeResponse.Model
- response.Choices = make([]dto.ChatCompletionsStreamResponseChoice, 0)
- tools := make([]dto.ToolCallResponse, 0)
- if claudeInfo != nil && claudeInfo.ToolCallStreamStates == nil {
- claudeInfo.ToolCallStreamStates = make(map[int]*ToolCallStreamState)
- }
- fcIdx := 0
- if claudeResponse.Index != nil {
- fcIdx = *claudeResponse.Index - 1
- if fcIdx < 0 {
- fcIdx = 0
- }
- }
- var choice dto.ChatCompletionsStreamResponseChoice
- if claudeResponse.Type == "message_start" {
- if claudeResponse.Message != nil {
- response.Id = claudeResponse.Message.Id
- response.Model = claudeResponse.Message.Model
- }
- //claudeUsage = &claudeResponse.Message.Usage
- choice.Delta.SetContentString("")
- choice.Delta.Role = "assistant"
- } else if claudeResponse.Type == "content_block_start" {
- if claudeResponse.ContentBlock != nil {
- // 如果是文本块,尽可能发送首段文本(若存在)
- if claudeResponse.ContentBlock.Type == "text" && claudeResponse.ContentBlock.Text != nil {
- choice.Delta.SetContentString(*claudeResponse.ContentBlock.Text)
- }
- if claudeResponse.ContentBlock.Type == "tool_use" {
- if claudeInfo != nil {
- claudeInfo.ToolCallStreamStates[fcIdx] = &ToolCallStreamState{
- ID: claudeResponse.ContentBlock.Id,
- Name: claudeResponse.ContentBlock.Name,
- }
- return nil
- }
- tools = append(tools, dto.ToolCallResponse{
- Index: common.GetPointer(fcIdx),
- ID: claudeResponse.ContentBlock.Id,
- Type: "function",
- Function: dto.FunctionResponse{
- Name: claudeResponse.ContentBlock.Name,
- Arguments: "",
- },
- })
- }
- } else {
- return nil
- }
- } else if claudeResponse.Type == "content_block_delta" {
- if claudeResponse.Delta != nil {
- choice.Delta.Content = claudeResponse.Delta.Text
- switch claudeResponse.Delta.Type {
- case "input_json_delta":
- if claudeResponse.Delta.PartialJson == nil {
- return nil
- }
- arguments := *claudeResponse.Delta.PartialJson
- if strings.TrimSpace(arguments) == "" {
- return nil
- }
- toolCall := dto.ToolCallResponse{
- Type: "function",
- Index: common.GetPointer(fcIdx),
- Function: dto.FunctionResponse{
- Arguments: arguments,
- },
- }
- if claudeInfo != nil {
- if state, ok := claudeInfo.ToolCallStreamStates[fcIdx]; ok {
- state.Emitted = true
- toolCall.ID = state.ID
- toolCall.Function.Name = state.Name
- }
- }
- tools = append(tools, toolCall)
- case "signature_delta":
- // 加密的不处理
- signatureContent := "\n"
- choice.Delta.ReasoningContent = &signatureContent
- case "thinking_delta":
- choice.Delta.ReasoningContent = claudeResponse.Delta.Thinking
- }
- }
- } else if claudeResponse.Type == "content_block_stop" {
- if claudeInfo == nil {
- return nil
- }
- state, ok := claudeInfo.ToolCallStreamStates[fcIdx]
- if !ok {
- return nil
- }
- delete(claudeInfo.ToolCallStreamStates, fcIdx)
- if state.Emitted {
- return nil
- }
- tools = append(tools, dto.ToolCallResponse{
- ID: state.ID,
- Type: "function",
- Index: common.GetPointer(fcIdx),
- Function: dto.FunctionResponse{
- Name: state.Name,
- Arguments: "{}",
- },
- })
- } else if claudeResponse.Type == "message_delta" {
- if claudeResponse.Delta != nil && claudeResponse.Delta.StopReason != nil {
- finishReason := stopReasonClaude2OpenAI(*claudeResponse.Delta.StopReason)
- if finishReason != "null" {
- choice.FinishReason = &finishReason
- }
- }
- //claudeUsage = &claudeResponse.Usage
- } else if claudeResponse.Type == "message_stop" {
- return nil
- } else {
- return nil
- }
- if len(tools) > 0 {
- choice.Delta.Content = nil // compatible with other OpenAI derivative applications, like LobeOpenAICompatibleFactory ...
- choice.Delta.ToolCalls = tools
- }
- response.Choices = append(response.Choices, choice)
- return &response
- }
- func ResponseClaude2OpenAI(claudeResponse *dto.ClaudeResponse) *dto.OpenAITextResponse {
- choices := make([]dto.OpenAITextResponseChoice, 0)
- fullTextResponse := dto.OpenAITextResponse{
- Id: fmt.Sprintf("chatcmpl-%s", common.GetUUID()),
- Object: "chat.completion",
- Created: common.GetTimestamp(),
- }
- var responseText string
- var responseThinking string
- if len(claudeResponse.Content) > 0 {
- responseText = claudeResponse.Content[0].GetText()
- if claudeResponse.Content[0].Thinking != nil {
- responseThinking = *claudeResponse.Content[0].Thinking
- }
- }
- tools := make([]dto.ToolCallResponse, 0)
- thinkingContent := ""
- fullTextResponse.Id = claudeResponse.Id
- for _, message := range claudeResponse.Content {
- switch message.Type {
- case "tool_use":
- args, _ := json.Marshal(message.Input)
- tools = append(tools, dto.ToolCallResponse{
- ID: message.Id,
- Type: "function", // compatible with other OpenAI derivative applications
- Function: dto.FunctionResponse{
- Name: message.Name,
- Arguments: string(args),
- },
- })
- case "thinking":
- // 加密的不管, 只输出明文的推理过程
- if message.Thinking != nil {
- thinkingContent = *message.Thinking
- }
- case "text":
- responseText = message.GetText()
- }
- }
- choice := dto.OpenAITextResponseChoice{
- Index: 0,
- Message: dto.Message{
- Role: "assistant",
- },
- FinishReason: stopReasonClaude2OpenAI(claudeResponse.StopReason),
- }
- choice.SetStringContent(responseText)
- if len(responseThinking) > 0 {
- choice.ReasoningContent = responseThinking
- }
- if len(tools) > 0 {
- choice.Message.SetToolCalls(tools)
- }
- choice.Message.ReasoningContent = thinkingContent
- fullTextResponse.Model = claudeResponse.Model
- choices = append(choices, choice)
- fullTextResponse.Choices = choices
- return &fullTextResponse
- }
- type ClaudeResponseInfo struct {
- ResponseId string
- Created int64
- Model string
- ResponseText strings.Builder
- Usage *dto.Usage
- Done bool
- ToolCallStreamStates map[int]*ToolCallStreamState
- }
- type ToolCallStreamState struct {
- ID string
- Name string
- Emitted bool
- }
- func buildMessageDeltaPatchUsage(claudeResponse *dto.ClaudeResponse, claudeInfo *ClaudeResponseInfo) *dto.ClaudeUsage {
- usage := &dto.ClaudeUsage{}
- if claudeResponse != nil && claudeResponse.Usage != nil {
- *usage = *claudeResponse.Usage
- }
- if claudeInfo == nil || claudeInfo.Usage == nil {
- return usage
- }
- if usage.InputTokens == 0 && claudeInfo.Usage.PromptTokens > 0 {
- usage.InputTokens = claudeInfo.Usage.PromptTokens
- }
- if usage.CacheReadInputTokens == 0 && claudeInfo.Usage.PromptTokensDetails.CachedTokens > 0 {
- usage.CacheReadInputTokens = claudeInfo.Usage.PromptTokensDetails.CachedTokens
- }
- if usage.CacheCreationInputTokens == 0 && claudeInfo.Usage.PromptTokensDetails.CachedCreationTokens > 0 {
- usage.CacheCreationInputTokens = claudeInfo.Usage.PromptTokensDetails.CachedCreationTokens
- }
- if usage.CacheCreation == nil && (claudeInfo.Usage.ClaudeCacheCreation5mTokens > 0 || claudeInfo.Usage.ClaudeCacheCreation1hTokens > 0) {
- usage.CacheCreation = &dto.ClaudeCacheCreationUsage{
- Ephemeral5mInputTokens: claudeInfo.Usage.ClaudeCacheCreation5mTokens,
- Ephemeral1hInputTokens: claudeInfo.Usage.ClaudeCacheCreation1hTokens,
- }
- }
- return usage
- }
- func shouldSkipClaudeMessageDeltaUsagePatch(info *relaycommon.RelayInfo) bool {
- if model_setting.GetGlobalSettings().PassThroughRequestEnabled {
- return true
- }
- if info == nil {
- return false
- }
- return info.ChannelSetting.PassThroughBodyEnabled
- }
- func patchClaudeMessageDeltaUsageData(data string, usage *dto.ClaudeUsage) string {
- if data == "" || usage == nil {
- return data
- }
- data = setMessageDeltaUsageInt(data, "usage.input_tokens", usage.InputTokens)
- data = setMessageDeltaUsageInt(data, "usage.cache_read_input_tokens", usage.CacheReadInputTokens)
- data = setMessageDeltaUsageInt(data, "usage.cache_creation_input_tokens", usage.CacheCreationInputTokens)
- if usage.CacheCreation != nil {
- data = setMessageDeltaUsageInt(data, "usage.cache_creation.ephemeral_5m_input_tokens", usage.CacheCreation.Ephemeral5mInputTokens)
- data = setMessageDeltaUsageInt(data, "usage.cache_creation.ephemeral_1h_input_tokens", usage.CacheCreation.Ephemeral1hInputTokens)
- }
- return data
- }
- func setMessageDeltaUsageInt(data string, path string, localValue int) string {
- if localValue <= 0 {
- return data
- }
- upstreamValue := gjson.Get(data, path)
- if upstreamValue.Exists() && upstreamValue.Int() > 0 {
- return data
- }
- patchedData, err := sjson.Set(data, path, localValue)
- if err != nil {
- return data
- }
- return patchedData
- }
- func FormatClaudeResponseInfo(claudeResponse *dto.ClaudeResponse, oaiResponse *dto.ChatCompletionsStreamResponse, claudeInfo *ClaudeResponseInfo) bool {
- if claudeInfo == nil {
- return false
- }
- if claudeInfo.Usage == nil {
- claudeInfo.Usage = &dto.Usage{}
- }
- if claudeResponse.Type == "message_start" {
- if claudeResponse.Message != nil {
- claudeInfo.ResponseId = claudeResponse.Message.Id
- claudeInfo.Model = claudeResponse.Message.Model
- }
- // message_start, 获取usage
- if claudeResponse.Message != nil && claudeResponse.Message.Usage != nil {
- claudeInfo.Usage.PromptTokens = claudeResponse.Message.Usage.InputTokens
- claudeInfo.Usage.PromptTokensDetails.CachedTokens = claudeResponse.Message.Usage.CacheReadInputTokens
- claudeInfo.Usage.PromptTokensDetails.CachedCreationTokens = claudeResponse.Message.Usage.CacheCreationInputTokens
- claudeInfo.Usage.ClaudeCacheCreation5mTokens = claudeResponse.Message.Usage.GetCacheCreation5mTokens()
- claudeInfo.Usage.ClaudeCacheCreation1hTokens = claudeResponse.Message.Usage.GetCacheCreation1hTokens()
- claudeInfo.Usage.CompletionTokens = claudeResponse.Message.Usage.OutputTokens
- }
- } else if claudeResponse.Type == "content_block_delta" {
- if claudeResponse.Delta != nil {
- if claudeResponse.Delta.Text != nil {
- claudeInfo.ResponseText.WriteString(*claudeResponse.Delta.Text)
- }
- if claudeResponse.Delta.Thinking != nil {
- claudeInfo.ResponseText.WriteString(*claudeResponse.Delta.Thinking)
- }
- }
- } else if claudeResponse.Type == "message_delta" {
- // 最终的usage获取
- if claudeResponse.Usage != nil {
- if claudeResponse.Usage.InputTokens > 0 {
- // 不叠加,只取最新的
- claudeInfo.Usage.PromptTokens = claudeResponse.Usage.InputTokens
- }
- if claudeResponse.Usage.CacheReadInputTokens > 0 {
- claudeInfo.Usage.PromptTokensDetails.CachedTokens = claudeResponse.Usage.CacheReadInputTokens
- }
- if claudeResponse.Usage.CacheCreationInputTokens > 0 {
- claudeInfo.Usage.PromptTokensDetails.CachedCreationTokens = claudeResponse.Usage.CacheCreationInputTokens
- }
- if cacheCreation5m := claudeResponse.Usage.GetCacheCreation5mTokens(); cacheCreation5m > 0 {
- claudeInfo.Usage.ClaudeCacheCreation5mTokens = cacheCreation5m
- }
- if cacheCreation1h := claudeResponse.Usage.GetCacheCreation1hTokens(); cacheCreation1h > 0 {
- claudeInfo.Usage.ClaudeCacheCreation1hTokens = cacheCreation1h
- }
- if claudeResponse.Usage.OutputTokens > 0 {
- claudeInfo.Usage.CompletionTokens = claudeResponse.Usage.OutputTokens
- }
- claudeInfo.Usage.TotalTokens = claudeInfo.Usage.PromptTokens + claudeInfo.Usage.CompletionTokens
- }
- // 判断是否完整
- claudeInfo.Done = true
- } else if claudeResponse.Type == "content_block_start" || claudeResponse.Type == "content_block_stop" {
- } else {
- return false
- }
- if oaiResponse != nil {
- oaiResponse.Id = claudeInfo.ResponseId
- oaiResponse.Created = claudeInfo.Created
- oaiResponse.Model = claudeInfo.Model
- }
- return true
- }
- func HandleStreamResponseData(c *gin.Context, info *relaycommon.RelayInfo, claudeInfo *ClaudeResponseInfo, data string) *types.NewAPIError {
- var claudeResponse dto.ClaudeResponse
- err := common.UnmarshalJsonStr(data, &claudeResponse)
- if err != nil {
- common.SysLog("error unmarshalling stream response: " + err.Error())
- return types.NewError(err, types.ErrorCodeBadResponseBody)
- }
- if claudeError := claudeResponse.GetClaudeError(); claudeError != nil && claudeError.Type != "" {
- return types.WithClaudeError(*claudeError, http.StatusInternalServerError)
- }
- if claudeResponse.StopReason != "" {
- maybeMarkClaudeRefusal(c, claudeResponse.StopReason)
- }
- if claudeResponse.Delta != nil && claudeResponse.Delta.StopReason != nil {
- maybeMarkClaudeRefusal(c, *claudeResponse.Delta.StopReason)
- }
- if info.RelayFormat == types.RelayFormatClaude {
- FormatClaudeResponseInfo(&claudeResponse, nil, claudeInfo)
- if claudeResponse.Type == "message_start" {
- // message_start, 获取usage
- if claudeResponse.Message != nil {
- info.UpstreamModelName = claudeResponse.Message.Model
- }
- } else if claudeResponse.Type == "message_delta" {
- // 确保 message_delta 的 usage 包含完整的 input_tokens 和 cache 相关字段
- // 解决 AWS Bedrock 等上游返回的 message_delta 缺少这些字段的问题
- if !shouldSkipClaudeMessageDeltaUsagePatch(info) {
- data = patchClaudeMessageDeltaUsageData(data, buildMessageDeltaPatchUsage(&claudeResponse, claudeInfo))
- }
- }
- helper.ClaudeChunkData(c, claudeResponse, data)
- } else if info.RelayFormat == types.RelayFormatOpenAI {
- response := StreamResponseClaude2OpenAI(&claudeResponse, claudeInfo)
- if response == nil {
- return nil
- }
- if !FormatClaudeResponseInfo(&claudeResponse, response, claudeInfo) {
- return nil
- }
- err = helper.ObjectData(c, response)
- if err != nil {
- logger.LogError(c, "send_stream_response_failed: "+err.Error())
- }
- }
- return nil
- }
- func HandleStreamFinalResponse(c *gin.Context, info *relaycommon.RelayInfo, claudeInfo *ClaudeResponseInfo) {
- if claudeInfo.Usage.PromptTokens == 0 {
- //上游出错
- }
- if claudeInfo.Usage.CompletionTokens == 0 || !claudeInfo.Done {
- if common.DebugEnabled {
- common.SysLog("claude response usage is not complete, maybe upstream error")
- }
- claudeInfo.Usage = service.ResponseText2Usage(c, claudeInfo.ResponseText.String(), info.UpstreamModelName, claudeInfo.Usage.PromptTokens)
- }
- if info.RelayFormat == types.RelayFormatClaude {
- //
- } else if info.RelayFormat == types.RelayFormatOpenAI {
- if info.ShouldIncludeUsage {
- response := helper.GenerateFinalUsageResponse(claudeInfo.ResponseId, claudeInfo.Created, info.UpstreamModelName, *claudeInfo.Usage)
- err := helper.ObjectData(c, response)
- if err != nil {
- common.SysLog("send final response failed: " + err.Error())
- }
- }
- helper.Done(c)
- }
- }
- func ClaudeStreamHandler(c *gin.Context, resp *http.Response, info *relaycommon.RelayInfo) (*dto.Usage, *types.NewAPIError) {
- claudeInfo := &ClaudeResponseInfo{
- ResponseId: helper.GetResponseID(c),
- Created: common.GetTimestamp(),
- Model: info.UpstreamModelName,
- ResponseText: strings.Builder{},
- Usage: &dto.Usage{},
- }
- var err *types.NewAPIError
- helper.StreamScannerHandler(c, resp, info, func(data string) bool {
- err = HandleStreamResponseData(c, info, claudeInfo, data)
- if err != nil {
- return false
- }
- return true
- })
- if err != nil {
- return nil, err
- }
- HandleStreamFinalResponse(c, info, claudeInfo)
- return claudeInfo.Usage, nil
- }
- func HandleClaudeResponseData(c *gin.Context, info *relaycommon.RelayInfo, claudeInfo *ClaudeResponseInfo, httpResp *http.Response, data []byte) *types.NewAPIError {
- var claudeResponse dto.ClaudeResponse
- err := common.Unmarshal(data, &claudeResponse)
- if err != nil {
- return types.NewError(err, types.ErrorCodeBadResponseBody)
- }
- if claudeError := claudeResponse.GetClaudeError(); claudeError != nil && claudeError.Type != "" {
- return types.WithClaudeError(*claudeError, http.StatusInternalServerError)
- }
- maybeMarkClaudeRefusal(c, claudeResponse.StopReason)
- if claudeInfo.Usage == nil {
- claudeInfo.Usage = &dto.Usage{}
- }
- if claudeResponse.Usage != nil {
- claudeInfo.Usage.PromptTokens = claudeResponse.Usage.InputTokens
- claudeInfo.Usage.CompletionTokens = claudeResponse.Usage.OutputTokens
- claudeInfo.Usage.TotalTokens = claudeResponse.Usage.InputTokens + claudeResponse.Usage.OutputTokens
- claudeInfo.Usage.PromptTokensDetails.CachedTokens = claudeResponse.Usage.CacheReadInputTokens
- claudeInfo.Usage.PromptTokensDetails.CachedCreationTokens = claudeResponse.Usage.CacheCreationInputTokens
- claudeInfo.Usage.ClaudeCacheCreation5mTokens = claudeResponse.Usage.GetCacheCreation5mTokens()
- claudeInfo.Usage.ClaudeCacheCreation1hTokens = claudeResponse.Usage.GetCacheCreation1hTokens()
- }
- var responseData []byte
- switch info.RelayFormat {
- case types.RelayFormatOpenAI:
- openaiResponse := ResponseClaude2OpenAI(&claudeResponse)
- openaiResponse.Usage = *claudeInfo.Usage
- responseData, err = json.Marshal(openaiResponse)
- if err != nil {
- return types.NewError(err, types.ErrorCodeBadResponseBody)
- }
- case types.RelayFormatClaude:
- responseData = data
- }
- if claudeResponse.Usage != nil && claudeResponse.Usage.ServerToolUse != nil && claudeResponse.Usage.ServerToolUse.WebSearchRequests > 0 {
- c.Set("claude_web_search_requests", claudeResponse.Usage.ServerToolUse.WebSearchRequests)
- }
- service.IOCopyBytesGracefully(c, httpResp, responseData)
- return nil
- }
- func ClaudeHandler(c *gin.Context, resp *http.Response, info *relaycommon.RelayInfo) (*dto.Usage, *types.NewAPIError) {
- defer service.CloseResponseBodyGracefully(resp)
- claudeInfo := &ClaudeResponseInfo{
- ResponseId: helper.GetResponseID(c),
- Created: common.GetTimestamp(),
- Model: info.UpstreamModelName,
- ResponseText: strings.Builder{},
- Usage: &dto.Usage{},
- }
- responseBody, err := io.ReadAll(resp.Body)
- if err != nil {
- return nil, types.NewError(err, types.ErrorCodeBadResponseBody)
- }
- if common.DebugEnabled {
- println("responseBody: ", string(responseBody))
- }
- handleErr := HandleClaudeResponseData(c, info, claudeInfo, resp, responseBody)
- if handleErr != nil {
- return nil, handleErr
- }
- return claudeInfo.Usage, nil
- }
- func mapToolChoice(toolChoice any, parallelToolCalls *bool) *dto.ClaudeToolChoice {
- var claudeToolChoice *dto.ClaudeToolChoice
- // 处理 tool_choice 字符串值
- if toolChoiceStr, ok := toolChoice.(string); ok {
- switch toolChoiceStr {
- case "auto":
- claudeToolChoice = &dto.ClaudeToolChoice{
- Type: "auto",
- }
- case "required":
- claudeToolChoice = &dto.ClaudeToolChoice{
- Type: "any",
- }
- case "none":
- claudeToolChoice = &dto.ClaudeToolChoice{
- Type: "none",
- }
- }
- } else if toolChoiceMap, ok := toolChoice.(map[string]interface{}); ok {
- // 处理 tool_choice 对象值
- if function, ok := toolChoiceMap["function"].(map[string]interface{}); ok {
- if toolName, ok := function["name"].(string); ok {
- claudeToolChoice = &dto.ClaudeToolChoice{
- Type: "tool",
- Name: toolName,
- }
- }
- }
- }
- // 处理 parallel_tool_calls
- if parallelToolCalls != nil {
- if claudeToolChoice == nil {
- // 如果没有 tool_choice,但有 parallel_tool_calls,创建默认的 auto 类型
- claudeToolChoice = &dto.ClaudeToolChoice{
- Type: "auto",
- }
- }
- // Anthropic schema: tool_choice.type=none does not accept extra fields.
- // When tools are disabled, parallel_tool_calls is irrelevant, so we drop it.
- if claudeToolChoice.Type != "none" {
- // 如果 parallel_tool_calls 为 true,则 disable_parallel_tool_use 为 false
- claudeToolChoice.DisableParallelToolUse = !*parallelToolCalls
- }
- }
- return claudeToolChoice
- }
|