result.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. package compapi
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "strings"
  8. "wechat-api/internal/types"
  9. )
  10. // 包装了原始的 API 响应,并提供了解析助手方法
  11. type ChatResult struct {
  12. *types.CompOpenApiResp
  13. err error
  14. }
  15. // NewChatResult 是 ChatResult 的构造函数
  16. func NewChatResult(resp any) *ChatResult {
  17. nresp, err := AnyToCompOpenApiResp(resp)
  18. res := ChatResult{nresp, err}
  19. if nresp == nil {
  20. res.err = errors.New("CompOpenApiRes is nil")
  21. }
  22. return &res
  23. }
  24. // 以下是ChatResul助手方法
  25. // GetContentString 返回第一个 Choice 的 Message Content (标准方式)
  26. func (r *ChatResult) GetContentString() (string, error) {
  27. var (
  28. content string = ""
  29. err error
  30. )
  31. if r.err == nil && len(r.Choices) > 0 {
  32. content = r.Choices[0].Message.Content
  33. } else if r.err == nil && len(r.Choices) == 0 {
  34. err = errors.New("choices empty")
  35. }
  36. return content, err
  37. }
  38. // ParseContentAs 解析 Message Content 中的 JSON 到指定的 Go 结构体
  39. // target 必须是一个指向目标结构体实例的指针 (e.g., &MyStruct{})
  40. func (r *ChatResult) ParseContentAs(target any) error {
  41. content, err := r.GetContentString()
  42. if err != nil {
  43. return fmt.Errorf("parseContent err: %s", err)
  44. } else if content == "" {
  45. return errors.New("parseContent err: content is empty or unavailable")
  46. }
  47. if !IsOpenaiModel(r.Model) { //不支持Response Schema的要特殊处理一下
  48. var isJsonContent bool
  49. content, isJsonContent = ExtractJSONContent(content)
  50. if !isJsonContent {
  51. return errors.New("invalid json content")
  52. }
  53. }
  54. err = json.Unmarshal([]byte(content), target)
  55. if err != nil {
  56. return fmt.Errorf("parseContent err: failed to unmarshal content JSON "+
  57. "into target type '%w'", err)
  58. }
  59. return nil
  60. }
  61. func AnyToBytes(in any) ([]byte, error) {
  62. switch v := in.(type) {
  63. case string:
  64. return []byte(v), nil
  65. case []byte:
  66. return v, nil
  67. default:
  68. return json.Marshal(v)
  69. }
  70. }
  71. func AnyToCompOpenApiResp(in any) (*types.CompOpenApiResp, error) {
  72. if resp, ok := in.(*types.CompOpenApiResp); ok {
  73. return resp, nil
  74. }
  75. if resp, ok := in.(types.CompOpenApiResp); ok {
  76. return &resp, nil
  77. }
  78. bs, err := AnyToBytes(in)
  79. if err != nil {
  80. return nil, err
  81. }
  82. nresp := &types.CompOpenApiResp{}
  83. err = json.Unmarshal(bs, nresp)
  84. if err != nil {
  85. return nil, err
  86. }
  87. return nresp, nil
  88. }
  89. func AnyToCompApiReq(in any) (*types.CompApiReq, error) {
  90. if req, ok := in.(*types.CompApiReq); ok {
  91. return req, nil
  92. }
  93. if req, ok := in.(types.CompApiReq); ok {
  94. return &req, nil
  95. }
  96. bs, err := AnyToBytes(in)
  97. if err != nil {
  98. return nil, err
  99. }
  100. nreq := &types.CompApiReq{}
  101. err = json.Unmarshal(bs, nreq)
  102. if err != nil {
  103. return nil, err
  104. }
  105. return nreq, nil
  106. }
  107. func CheckJSON(input any, checkEmpty bool) (bool, error) {
  108. inputBytes, err := AnyToBytes(input)
  109. if err != nil {
  110. return false, err
  111. }
  112. var raw json.RawMessage
  113. err = json.Unmarshal(inputBytes, &raw)
  114. if err != nil {
  115. return false, fmt.Errorf("input is not valid JSON: %w", err)
  116. }
  117. if checkEmpty {
  118. trimmed := bytes.TrimSpace(inputBytes)
  119. if len(trimmed) == 0 {
  120. return false, fmt.Errorf("input is empty")
  121. }
  122. }
  123. return true, nil
  124. }
  125. func WrapJSON(input any, warpKey string, checkValid bool) ([]byte, error) {
  126. var (
  127. inputBytes []byte
  128. outputBytes []byte
  129. err error
  130. )
  131. if inputBytes, err = AnyToBytes(input); err != nil {
  132. return nil, err
  133. }
  134. if checkValid {
  135. if _, err = CheckJSON(inputBytes, true); err != nil {
  136. return nil, err
  137. }
  138. }
  139. if len(warpKey) == 0 {
  140. return inputBytes, nil
  141. }
  142. wrapper := map[string]json.RawMessage{
  143. warpKey: json.RawMessage(inputBytes),
  144. }
  145. if outputBytes, err = json.Marshal(wrapper); err != nil {
  146. return nil, fmt.Errorf("failed to marshal wrapper structure: %w", err)
  147. }
  148. return outputBytes, nil
  149. }
  150. func ParseContentAs(content string, target any) error {
  151. // 清理可能的 Markdown ```json ``` 包装
  152. if strings.HasPrefix(content, "```json") && strings.HasSuffix(content, "```") {
  153. content = strings.TrimSuffix(strings.TrimPrefix(content, "```json"), "```")
  154. content = strings.TrimSpace(content)
  155. }
  156. if err := json.Unmarshal([]byte(content), target); err != nil {
  157. return fmt.Errorf("parseContent err:failed to unmarshal"+
  158. " content JSON into target type '%w'", err)
  159. }
  160. return nil
  161. }
  162. func ExtractJSONContent(s string) (string, bool) {
  163. startMarker := "```json"
  164. endMarker := "```"
  165. // 寻找起始标记
  166. startIdx := strings.Index(s, startMarker)
  167. if startIdx == -1 {
  168. return "", false // 没有起始标记
  169. }
  170. // 寻找结束标记(需在起始标记之后查找)
  171. endIdx := strings.LastIndex(s, endMarker)
  172. if endIdx == -1 || endIdx <= startIdx {
  173. return "", false // 没有结束标记或标记顺序错误
  174. }
  175. // 计算内容范围
  176. contentStart := startIdx + len(startMarker)
  177. contentEnd := endIdx
  178. // 提取内容并去除前后空白
  179. content := strings.TrimSpace(s[contentStart:contentEnd])
  180. // 若内容为空视为无效
  181. if content == "" {
  182. return "", false
  183. }
  184. return content, true
  185. }