convert.go 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923
  1. package service
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "strings"
  6. "github.com/QuantumNous/new-api/common"
  7. "github.com/QuantumNous/new-api/constant"
  8. "github.com/QuantumNous/new-api/dto"
  9. "github.com/QuantumNous/new-api/relay/channel/openrouter"
  10. relaycommon "github.com/QuantumNous/new-api/relay/common"
  11. "github.com/QuantumNous/new-api/relay/reasonmap"
  12. )
  13. func ClaudeToOpenAIRequest(claudeRequest dto.ClaudeRequest, info *relaycommon.RelayInfo) (*dto.GeneralOpenAIRequest, error) {
  14. openAIRequest := dto.GeneralOpenAIRequest{
  15. Model: claudeRequest.Model,
  16. MaxTokens: claudeRequest.MaxTokens,
  17. Temperature: claudeRequest.Temperature,
  18. TopP: claudeRequest.TopP,
  19. Stream: claudeRequest.Stream,
  20. }
  21. isOpenRouter := info.ChannelType == constant.ChannelTypeOpenRouter
  22. if claudeRequest.Thinking != nil && claudeRequest.Thinking.Type == "enabled" {
  23. if isOpenRouter {
  24. reasoning := openrouter.RequestReasoning{
  25. MaxTokens: claudeRequest.Thinking.GetBudgetTokens(),
  26. }
  27. reasoningJSON, err := json.Marshal(reasoning)
  28. if err != nil {
  29. return nil, fmt.Errorf("failed to marshal reasoning: %w", err)
  30. }
  31. openAIRequest.Reasoning = reasoningJSON
  32. } else {
  33. thinkingSuffix := "-thinking"
  34. if strings.HasSuffix(info.OriginModelName, thinkingSuffix) &&
  35. !strings.HasSuffix(openAIRequest.Model, thinkingSuffix) {
  36. openAIRequest.Model = openAIRequest.Model + thinkingSuffix
  37. }
  38. }
  39. }
  40. // Convert stop sequences
  41. if len(claudeRequest.StopSequences) == 1 {
  42. openAIRequest.Stop = claudeRequest.StopSequences[0]
  43. } else if len(claudeRequest.StopSequences) > 1 {
  44. openAIRequest.Stop = claudeRequest.StopSequences
  45. }
  46. // Convert tools
  47. tools, _ := common.Any2Type[[]dto.Tool](claudeRequest.Tools)
  48. openAITools := make([]dto.ToolCallRequest, 0)
  49. for _, claudeTool := range tools {
  50. openAITool := dto.ToolCallRequest{
  51. Type: "function",
  52. Function: dto.FunctionRequest{
  53. Name: claudeTool.Name,
  54. Description: claudeTool.Description,
  55. Parameters: claudeTool.InputSchema,
  56. },
  57. }
  58. openAITools = append(openAITools, openAITool)
  59. }
  60. openAIRequest.Tools = openAITools
  61. // Convert messages
  62. openAIMessages := make([]dto.Message, 0)
  63. // Add system message if present
  64. if claudeRequest.System != nil {
  65. if claudeRequest.IsStringSystem() && claudeRequest.GetStringSystem() != "" {
  66. openAIMessage := dto.Message{
  67. Role: "system",
  68. }
  69. openAIMessage.SetStringContent(claudeRequest.GetStringSystem())
  70. openAIMessages = append(openAIMessages, openAIMessage)
  71. } else {
  72. systems := claudeRequest.ParseSystem()
  73. if len(systems) > 0 {
  74. openAIMessage := dto.Message{
  75. Role: "system",
  76. }
  77. isOpenRouterClaude := isOpenRouter && strings.HasPrefix(info.UpstreamModelName, "anthropic/claude")
  78. if isOpenRouterClaude {
  79. systemMediaMessages := make([]dto.MediaContent, 0, len(systems))
  80. for _, system := range systems {
  81. message := dto.MediaContent{
  82. Type: "text",
  83. Text: system.GetText(),
  84. CacheControl: system.CacheControl,
  85. }
  86. systemMediaMessages = append(systemMediaMessages, message)
  87. }
  88. openAIMessage.SetMediaContent(systemMediaMessages)
  89. } else {
  90. systemStr := ""
  91. for _, system := range systems {
  92. if system.Text != nil {
  93. systemStr += *system.Text
  94. }
  95. }
  96. openAIMessage.SetStringContent(systemStr)
  97. }
  98. openAIMessages = append(openAIMessages, openAIMessage)
  99. }
  100. }
  101. }
  102. for _, claudeMessage := range claudeRequest.Messages {
  103. openAIMessage := dto.Message{
  104. Role: claudeMessage.Role,
  105. }
  106. //log.Printf("claudeMessage.Content: %v", claudeMessage.Content)
  107. if claudeMessage.IsStringContent() {
  108. openAIMessage.SetStringContent(claudeMessage.GetStringContent())
  109. } else {
  110. content, err := claudeMessage.ParseContent()
  111. if err != nil {
  112. return nil, err
  113. }
  114. contents := content
  115. var toolCalls []dto.ToolCallRequest
  116. mediaMessages := make([]dto.MediaContent, 0, len(contents))
  117. for _, mediaMsg := range contents {
  118. switch mediaMsg.Type {
  119. case "text":
  120. message := dto.MediaContent{
  121. Type: "text",
  122. Text: mediaMsg.GetText(),
  123. CacheControl: mediaMsg.CacheControl,
  124. }
  125. mediaMessages = append(mediaMessages, message)
  126. case "image":
  127. // Handle image conversion (base64 to URL or keep as is)
  128. imageData := fmt.Sprintf("data:%s;base64,%s", mediaMsg.Source.MediaType, mediaMsg.Source.Data)
  129. //textContent += fmt.Sprintf("[Image: %s]", imageData)
  130. mediaMessage := dto.MediaContent{
  131. Type: "image_url",
  132. ImageUrl: &dto.MessageImageUrl{Url: imageData},
  133. }
  134. mediaMessages = append(mediaMessages, mediaMessage)
  135. case "tool_use":
  136. toolCall := dto.ToolCallRequest{
  137. ID: mediaMsg.Id,
  138. Type: "function",
  139. Function: dto.FunctionRequest{
  140. Name: mediaMsg.Name,
  141. Arguments: toJSONString(mediaMsg.Input),
  142. },
  143. }
  144. toolCalls = append(toolCalls, toolCall)
  145. case "tool_result":
  146. // Add tool result as a separate message
  147. toolName := mediaMsg.Name
  148. if toolName == "" {
  149. toolName = claudeRequest.SearchToolNameByToolCallId(mediaMsg.ToolUseId)
  150. }
  151. oaiToolMessage := dto.Message{
  152. Role: "tool",
  153. Name: &toolName,
  154. ToolCallId: mediaMsg.ToolUseId,
  155. }
  156. //oaiToolMessage.SetStringContent(*mediaMsg.GetMediaContent().Text)
  157. if mediaMsg.IsStringContent() {
  158. oaiToolMessage.SetStringContent(mediaMsg.GetStringContent())
  159. } else {
  160. mediaContents := mediaMsg.ParseMediaContent()
  161. encodeJson, _ := common.Marshal(mediaContents)
  162. oaiToolMessage.SetStringContent(string(encodeJson))
  163. }
  164. openAIMessages = append(openAIMessages, oaiToolMessage)
  165. }
  166. }
  167. if len(toolCalls) > 0 {
  168. openAIMessage.SetToolCalls(toolCalls)
  169. }
  170. if len(mediaMessages) > 0 && len(toolCalls) == 0 {
  171. openAIMessage.SetMediaContent(mediaMessages)
  172. }
  173. }
  174. if len(openAIMessage.ParseContent()) > 0 || len(openAIMessage.ToolCalls) > 0 {
  175. openAIMessages = append(openAIMessages, openAIMessage)
  176. }
  177. }
  178. openAIRequest.Messages = openAIMessages
  179. return &openAIRequest, nil
  180. }
  181. func generateStopBlock(index int) *dto.ClaudeResponse {
  182. return &dto.ClaudeResponse{
  183. Type: "content_block_stop",
  184. Index: common.GetPointer[int](index),
  185. }
  186. }
  187. func StreamResponseOpenAI2Claude(openAIResponse *dto.ChatCompletionsStreamResponse, info *relaycommon.RelayInfo) []*dto.ClaudeResponse {
  188. if info.ClaudeConvertInfo.Done {
  189. return nil
  190. }
  191. var claudeResponses []*dto.ClaudeResponse
  192. if info.SendResponseCount == 1 {
  193. msg := &dto.ClaudeMediaMessage{
  194. Id: openAIResponse.Id,
  195. Model: openAIResponse.Model,
  196. Type: "message",
  197. Role: "assistant",
  198. Usage: &dto.ClaudeUsage{
  199. InputTokens: info.GetEstimatePromptTokens(),
  200. OutputTokens: 0,
  201. },
  202. }
  203. msg.SetContent(make([]any, 0))
  204. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  205. Type: "message_start",
  206. Message: msg,
  207. })
  208. //claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  209. // Type: "ping",
  210. //})
  211. if openAIResponse.IsToolCall() {
  212. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeTools
  213. var toolCall dto.ToolCallResponse
  214. if len(openAIResponse.Choices) > 0 && len(openAIResponse.Choices[0].Delta.ToolCalls) > 0 {
  215. toolCall = openAIResponse.Choices[0].Delta.ToolCalls[0]
  216. } else {
  217. first := openAIResponse.GetFirstToolCall()
  218. if first != nil {
  219. toolCall = *first
  220. } else {
  221. toolCall = dto.ToolCallResponse{}
  222. }
  223. }
  224. resp := &dto.ClaudeResponse{
  225. Type: "content_block_start",
  226. ContentBlock: &dto.ClaudeMediaMessage{
  227. Id: toolCall.ID,
  228. Type: "tool_use",
  229. Name: toolCall.Function.Name,
  230. Input: map[string]interface{}{},
  231. },
  232. }
  233. resp.SetIndex(0)
  234. claudeResponses = append(claudeResponses, resp)
  235. // 首块包含工具 delta,则追加 input_json_delta
  236. if toolCall.Function.Arguments != "" {
  237. idx := 0
  238. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  239. Index: &idx,
  240. Type: "content_block_delta",
  241. Delta: &dto.ClaudeMediaMessage{
  242. Type: "input_json_delta",
  243. PartialJson: &toolCall.Function.Arguments,
  244. },
  245. })
  246. }
  247. } else {
  248. }
  249. // 判断首个响应是否存在内容(非标准的 OpenAI 响应)
  250. if len(openAIResponse.Choices) > 0 {
  251. reasoning := openAIResponse.Choices[0].Delta.GetReasoningContent()
  252. content := openAIResponse.Choices[0].Delta.GetContentString()
  253. if reasoning != "" {
  254. if info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeTools || info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeText {
  255. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  256. info.ClaudeConvertInfo.Index++
  257. }
  258. idx := info.ClaudeConvertInfo.Index
  259. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  260. Index: &idx,
  261. Type: "content_block_start",
  262. ContentBlock: &dto.ClaudeMediaMessage{
  263. Type: "thinking",
  264. Thinking: common.GetPointer[string](""),
  265. },
  266. })
  267. idx2 := idx
  268. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  269. Index: &idx2,
  270. Type: "content_block_delta",
  271. Delta: &dto.ClaudeMediaMessage{
  272. Type: "thinking_delta",
  273. Thinking: &reasoning,
  274. },
  275. })
  276. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeThinking
  277. } else if content != "" {
  278. if info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeTools || info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeThinking {
  279. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  280. info.ClaudeConvertInfo.Index++
  281. }
  282. idx := info.ClaudeConvertInfo.Index
  283. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  284. Index: &idx,
  285. Type: "content_block_start",
  286. ContentBlock: &dto.ClaudeMediaMessage{
  287. Type: "text",
  288. Text: common.GetPointer[string](""),
  289. },
  290. })
  291. idx2 := idx
  292. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  293. Index: &idx2,
  294. Type: "content_block_delta",
  295. Delta: &dto.ClaudeMediaMessage{
  296. Type: "text_delta",
  297. Text: common.GetPointer[string](content),
  298. },
  299. })
  300. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeText
  301. }
  302. }
  303. // 如果首块就带 finish_reason,需要立即发送停止块
  304. if len(openAIResponse.Choices) > 0 && openAIResponse.Choices[0].FinishReason != nil && *openAIResponse.Choices[0].FinishReason != "" {
  305. info.FinishReason = *openAIResponse.Choices[0].FinishReason
  306. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  307. oaiUsage := openAIResponse.Usage
  308. if oaiUsage == nil {
  309. oaiUsage = info.ClaudeConvertInfo.Usage
  310. }
  311. if oaiUsage != nil {
  312. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  313. Type: "message_delta",
  314. Usage: &dto.ClaudeUsage{
  315. InputTokens: oaiUsage.PromptTokens,
  316. OutputTokens: oaiUsage.CompletionTokens,
  317. CacheCreationInputTokens: oaiUsage.PromptTokensDetails.CachedCreationTokens,
  318. CacheReadInputTokens: oaiUsage.PromptTokensDetails.CachedTokens,
  319. },
  320. Delta: &dto.ClaudeMediaMessage{
  321. StopReason: common.GetPointer[string](stopReasonOpenAI2Claude(info.FinishReason)),
  322. },
  323. })
  324. }
  325. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  326. Type: "message_stop",
  327. })
  328. info.ClaudeConvertInfo.Done = true
  329. }
  330. return claudeResponses
  331. }
  332. if len(openAIResponse.Choices) == 0 {
  333. // no choices
  334. // 可能为非标准的 OpenAI 响应,判断是否已经完成
  335. if info.ClaudeConvertInfo.Done {
  336. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  337. oaiUsage := info.ClaudeConvertInfo.Usage
  338. if oaiUsage != nil {
  339. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  340. Type: "message_delta",
  341. Usage: &dto.ClaudeUsage{
  342. InputTokens: oaiUsage.PromptTokens,
  343. OutputTokens: oaiUsage.CompletionTokens,
  344. CacheCreationInputTokens: oaiUsage.PromptTokensDetails.CachedCreationTokens,
  345. CacheReadInputTokens: oaiUsage.PromptTokensDetails.CachedTokens,
  346. },
  347. Delta: &dto.ClaudeMediaMessage{
  348. StopReason: common.GetPointer[string](stopReasonOpenAI2Claude(info.FinishReason)),
  349. },
  350. })
  351. }
  352. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  353. Type: "message_stop",
  354. })
  355. }
  356. return claudeResponses
  357. } else {
  358. chosenChoice := openAIResponse.Choices[0]
  359. doneChunk := chosenChoice.FinishReason != nil && *chosenChoice.FinishReason != ""
  360. if doneChunk {
  361. info.FinishReason = *chosenChoice.FinishReason
  362. }
  363. var claudeResponse dto.ClaudeResponse
  364. var isEmpty bool
  365. claudeResponse.Type = "content_block_delta"
  366. if len(chosenChoice.Delta.ToolCalls) > 0 {
  367. toolCalls := chosenChoice.Delta.ToolCalls
  368. if info.ClaudeConvertInfo.LastMessagesType != relaycommon.LastMessageTypeTools {
  369. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  370. info.ClaudeConvertInfo.Index++
  371. }
  372. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeTools
  373. for i, toolCall := range toolCalls {
  374. blockIndex := info.ClaudeConvertInfo.Index
  375. if toolCall.Index != nil {
  376. blockIndex = *toolCall.Index
  377. } else if len(toolCalls) > 1 {
  378. blockIndex = info.ClaudeConvertInfo.Index + i
  379. }
  380. idx := blockIndex
  381. if toolCall.Function.Name != "" {
  382. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  383. Index: &idx,
  384. Type: "content_block_start",
  385. ContentBlock: &dto.ClaudeMediaMessage{
  386. Id: toolCall.ID,
  387. Type: "tool_use",
  388. Name: toolCall.Function.Name,
  389. Input: map[string]interface{}{},
  390. },
  391. })
  392. }
  393. if len(toolCall.Function.Arguments) > 0 {
  394. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  395. Index: &idx,
  396. Type: "content_block_delta",
  397. Delta: &dto.ClaudeMediaMessage{
  398. Type: "input_json_delta",
  399. PartialJson: &toolCall.Function.Arguments,
  400. },
  401. })
  402. }
  403. info.ClaudeConvertInfo.Index = blockIndex
  404. }
  405. } else {
  406. reasoning := chosenChoice.Delta.GetReasoningContent()
  407. textContent := chosenChoice.Delta.GetContentString()
  408. if reasoning != "" || textContent != "" {
  409. if reasoning != "" {
  410. if info.ClaudeConvertInfo.LastMessagesType != relaycommon.LastMessageTypeThinking {
  411. if info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeText || info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeTools {
  412. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  413. info.ClaudeConvertInfo.Index++
  414. }
  415. idx := info.ClaudeConvertInfo.Index
  416. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  417. Index: &idx,
  418. Type: "content_block_start",
  419. ContentBlock: &dto.ClaudeMediaMessage{
  420. Type: "thinking",
  421. Thinking: common.GetPointer[string](""),
  422. },
  423. })
  424. }
  425. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeThinking
  426. claudeResponse.Delta = &dto.ClaudeMediaMessage{
  427. Type: "thinking_delta",
  428. Thinking: &reasoning,
  429. }
  430. } else {
  431. if info.ClaudeConvertInfo.LastMessagesType != relaycommon.LastMessageTypeText {
  432. if info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeThinking || info.ClaudeConvertInfo.LastMessagesType == relaycommon.LastMessageTypeTools {
  433. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  434. info.ClaudeConvertInfo.Index++
  435. }
  436. idx := info.ClaudeConvertInfo.Index
  437. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  438. Index: &idx,
  439. Type: "content_block_start",
  440. ContentBlock: &dto.ClaudeMediaMessage{
  441. Type: "text",
  442. Text: common.GetPointer[string](""),
  443. },
  444. })
  445. }
  446. info.ClaudeConvertInfo.LastMessagesType = relaycommon.LastMessageTypeText
  447. claudeResponse.Delta = &dto.ClaudeMediaMessage{
  448. Type: "text_delta",
  449. Text: common.GetPointer[string](textContent),
  450. }
  451. }
  452. } else {
  453. isEmpty = true
  454. }
  455. }
  456. claudeResponse.Index = common.GetPointer[int](info.ClaudeConvertInfo.Index)
  457. if !isEmpty && claudeResponse.Delta != nil {
  458. claudeResponses = append(claudeResponses, &claudeResponse)
  459. }
  460. if doneChunk || info.ClaudeConvertInfo.Done {
  461. claudeResponses = append(claudeResponses, generateStopBlock(info.ClaudeConvertInfo.Index))
  462. oaiUsage := openAIResponse.Usage
  463. if oaiUsage == nil {
  464. oaiUsage = info.ClaudeConvertInfo.Usage
  465. }
  466. if oaiUsage != nil {
  467. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  468. Type: "message_delta",
  469. Usage: &dto.ClaudeUsage{
  470. InputTokens: oaiUsage.PromptTokens,
  471. OutputTokens: oaiUsage.CompletionTokens,
  472. CacheCreationInputTokens: oaiUsage.PromptTokensDetails.CachedCreationTokens,
  473. CacheReadInputTokens: oaiUsage.PromptTokensDetails.CachedTokens,
  474. },
  475. Delta: &dto.ClaudeMediaMessage{
  476. StopReason: common.GetPointer[string](stopReasonOpenAI2Claude(info.FinishReason)),
  477. },
  478. })
  479. }
  480. claudeResponses = append(claudeResponses, &dto.ClaudeResponse{
  481. Type: "message_stop",
  482. })
  483. info.ClaudeConvertInfo.Done = true
  484. return claudeResponses
  485. }
  486. }
  487. return claudeResponses
  488. }
  489. func ResponseOpenAI2Claude(openAIResponse *dto.OpenAITextResponse, info *relaycommon.RelayInfo) *dto.ClaudeResponse {
  490. var stopReason string
  491. contents := make([]dto.ClaudeMediaMessage, 0)
  492. claudeResponse := &dto.ClaudeResponse{
  493. Id: openAIResponse.Id,
  494. Type: "message",
  495. Role: "assistant",
  496. Model: openAIResponse.Model,
  497. }
  498. for _, choice := range openAIResponse.Choices {
  499. stopReason = stopReasonOpenAI2Claude(choice.FinishReason)
  500. if choice.FinishReason == "tool_calls" {
  501. for _, toolUse := range choice.Message.ParseToolCalls() {
  502. claudeContent := dto.ClaudeMediaMessage{}
  503. claudeContent.Type = "tool_use"
  504. claudeContent.Id = toolUse.ID
  505. claudeContent.Name = toolUse.Function.Name
  506. var mapParams map[string]interface{}
  507. if err := common.Unmarshal([]byte(toolUse.Function.Arguments), &mapParams); err == nil {
  508. claudeContent.Input = mapParams
  509. } else {
  510. claudeContent.Input = toolUse.Function.Arguments
  511. }
  512. contents = append(contents, claudeContent)
  513. }
  514. } else {
  515. claudeContent := dto.ClaudeMediaMessage{}
  516. claudeContent.Type = "text"
  517. claudeContent.SetText(choice.Message.StringContent())
  518. contents = append(contents, claudeContent)
  519. }
  520. }
  521. claudeResponse.Content = contents
  522. claudeResponse.StopReason = stopReason
  523. claudeResponse.Usage = &dto.ClaudeUsage{
  524. InputTokens: openAIResponse.PromptTokens,
  525. OutputTokens: openAIResponse.CompletionTokens,
  526. }
  527. return claudeResponse
  528. }
  529. func stopReasonOpenAI2Claude(reason string) string {
  530. return reasonmap.OpenAIFinishReasonToClaudeStopReason(reason)
  531. }
  532. func toJSONString(v interface{}) string {
  533. b, err := json.Marshal(v)
  534. if err != nil {
  535. return "{}"
  536. }
  537. return string(b)
  538. }
  539. func GeminiToOpenAIRequest(geminiRequest *dto.GeminiChatRequest, info *relaycommon.RelayInfo) (*dto.GeneralOpenAIRequest, error) {
  540. openaiRequest := &dto.GeneralOpenAIRequest{
  541. Model: info.UpstreamModelName,
  542. Stream: info.IsStream,
  543. }
  544. // 转换 messages
  545. var messages []dto.Message
  546. for _, content := range geminiRequest.Contents {
  547. message := dto.Message{
  548. Role: convertGeminiRoleToOpenAI(content.Role),
  549. }
  550. // 处理 parts
  551. var mediaContents []dto.MediaContent
  552. var toolCalls []dto.ToolCallRequest
  553. for _, part := range content.Parts {
  554. if part.Text != "" {
  555. mediaContent := dto.MediaContent{
  556. Type: "text",
  557. Text: part.Text,
  558. }
  559. mediaContents = append(mediaContents, mediaContent)
  560. } else if part.InlineData != nil {
  561. mediaContent := dto.MediaContent{
  562. Type: "image_url",
  563. ImageUrl: &dto.MessageImageUrl{
  564. Url: fmt.Sprintf("data:%s;base64,%s", part.InlineData.MimeType, part.InlineData.Data),
  565. Detail: "auto",
  566. MimeType: part.InlineData.MimeType,
  567. },
  568. }
  569. mediaContents = append(mediaContents, mediaContent)
  570. } else if part.FileData != nil {
  571. mediaContent := dto.MediaContent{
  572. Type: "image_url",
  573. ImageUrl: &dto.MessageImageUrl{
  574. Url: part.FileData.FileUri,
  575. Detail: "auto",
  576. MimeType: part.FileData.MimeType,
  577. },
  578. }
  579. mediaContents = append(mediaContents, mediaContent)
  580. } else if part.FunctionCall != nil {
  581. // 处理 Gemini 的工具调用
  582. toolCall := dto.ToolCallRequest{
  583. ID: fmt.Sprintf("call_%d", len(toolCalls)+1), // 生成唯一ID
  584. Type: "function",
  585. Function: dto.FunctionRequest{
  586. Name: part.FunctionCall.FunctionName,
  587. Arguments: toJSONString(part.FunctionCall.Arguments),
  588. },
  589. }
  590. toolCalls = append(toolCalls, toolCall)
  591. } else if part.FunctionResponse != nil {
  592. // 处理 Gemini 的工具响应,创建单独的 tool 消息
  593. toolMessage := dto.Message{
  594. Role: "tool",
  595. ToolCallId: fmt.Sprintf("call_%d", len(toolCalls)), // 使用对应的调用ID
  596. }
  597. toolMessage.SetStringContent(toJSONString(part.FunctionResponse.Response))
  598. messages = append(messages, toolMessage)
  599. }
  600. }
  601. // 设置消息内容
  602. if len(toolCalls) > 0 {
  603. // 如果有工具调用,设置工具调用
  604. message.SetToolCalls(toolCalls)
  605. } else if len(mediaContents) == 1 && mediaContents[0].Type == "text" {
  606. // 如果只有一个文本内容,直接设置字符串
  607. message.Content = mediaContents[0].Text
  608. } else if len(mediaContents) > 0 {
  609. // 如果有多个内容或包含媒体,设置为数组
  610. message.SetMediaContent(mediaContents)
  611. }
  612. // 只有当消息有内容或工具调用时才添加
  613. if len(message.ParseContent()) > 0 || len(message.ToolCalls) > 0 {
  614. messages = append(messages, message)
  615. }
  616. }
  617. openaiRequest.Messages = messages
  618. if geminiRequest.GenerationConfig.Temperature != nil {
  619. openaiRequest.Temperature = geminiRequest.GenerationConfig.Temperature
  620. }
  621. if geminiRequest.GenerationConfig.TopP > 0 {
  622. openaiRequest.TopP = geminiRequest.GenerationConfig.TopP
  623. }
  624. if geminiRequest.GenerationConfig.TopK > 0 {
  625. openaiRequest.TopK = int(geminiRequest.GenerationConfig.TopK)
  626. }
  627. if geminiRequest.GenerationConfig.MaxOutputTokens > 0 {
  628. openaiRequest.MaxTokens = geminiRequest.GenerationConfig.MaxOutputTokens
  629. }
  630. // gemini stop sequences 最多 5 个,openai stop 最多 4 个
  631. if len(geminiRequest.GenerationConfig.StopSequences) > 0 {
  632. openaiRequest.Stop = geminiRequest.GenerationConfig.StopSequences[:4]
  633. }
  634. if geminiRequest.GenerationConfig.CandidateCount > 0 {
  635. openaiRequest.N = geminiRequest.GenerationConfig.CandidateCount
  636. }
  637. // 转换工具调用
  638. if len(geminiRequest.GetTools()) > 0 {
  639. var tools []dto.ToolCallRequest
  640. for _, tool := range geminiRequest.GetTools() {
  641. if tool.FunctionDeclarations != nil {
  642. functionDeclarations, err := common.Any2Type[[]dto.FunctionRequest](tool.FunctionDeclarations)
  643. if err != nil {
  644. common.SysError(fmt.Sprintf("failed to parse gemini function declarations: %v (type=%T)", err, tool.FunctionDeclarations))
  645. continue
  646. }
  647. for _, function := range functionDeclarations {
  648. openAITool := dto.ToolCallRequest{
  649. Type: "function",
  650. Function: dto.FunctionRequest{
  651. Name: function.Name,
  652. Description: function.Description,
  653. Parameters: function.Parameters,
  654. },
  655. }
  656. tools = append(tools, openAITool)
  657. }
  658. }
  659. }
  660. if len(tools) > 0 {
  661. openaiRequest.Tools = tools
  662. }
  663. }
  664. // gemini system instructions
  665. if geminiRequest.SystemInstructions != nil {
  666. // 将系统指令作为第一条消息插入
  667. systemMessage := dto.Message{
  668. Role: "system",
  669. Content: extractTextFromGeminiParts(geminiRequest.SystemInstructions.Parts),
  670. }
  671. openaiRequest.Messages = append([]dto.Message{systemMessage}, openaiRequest.Messages...)
  672. }
  673. return openaiRequest, nil
  674. }
  675. func convertGeminiRoleToOpenAI(geminiRole string) string {
  676. switch geminiRole {
  677. case "user":
  678. return "user"
  679. case "model":
  680. return "assistant"
  681. case "function":
  682. return "function"
  683. default:
  684. return "user"
  685. }
  686. }
  687. func extractTextFromGeminiParts(parts []dto.GeminiPart) string {
  688. var texts []string
  689. for _, part := range parts {
  690. if part.Text != "" {
  691. texts = append(texts, part.Text)
  692. }
  693. }
  694. return strings.Join(texts, "\n")
  695. }
  696. // ResponseOpenAI2Gemini 将 OpenAI 响应转换为 Gemini 格式
  697. func ResponseOpenAI2Gemini(openAIResponse *dto.OpenAITextResponse, info *relaycommon.RelayInfo) *dto.GeminiChatResponse {
  698. geminiResponse := &dto.GeminiChatResponse{
  699. Candidates: make([]dto.GeminiChatCandidate, 0, len(openAIResponse.Choices)),
  700. UsageMetadata: dto.GeminiUsageMetadata{
  701. PromptTokenCount: openAIResponse.PromptTokens,
  702. CandidatesTokenCount: openAIResponse.CompletionTokens,
  703. TotalTokenCount: openAIResponse.PromptTokens + openAIResponse.CompletionTokens,
  704. },
  705. }
  706. for _, choice := range openAIResponse.Choices {
  707. candidate := dto.GeminiChatCandidate{
  708. Index: int64(choice.Index),
  709. SafetyRatings: []dto.GeminiChatSafetyRating{},
  710. }
  711. // 设置结束原因
  712. var finishReason string
  713. switch choice.FinishReason {
  714. case "stop":
  715. finishReason = "STOP"
  716. case "length":
  717. finishReason = "MAX_TOKENS"
  718. case "content_filter":
  719. finishReason = "SAFETY"
  720. case "tool_calls":
  721. finishReason = "STOP"
  722. default:
  723. finishReason = "STOP"
  724. }
  725. candidate.FinishReason = &finishReason
  726. // 转换消息内容
  727. content := dto.GeminiChatContent{
  728. Role: "model",
  729. Parts: make([]dto.GeminiPart, 0),
  730. }
  731. // 处理工具调用
  732. toolCalls := choice.Message.ParseToolCalls()
  733. if len(toolCalls) > 0 {
  734. for _, toolCall := range toolCalls {
  735. // 解析参数
  736. var args map[string]interface{}
  737. if toolCall.Function.Arguments != "" {
  738. if err := json.Unmarshal([]byte(toolCall.Function.Arguments), &args); err != nil {
  739. args = map[string]interface{}{"arguments": toolCall.Function.Arguments}
  740. }
  741. } else {
  742. args = make(map[string]interface{})
  743. }
  744. part := dto.GeminiPart{
  745. FunctionCall: &dto.FunctionCall{
  746. FunctionName: toolCall.Function.Name,
  747. Arguments: args,
  748. },
  749. }
  750. content.Parts = append(content.Parts, part)
  751. }
  752. } else {
  753. // 处理文本内容
  754. textContent := choice.Message.StringContent()
  755. if textContent != "" {
  756. part := dto.GeminiPart{
  757. Text: textContent,
  758. }
  759. content.Parts = append(content.Parts, part)
  760. }
  761. }
  762. candidate.Content = content
  763. geminiResponse.Candidates = append(geminiResponse.Candidates, candidate)
  764. }
  765. return geminiResponse
  766. }
  767. // StreamResponseOpenAI2Gemini 将 OpenAI 流式响应转换为 Gemini 格式
  768. func StreamResponseOpenAI2Gemini(openAIResponse *dto.ChatCompletionsStreamResponse, info *relaycommon.RelayInfo) *dto.GeminiChatResponse {
  769. // 检查是否有实际内容或结束标志
  770. hasContent := false
  771. hasFinishReason := false
  772. for _, choice := range openAIResponse.Choices {
  773. if len(choice.Delta.GetContentString()) > 0 || (choice.Delta.ToolCalls != nil && len(choice.Delta.ToolCalls) > 0) {
  774. hasContent = true
  775. }
  776. if choice.FinishReason != nil {
  777. hasFinishReason = true
  778. }
  779. }
  780. // 如果没有实际内容且没有结束标志,跳过。主要针对 openai 流响应开头的空数据
  781. if !hasContent && !hasFinishReason {
  782. return nil
  783. }
  784. geminiResponse := &dto.GeminiChatResponse{
  785. Candidates: make([]dto.GeminiChatCandidate, 0, len(openAIResponse.Choices)),
  786. UsageMetadata: dto.GeminiUsageMetadata{
  787. PromptTokenCount: info.GetEstimatePromptTokens(),
  788. CandidatesTokenCount: 0, // 流式响应中可能没有完整的 usage 信息
  789. TotalTokenCount: info.GetEstimatePromptTokens(),
  790. },
  791. }
  792. if openAIResponse.Usage != nil {
  793. geminiResponse.UsageMetadata.PromptTokenCount = openAIResponse.Usage.PromptTokens
  794. geminiResponse.UsageMetadata.CandidatesTokenCount = openAIResponse.Usage.CompletionTokens
  795. geminiResponse.UsageMetadata.TotalTokenCount = openAIResponse.Usage.TotalTokens
  796. }
  797. for _, choice := range openAIResponse.Choices {
  798. candidate := dto.GeminiChatCandidate{
  799. Index: int64(choice.Index),
  800. SafetyRatings: []dto.GeminiChatSafetyRating{},
  801. }
  802. // 设置结束原因
  803. if choice.FinishReason != nil {
  804. var finishReason string
  805. switch *choice.FinishReason {
  806. case "stop":
  807. finishReason = "STOP"
  808. case "length":
  809. finishReason = "MAX_TOKENS"
  810. case "content_filter":
  811. finishReason = "SAFETY"
  812. case "tool_calls":
  813. finishReason = "STOP"
  814. default:
  815. finishReason = "STOP"
  816. }
  817. candidate.FinishReason = &finishReason
  818. }
  819. // 转换消息内容
  820. content := dto.GeminiChatContent{
  821. Role: "model",
  822. Parts: make([]dto.GeminiPart, 0),
  823. }
  824. // 处理工具调用
  825. if choice.Delta.ToolCalls != nil {
  826. for _, toolCall := range choice.Delta.ToolCalls {
  827. // 解析参数
  828. var args map[string]interface{}
  829. if toolCall.Function.Arguments != "" {
  830. if err := json.Unmarshal([]byte(toolCall.Function.Arguments), &args); err != nil {
  831. args = map[string]interface{}{"arguments": toolCall.Function.Arguments}
  832. }
  833. } else {
  834. args = make(map[string]interface{})
  835. }
  836. part := dto.GeminiPart{
  837. FunctionCall: &dto.FunctionCall{
  838. FunctionName: toolCall.Function.Name,
  839. Arguments: args,
  840. },
  841. }
  842. content.Parts = append(content.Parts, part)
  843. }
  844. } else {
  845. // 处理文本内容
  846. textContent := choice.Delta.GetContentString()
  847. if textContent != "" {
  848. part := dto.GeminiPart{
  849. Text: textContent,
  850. }
  851. content.Parts = append(content.Parts, part)
  852. }
  853. }
  854. candidate.Content = content
  855. geminiResponse.Candidates = append(geminiResponse.Candidates, candidate)
  856. }
  857. return geminiResponse
  858. }