2025-03-13 19:32:08 +08:00
|
|
|
|
package openai
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"encoding/json"
|
|
|
|
|
|
"strings"
|
|
|
|
|
|
|
2025-10-11 15:30:09 +08:00
|
|
|
|
"github.com/QuantumNous/new-api/common"
|
|
|
|
|
|
"github.com/QuantumNous/new-api/dto"
|
|
|
|
|
|
"github.com/QuantumNous/new-api/logger"
|
|
|
|
|
|
relaycommon "github.com/QuantumNous/new-api/relay/common"
|
|
|
|
|
|
relayconstant "github.com/QuantumNous/new-api/relay/constant"
|
|
|
|
|
|
"github.com/QuantumNous/new-api/relay/helper"
|
|
|
|
|
|
"github.com/QuantumNous/new-api/service"
|
|
|
|
|
|
"github.com/QuantumNous/new-api/types"
|
|
|
|
|
|
|
2025-08-17 15:30:31 +08:00
|
|
|
|
"github.com/samber/lo"
|
|
|
|
|
|
|
2025-03-13 19:32:08 +08:00
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
// 辅助函数
|
2025-07-26 13:31:33 +08:00
|
|
|
|
func HandleStreamFormat(c *gin.Context, info *relaycommon.RelayInfo, data string, forceFormat bool, thinkToContent bool) error {
|
2025-03-13 19:32:08 +08:00
|
|
|
|
info.SendResponseCount++
|
2025-08-01 22:23:35 +08:00
|
|
|
|
|
2025-03-13 19:32:08 +08:00
|
|
|
|
switch info.RelayFormat {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatOpenAI:
|
2025-03-13 19:32:08 +08:00
|
|
|
|
return sendStreamData(c, info, data, forceFormat, thinkToContent)
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatClaude:
|
2025-03-13 19:32:08 +08:00
|
|
|
|
return handleClaudeFormat(c, data, info)
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatGemini:
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return handleGeminiFormat(c, data, info)
|
2025-03-13 19:32:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func handleClaudeFormat(c *gin.Context, data string, info *relaycommon.RelayInfo) error {
|
|
|
|
|
|
var streamResponse dto.ChatCompletionsStreamResponse
|
2025-07-23 20:59:56 +08:00
|
|
|
|
if err := common.Unmarshal(common.StringToByteSlice(data), &streamResponse); err != nil {
|
2025-03-13 19:32:08 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-11 18:28:50 +08:00
|
|
|
|
if streamResponse.Usage != nil {
|
|
|
|
|
|
info.ClaudeConvertInfo.Usage = streamResponse.Usage
|
|
|
|
|
|
}
|
2025-03-13 19:32:08 +08:00
|
|
|
|
claudeResponses := service.StreamResponseOpenAI2Claude(&streamResponse, info)
|
|
|
|
|
|
for _, resp := range claudeResponses {
|
|
|
|
|
|
helper.ClaudeData(c, *resp)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-01 22:23:35 +08:00
|
|
|
|
func handleGeminiFormat(c *gin.Context, data string, info *relaycommon.RelayInfo) error {
|
|
|
|
|
|
var streamResponse dto.ChatCompletionsStreamResponse
|
|
|
|
|
|
if err := common.Unmarshal(common.StringToByteSlice(data), &streamResponse); err != nil {
|
2025-08-14 20:05:06 +08:00
|
|
|
|
logger.LogError(c, "failed to unmarshal stream response: "+err.Error())
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
geminiResponse := service.StreamResponseOpenAI2Gemini(&streamResponse, info)
|
|
|
|
|
|
|
|
|
|
|
|
// 如果返回 nil,表示没有实际内容,跳过发送
|
|
|
|
|
|
if geminiResponse == nil {
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
geminiResponseStr, err := common.Marshal(geminiResponse)
|
|
|
|
|
|
if err != nil {
|
2025-08-14 20:05:06 +08:00
|
|
|
|
logger.LogError(c, "failed to marshal gemini response: "+err.Error())
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// send gemini format response
|
|
|
|
|
|
c.Render(-1, common.CustomEvent{Data: "data: " + string(geminiResponseStr)})
|
2025-08-17 15:30:31 +08:00
|
|
|
|
_ = helper.FlushWriter(c)
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-11 23:31:32 +08:00
|
|
|
|
func ProcessStreamResponse(streamResponse dto.ChatCompletionsStreamResponse, responseTextBuilder *strings.Builder, toolCount *int) error {
|
2025-03-13 19:32:08 +08:00
|
|
|
|
for _, choice := range streamResponse.Choices {
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Delta.GetContentString())
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Delta.GetReasoningContent())
|
|
|
|
|
|
if choice.Delta.ToolCalls != nil {
|
|
|
|
|
|
if len(choice.Delta.ToolCalls) > *toolCount {
|
|
|
|
|
|
*toolCount = len(choice.Delta.ToolCalls)
|
|
|
|
|
|
}
|
|
|
|
|
|
for _, tool := range choice.Delta.ToolCalls {
|
|
|
|
|
|
responseTextBuilder.WriteString(tool.Function.Name)
|
|
|
|
|
|
responseTextBuilder.WriteString(tool.Function.Arguments)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func processTokens(relayMode int, streamItems []string, responseTextBuilder *strings.Builder, toolCount *int) error {
|
|
|
|
|
|
streamResp := "[" + strings.Join(streamItems, ",") + "]"
|
|
|
|
|
|
|
|
|
|
|
|
switch relayMode {
|
|
|
|
|
|
case relayconstant.RelayModeChatCompletions:
|
|
|
|
|
|
return processChatCompletions(streamResp, streamItems, responseTextBuilder, toolCount)
|
|
|
|
|
|
case relayconstant.RelayModeCompletions:
|
|
|
|
|
|
return processCompletions(streamResp, streamItems, responseTextBuilder)
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func processChatCompletions(streamResp string, streamItems []string, responseTextBuilder *strings.Builder, toolCount *int) error {
|
|
|
|
|
|
var streamResponses []dto.ChatCompletionsStreamResponse
|
|
|
|
|
|
if err := json.Unmarshal(common.StringToByteSlice(streamResp), &streamResponses); err != nil {
|
|
|
|
|
|
// 一次性解析失败,逐个解析
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error unmarshalling stream response: " + err.Error())
|
2025-03-13 19:32:08 +08:00
|
|
|
|
for _, item := range streamItems {
|
2025-04-11 23:31:32 +08:00
|
|
|
|
var streamResponse dto.ChatCompletionsStreamResponse
|
|
|
|
|
|
if err := json.Unmarshal(common.StringToByteSlice(item), &streamResponse); err != nil {
|
|
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
if err := ProcessStreamResponse(streamResponse, responseTextBuilder, toolCount); err != nil {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error processing stream response: " + err.Error())
|
2025-03-13 19:32:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 批量处理所有响应
|
|
|
|
|
|
for _, streamResponse := range streamResponses {
|
|
|
|
|
|
for _, choice := range streamResponse.Choices {
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Delta.GetContentString())
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Delta.GetReasoningContent())
|
|
|
|
|
|
if choice.Delta.ToolCalls != nil {
|
|
|
|
|
|
if len(choice.Delta.ToolCalls) > *toolCount {
|
|
|
|
|
|
*toolCount = len(choice.Delta.ToolCalls)
|
|
|
|
|
|
}
|
|
|
|
|
|
for _, tool := range choice.Delta.ToolCalls {
|
|
|
|
|
|
responseTextBuilder.WriteString(tool.Function.Name)
|
|
|
|
|
|
responseTextBuilder.WriteString(tool.Function.Arguments)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func processCompletions(streamResp string, streamItems []string, responseTextBuilder *strings.Builder) error {
|
|
|
|
|
|
var streamResponses []dto.CompletionsStreamResponse
|
|
|
|
|
|
if err := json.Unmarshal(common.StringToByteSlice(streamResp), &streamResponses); err != nil {
|
|
|
|
|
|
// 一次性解析失败,逐个解析
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error unmarshalling stream response: " + err.Error())
|
2025-03-13 19:32:08 +08:00
|
|
|
|
for _, item := range streamItems {
|
|
|
|
|
|
var streamResponse dto.CompletionsStreamResponse
|
|
|
|
|
|
if err := json.Unmarshal(common.StringToByteSlice(item), &streamResponse); err != nil {
|
|
|
|
|
|
continue
|
|
|
|
|
|
}
|
|
|
|
|
|
for _, choice := range streamResponse.Choices {
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Text)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 批量处理所有响应
|
|
|
|
|
|
for _, streamResponse := range streamResponses {
|
|
|
|
|
|
for _, choice := range streamResponse.Choices {
|
|
|
|
|
|
responseTextBuilder.WriteString(choice.Text)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func handleLastResponse(lastStreamData string, responseId *string, createAt *int64,
|
|
|
|
|
|
systemFingerprint *string, model *string, usage **dto.Usage,
|
|
|
|
|
|
containStreamUsage *bool, info *relaycommon.RelayInfo,
|
|
|
|
|
|
shouldSendLastResp *bool) error {
|
|
|
|
|
|
|
|
|
|
|
|
var lastStreamResponse dto.ChatCompletionsStreamResponse
|
2025-12-12 17:59:21 +08:00
|
|
|
|
if err := common.Unmarshal(common.StringToByteSlice(lastStreamData), &lastStreamResponse); err != nil {
|
2025-03-13 19:32:08 +08:00
|
|
|
|
return err
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
*responseId = lastStreamResponse.Id
|
|
|
|
|
|
*createAt = lastStreamResponse.Created
|
|
|
|
|
|
*systemFingerprint = lastStreamResponse.GetSystemFingerprint()
|
|
|
|
|
|
*model = lastStreamResponse.Model
|
|
|
|
|
|
|
|
|
|
|
|
if service.ValidUsage(lastStreamResponse.Usage) {
|
|
|
|
|
|
*containStreamUsage = true
|
|
|
|
|
|
*usage = lastStreamResponse.Usage
|
|
|
|
|
|
if !info.ShouldIncludeUsage {
|
2025-08-08 08:36:56 +08:00
|
|
|
|
*shouldSendLastResp = lo.SomeBy(lastStreamResponse.Choices, func(choice dto.ChatCompletionsStreamResponseChoice) bool {
|
|
|
|
|
|
return choice.Delta.GetContentString() != "" || choice.Delta.GetReasoningContent() != ""
|
|
|
|
|
|
})
|
2025-03-13 19:32:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-26 13:31:33 +08:00
|
|
|
|
func HandleFinalResponse(c *gin.Context, info *relaycommon.RelayInfo, lastStreamData string,
|
2025-03-13 19:32:08 +08:00
|
|
|
|
responseId string, createAt int64, model string, systemFingerprint string,
|
|
|
|
|
|
usage *dto.Usage, containStreamUsage bool) {
|
|
|
|
|
|
|
|
|
|
|
|
switch info.RelayFormat {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatOpenAI:
|
2025-03-13 19:32:08 +08:00
|
|
|
|
if info.ShouldIncludeUsage && !containStreamUsage {
|
|
|
|
|
|
response := helper.GenerateFinalUsageResponse(responseId, createAt, model, *usage)
|
|
|
|
|
|
response.SetSystemFingerprint(systemFingerprint)
|
|
|
|
|
|
helper.ObjectData(c, response)
|
|
|
|
|
|
}
|
|
|
|
|
|
helper.Done(c)
|
|
|
|
|
|
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatClaude:
|
2025-03-13 19:32:08 +08:00
|
|
|
|
var streamResponse dto.ChatCompletionsStreamResponse
|
2025-07-23 20:59:56 +08:00
|
|
|
|
if err := common.Unmarshal(common.StringToByteSlice(lastStreamData), &streamResponse); err != nil {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error unmarshalling stream response: " + err.Error())
|
2025-03-13 19:32:08 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-11 18:28:50 +08:00
|
|
|
|
info.ClaudeConvertInfo.Usage = usage
|
2025-03-13 19:32:08 +08:00
|
|
|
|
|
|
|
|
|
|
claudeResponses := service.StreamResponseOpenAI2Claude(&streamResponse, info)
|
|
|
|
|
|
for _, resp := range claudeResponses {
|
2025-07-23 20:59:56 +08:00
|
|
|
|
_ = helper.ClaudeData(c, *resp)
|
2025-03-13 19:32:08 +08:00
|
|
|
|
}
|
2025-12-29 19:41:15 +08:00
|
|
|
|
info.ClaudeConvertInfo.Done = true
|
2025-08-01 22:23:35 +08:00
|
|
|
|
|
2025-08-14 21:10:04 +08:00
|
|
|
|
case types.RelayFormatGemini:
|
2025-08-01 22:23:35 +08:00
|
|
|
|
var streamResponse dto.ChatCompletionsStreamResponse
|
|
|
|
|
|
if err := common.Unmarshal(common.StringToByteSlice(lastStreamData), &streamResponse); err != nil {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error unmarshalling stream response: " + err.Error())
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 这里处理的是 openai 最后一个流响应,其 delta 为空,有 finish_reason 字段
|
|
|
|
|
|
// 因此相比较于 google 官方的流响应,由 openai 转换而来会多一个 parts 为空,finishReason 为 STOP 的响应
|
|
|
|
|
|
// 而包含最后一段文本输出的响应(倒数第二个)的 finishReason 为 null
|
|
|
|
|
|
// 暂不知是否有程序会不兼容。
|
|
|
|
|
|
|
|
|
|
|
|
geminiResponse := service.StreamResponseOpenAI2Gemini(&streamResponse, info)
|
|
|
|
|
|
|
|
|
|
|
|
// openai 流响应开头的空数据
|
|
|
|
|
|
if geminiResponse == nil {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
geminiResponseStr, err := common.Marshal(geminiResponse)
|
|
|
|
|
|
if err != nil {
|
2025-08-14 21:10:04 +08:00
|
|
|
|
common.SysLog("error marshalling gemini response: " + err.Error())
|
2025-08-01 22:23:35 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 发送最终的 Gemini 响应
|
|
|
|
|
|
c.Render(-1, common.CustomEvent{Data: "data: " + string(geminiResponseStr)})
|
2025-08-17 15:30:31 +08:00
|
|
|
|
_ = helper.FlushWriter(c)
|
2025-03-13 19:32:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-05-05 00:40:16 +08:00
|
|
|
|
|
|
|
|
|
|
func sendResponsesStreamData(c *gin.Context, streamResponse dto.ResponsesStreamResponse, data string) {
|
|
|
|
|
|
if data == "" {
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
helper.ResponseChunkData(c, streamResponse, data)
|
|
|
|
|
|
}
|