claude.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572
  1. package dto
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "strings"
  6. "github.com/QuantumNous/new-api/common"
  7. "github.com/QuantumNous/new-api/types"
  8. "github.com/gin-gonic/gin"
  9. )
  10. type ClaudeMetadata struct {
  11. UserId string `json:"user_id"`
  12. }
  13. type ClaudeMediaMessage struct {
  14. Type string `json:"type,omitempty"`
  15. Text *string `json:"text,omitempty"`
  16. Model string `json:"model,omitempty"`
  17. Source *ClaudeMessageSource `json:"source,omitempty"`
  18. Usage *ClaudeUsage `json:"usage,omitempty"`
  19. StopReason *string `json:"stop_reason,omitempty"`
  20. PartialJson *string `json:"partial_json,omitempty"`
  21. Role string `json:"role,omitempty"`
  22. Thinking *string `json:"thinking,omitempty"`
  23. Signature string `json:"signature,omitempty"`
  24. Delta string `json:"delta,omitempty"`
  25. CacheControl json.RawMessage `json:"cache_control,omitempty"`
  26. // tool_calls
  27. Id string `json:"id,omitempty"`
  28. Name string `json:"name,omitempty"`
  29. Input any `json:"input,omitempty"`
  30. Content any `json:"content,omitempty"`
  31. ToolUseId string `json:"tool_use_id,omitempty"`
  32. }
  33. func (c *ClaudeMediaMessage) SetText(s string) {
  34. c.Text = &s
  35. }
  36. func (c *ClaudeMediaMessage) GetText() string {
  37. if c.Text == nil {
  38. return ""
  39. }
  40. return *c.Text
  41. }
  42. func (c *ClaudeMediaMessage) IsStringContent() bool {
  43. if c.Content == nil {
  44. return false
  45. }
  46. _, ok := c.Content.(string)
  47. if ok {
  48. return true
  49. }
  50. return false
  51. }
  52. func (c *ClaudeMediaMessage) GetStringContent() string {
  53. if c.Content == nil {
  54. return ""
  55. }
  56. switch c.Content.(type) {
  57. case string:
  58. return c.Content.(string)
  59. case []any:
  60. var contentStr string
  61. for _, contentItem := range c.Content.([]any) {
  62. contentMap, ok := contentItem.(map[string]any)
  63. if !ok {
  64. continue
  65. }
  66. if contentMap["type"] == ContentTypeText {
  67. if subStr, ok := contentMap["text"].(string); ok {
  68. contentStr += subStr
  69. }
  70. }
  71. }
  72. return contentStr
  73. }
  74. return ""
  75. }
  76. func (c *ClaudeMediaMessage) GetJsonRowString() string {
  77. jsonContent, _ := common.Marshal(c)
  78. return string(jsonContent)
  79. }
  80. func (c *ClaudeMediaMessage) SetContent(content any) {
  81. c.Content = content
  82. }
  83. func (c *ClaudeMediaMessage) ParseMediaContent() []ClaudeMediaMessage {
  84. mediaContent, _ := common.Any2Type[[]ClaudeMediaMessage](c.Content)
  85. return mediaContent
  86. }
  87. type ClaudeMessageSource struct {
  88. Type string `json:"type"`
  89. MediaType string `json:"media_type,omitempty"`
  90. Data any `json:"data,omitempty"`
  91. Url string `json:"url,omitempty"`
  92. }
  93. type ClaudeMessage struct {
  94. Role string `json:"role"`
  95. Content any `json:"content"`
  96. }
  97. func (c *ClaudeMessage) IsStringContent() bool {
  98. if c.Content == nil {
  99. return false
  100. }
  101. _, ok := c.Content.(string)
  102. return ok
  103. }
  104. func (c *ClaudeMessage) GetStringContent() string {
  105. if c.Content == nil {
  106. return ""
  107. }
  108. switch c.Content.(type) {
  109. case string:
  110. return c.Content.(string)
  111. case []any:
  112. var contentStr string
  113. for _, contentItem := range c.Content.([]any) {
  114. contentMap, ok := contentItem.(map[string]any)
  115. if !ok {
  116. continue
  117. }
  118. if contentMap["type"] == ContentTypeText {
  119. if subStr, ok := contentMap["text"].(string); ok {
  120. contentStr += subStr
  121. }
  122. }
  123. }
  124. return contentStr
  125. }
  126. return ""
  127. }
  128. func (c *ClaudeMessage) SetStringContent(content string) {
  129. c.Content = content
  130. }
  131. func (c *ClaudeMessage) SetContent(content any) {
  132. c.Content = content
  133. }
  134. func (c *ClaudeMessage) ParseContent() ([]ClaudeMediaMessage, error) {
  135. return common.Any2Type[[]ClaudeMediaMessage](c.Content)
  136. }
  137. type Tool struct {
  138. Name string `json:"name"`
  139. Description string `json:"description,omitempty"`
  140. InputSchema map[string]interface{} `json:"input_schema"`
  141. }
  142. type InputSchema struct {
  143. Type string `json:"type"`
  144. Properties any `json:"properties,omitempty"`
  145. Required any `json:"required,omitempty"`
  146. }
  147. type ClaudeWebSearchTool struct {
  148. Type string `json:"type"`
  149. Name string `json:"name"`
  150. MaxUses int `json:"max_uses,omitempty"`
  151. UserLocation *ClaudeWebSearchUserLocation `json:"user_location,omitempty"`
  152. }
  153. type ClaudeWebSearchUserLocation struct {
  154. Type string `json:"type"`
  155. Timezone string `json:"timezone,omitempty"`
  156. Country string `json:"country,omitempty"`
  157. Region string `json:"region,omitempty"`
  158. City string `json:"city,omitempty"`
  159. }
  160. type ClaudeToolChoice struct {
  161. Type string `json:"type"`
  162. Name string `json:"name,omitempty"`
  163. DisableParallelToolUse bool `json:"disable_parallel_tool_use,omitempty"`
  164. }
  165. type ClaudeRequest struct {
  166. Model string `json:"model"`
  167. Prompt string `json:"prompt,omitempty"`
  168. System any `json:"system,omitempty"`
  169. Messages []ClaudeMessage `json:"messages,omitempty"`
  170. MaxTokens uint `json:"max_tokens,omitempty"`
  171. MaxTokensToSample uint `json:"max_tokens_to_sample,omitempty"`
  172. StopSequences []string `json:"stop_sequences,omitempty"`
  173. Temperature *float64 `json:"temperature,omitempty"`
  174. TopP float64 `json:"top_p,omitempty"`
  175. TopK int `json:"top_k,omitempty"`
  176. Stream bool `json:"stream,omitempty"`
  177. Tools any `json:"tools,omitempty"`
  178. ContextManagement json.RawMessage `json:"context_management,omitempty"`
  179. OutputConfig json.RawMessage `json:"output_config,omitempty"`
  180. OutputFormat json.RawMessage `json:"output_format,omitempty"`
  181. Container json.RawMessage `json:"container,omitempty"`
  182. ToolChoice any `json:"tool_choice,omitempty"`
  183. Thinking *Thinking `json:"thinking,omitempty"`
  184. McpServers json.RawMessage `json:"mcp_servers,omitempty"`
  185. Metadata json.RawMessage `json:"metadata,omitempty"`
  186. // 服务层级字段,用于指定 API 服务等级。允许透传可能导致实际计费高于预期,默认应过滤
  187. ServiceTier string `json:"service_tier,omitempty"`
  188. }
  189. // createClaudeFileSource 根据数据内容创建正确类型的 FileSource
  190. func createClaudeFileSource(data string) *types.FileSource {
  191. if strings.HasPrefix(data, "http://") || strings.HasPrefix(data, "https://") {
  192. return types.NewURLFileSource(data)
  193. }
  194. return types.NewBase64FileSource(data, "")
  195. }
  196. func (c *ClaudeRequest) GetTokenCountMeta() *types.TokenCountMeta {
  197. var tokenCountMeta = types.TokenCountMeta{
  198. TokenType: types.TokenTypeTokenizer,
  199. MaxTokens: int(c.MaxTokens),
  200. }
  201. var texts = make([]string, 0)
  202. var fileMeta = make([]*types.FileMeta, 0)
  203. // system
  204. if c.System != nil {
  205. if c.IsStringSystem() {
  206. sys := c.GetStringSystem()
  207. if sys != "" {
  208. texts = append(texts, sys)
  209. }
  210. } else {
  211. systemMedia := c.ParseSystem()
  212. for _, media := range systemMedia {
  213. switch media.Type {
  214. case "text":
  215. texts = append(texts, media.GetText())
  216. case "image":
  217. if media.Source != nil {
  218. data := media.Source.Url
  219. if data == "" {
  220. data = common.Interface2String(media.Source.Data)
  221. }
  222. if data != "" {
  223. fileMeta = append(fileMeta, &types.FileMeta{
  224. FileType: types.FileTypeImage,
  225. Source: createClaudeFileSource(data),
  226. })
  227. }
  228. }
  229. }
  230. }
  231. }
  232. }
  233. // messages
  234. for _, message := range c.Messages {
  235. tokenCountMeta.MessagesCount++
  236. texts = append(texts, message.Role)
  237. if message.IsStringContent() {
  238. content := message.GetStringContent()
  239. if content != "" {
  240. texts = append(texts, content)
  241. }
  242. continue
  243. }
  244. content, _ := message.ParseContent()
  245. for _, media := range content {
  246. switch media.Type {
  247. case "text":
  248. texts = append(texts, media.GetText())
  249. case "image":
  250. if media.Source != nil {
  251. data := media.Source.Url
  252. if data == "" {
  253. data = common.Interface2String(media.Source.Data)
  254. }
  255. if data != "" {
  256. fileMeta = append(fileMeta, &types.FileMeta{
  257. FileType: types.FileTypeImage,
  258. Source: createClaudeFileSource(data),
  259. })
  260. }
  261. }
  262. case "tool_use":
  263. if media.Name != "" {
  264. texts = append(texts, media.Name)
  265. }
  266. if media.Input != nil {
  267. b, _ := common.Marshal(media.Input)
  268. texts = append(texts, string(b))
  269. }
  270. case "tool_result":
  271. if media.Content != nil {
  272. b, _ := common.Marshal(media.Content)
  273. texts = append(texts, string(b))
  274. }
  275. }
  276. }
  277. }
  278. // tools
  279. if c.Tools != nil {
  280. tools := c.GetTools()
  281. normalTools, webSearchTools := ProcessTools(tools)
  282. if normalTools != nil {
  283. for _, t := range normalTools {
  284. tokenCountMeta.ToolsCount++
  285. if t.Name != "" {
  286. texts = append(texts, t.Name)
  287. }
  288. if t.Description != "" {
  289. texts = append(texts, t.Description)
  290. }
  291. if t.InputSchema != nil {
  292. b, _ := common.Marshal(t.InputSchema)
  293. texts = append(texts, string(b))
  294. }
  295. }
  296. }
  297. if webSearchTools != nil {
  298. for _, t := range webSearchTools {
  299. tokenCountMeta.ToolsCount++
  300. if t.Name != "" {
  301. texts = append(texts, t.Name)
  302. }
  303. if t.UserLocation != nil {
  304. b, _ := common.Marshal(t.UserLocation)
  305. texts = append(texts, string(b))
  306. }
  307. }
  308. }
  309. }
  310. tokenCountMeta.CombineText = strings.Join(texts, "\n")
  311. tokenCountMeta.Files = fileMeta
  312. return &tokenCountMeta
  313. }
  314. func (c *ClaudeRequest) IsStream(ctx *gin.Context) bool {
  315. return c.Stream
  316. }
  317. func (c *ClaudeRequest) SetModelName(modelName string) {
  318. if modelName != "" {
  319. c.Model = modelName
  320. }
  321. }
  322. func (c *ClaudeRequest) SearchToolNameByToolCallId(toolCallId string) string {
  323. for _, message := range c.Messages {
  324. content, _ := message.ParseContent()
  325. for _, mediaMessage := range content {
  326. if mediaMessage.Id == toolCallId {
  327. return mediaMessage.Name
  328. }
  329. }
  330. }
  331. return ""
  332. }
  333. // AddTool 添加工具到请求中
  334. func (c *ClaudeRequest) AddTool(tool any) {
  335. if c.Tools == nil {
  336. c.Tools = make([]any, 0)
  337. }
  338. switch tools := c.Tools.(type) {
  339. case []any:
  340. c.Tools = append(tools, tool)
  341. default:
  342. // 如果Tools不是[]any类型,重新初始化为[]any
  343. c.Tools = []any{tool}
  344. }
  345. }
  346. // GetTools 获取工具列表
  347. func (c *ClaudeRequest) GetTools() []any {
  348. if c.Tools == nil {
  349. return nil
  350. }
  351. switch tools := c.Tools.(type) {
  352. case []any:
  353. return tools
  354. default:
  355. return nil
  356. }
  357. }
  358. // ProcessTools 处理工具列表,支持类型断言
  359. func ProcessTools(tools []any) ([]*Tool, []*ClaudeWebSearchTool) {
  360. var normalTools []*Tool
  361. var webSearchTools []*ClaudeWebSearchTool
  362. for _, tool := range tools {
  363. switch t := tool.(type) {
  364. case *Tool:
  365. normalTools = append(normalTools, t)
  366. case *ClaudeWebSearchTool:
  367. webSearchTools = append(webSearchTools, t)
  368. case Tool:
  369. normalTools = append(normalTools, &t)
  370. case ClaudeWebSearchTool:
  371. webSearchTools = append(webSearchTools, &t)
  372. default:
  373. // 未知类型,跳过
  374. continue
  375. }
  376. }
  377. return normalTools, webSearchTools
  378. }
  379. type Thinking struct {
  380. Type string `json:"type"`
  381. BudgetTokens *int `json:"budget_tokens,omitempty"`
  382. }
  383. func (c *Thinking) GetBudgetTokens() int {
  384. if c.BudgetTokens == nil {
  385. return 0
  386. }
  387. return *c.BudgetTokens
  388. }
  389. func (c *ClaudeRequest) IsStringSystem() bool {
  390. _, ok := c.System.(string)
  391. return ok
  392. }
  393. func (c *ClaudeRequest) GetStringSystem() string {
  394. if c.IsStringSystem() {
  395. return c.System.(string)
  396. }
  397. return ""
  398. }
  399. func (c *ClaudeRequest) SetStringSystem(system string) {
  400. c.System = system
  401. }
  402. func (c *ClaudeRequest) ParseSystem() []ClaudeMediaMessage {
  403. mediaContent, _ := common.Any2Type[[]ClaudeMediaMessage](c.System)
  404. return mediaContent
  405. }
  406. type ClaudeErrorWithStatusCode struct {
  407. Error types.ClaudeError `json:"error"`
  408. StatusCode int `json:"status_code"`
  409. LocalError bool
  410. }
  411. type ClaudeResponse struct {
  412. Id string `json:"id,omitempty"`
  413. Type string `json:"type"`
  414. Role string `json:"role,omitempty"`
  415. Content []ClaudeMediaMessage `json:"content,omitempty"`
  416. Completion string `json:"completion,omitempty"`
  417. StopReason string `json:"stop_reason,omitempty"`
  418. Model string `json:"model,omitempty"`
  419. Error any `json:"error,omitempty"`
  420. Usage *ClaudeUsage `json:"usage,omitempty"`
  421. Index *int `json:"index,omitempty"`
  422. ContentBlock *ClaudeMediaMessage `json:"content_block,omitempty"`
  423. Delta *ClaudeMediaMessage `json:"delta,omitempty"`
  424. Message *ClaudeMediaMessage `json:"message,omitempty"`
  425. }
  426. // set index
  427. func (c *ClaudeResponse) SetIndex(i int) {
  428. c.Index = &i
  429. }
  430. // get index
  431. func (c *ClaudeResponse) GetIndex() int {
  432. if c.Index == nil {
  433. return 0
  434. }
  435. return *c.Index
  436. }
  437. // GetClaudeError 从动态错误类型中提取ClaudeError结构
  438. func (c *ClaudeResponse) GetClaudeError() *types.ClaudeError {
  439. if c.Error == nil {
  440. return nil
  441. }
  442. switch err := c.Error.(type) {
  443. case types.ClaudeError:
  444. return &err
  445. case *types.ClaudeError:
  446. return err
  447. case map[string]interface{}:
  448. // 处理从JSON解析来的map结构
  449. claudeErr := &types.ClaudeError{}
  450. if errType, ok := err["type"].(string); ok {
  451. claudeErr.Type = errType
  452. }
  453. if errMsg, ok := err["message"].(string); ok {
  454. claudeErr.Message = errMsg
  455. }
  456. return claudeErr
  457. case string:
  458. // 处理简单字符串错误
  459. return &types.ClaudeError{
  460. Type: "upstream_error",
  461. Message: err,
  462. }
  463. default:
  464. // 未知类型,尝试转换为字符串
  465. return &types.ClaudeError{
  466. Type: "unknown_upstream_error",
  467. Message: fmt.Sprintf("unknown_error: %v", err),
  468. }
  469. }
  470. }
  471. type ClaudeUsage struct {
  472. InputTokens int `json:"input_tokens"`
  473. CacheCreationInputTokens int `json:"cache_creation_input_tokens"`
  474. CacheReadInputTokens int `json:"cache_read_input_tokens"`
  475. OutputTokens int `json:"output_tokens"`
  476. CacheCreation *ClaudeCacheCreationUsage `json:"cache_creation,omitempty"`
  477. // claude cache 1h
  478. ClaudeCacheCreation5mTokens int `json:"claude_cache_creation_5_m_tokens"`
  479. ClaudeCacheCreation1hTokens int `json:"claude_cache_creation_1_h_tokens"`
  480. ServerToolUse *ClaudeServerToolUse `json:"server_tool_use,omitempty"`
  481. }
  482. type ClaudeCacheCreationUsage struct {
  483. Ephemeral5mInputTokens int `json:"ephemeral_5m_input_tokens,omitempty"`
  484. Ephemeral1hInputTokens int `json:"ephemeral_1h_input_tokens,omitempty"`
  485. }
  486. func (u *ClaudeUsage) GetCacheCreation5mTokens() int {
  487. if u == nil || u.CacheCreation == nil {
  488. return 0
  489. }
  490. return u.CacheCreation.Ephemeral5mInputTokens
  491. }
  492. func (u *ClaudeUsage) GetCacheCreation1hTokens() int {
  493. if u == nil || u.CacheCreation == nil {
  494. return 0
  495. }
  496. return u.CacheCreation.Ephemeral1hInputTokens
  497. }
  498. func (u *ClaudeUsage) GetCacheCreationTotalTokens() int {
  499. if u == nil {
  500. return 0
  501. }
  502. if u.CacheCreationInputTokens > 0 {
  503. return u.CacheCreationInputTokens
  504. }
  505. return u.GetCacheCreation5mTokens() + u.GetCacheCreation1hTokens()
  506. }
  507. type ClaudeServerToolUse struct {
  508. WebSearchRequests int `json:"web_search_requests"`
  509. }