main.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233
  1. package main
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "io/ioutil"
  6. "net/http"
  7. "strings"
  8. "time"
  9. "github.com/eryajf/chatgpt-dingtalk/config"
  10. "github.com/eryajf/chatgpt-dingtalk/public"
  11. "github.com/eryajf/chatgpt-dingtalk/public/logger"
  12. "github.com/solywsh/chatgpt"
  13. )
  14. func init() {
  15. public.InitSvc()
  16. }
  17. func main() {
  18. Start()
  19. }
  20. var Welcome string = `Commands:
  21. =================================
  22. 🙋 单聊 👉 单独聊天
  23. 📣 串聊 👉 带上下文聊天
  24. 🔃 重置 👉 重置带上下文聊天
  25. 💵 余额 👉 查询剩余额度
  26. 🚀 帮助 👉 显示帮助信息
  27. =================================
  28. 🚜 例:@我发送 空 或 帮助 将返回此帮助信息
  29. 💪 Power By https://github.com/eryajf/chatgpt-dingtalk
  30. `
  31. // 💵 余额 👉 查看接口可调用额度
  32. func Start() {
  33. // 定义一个处理器函数
  34. handler := func(w http.ResponseWriter, r *http.Request) {
  35. data, err := ioutil.ReadAll(r.Body)
  36. if err != nil {
  37. http.Error(w, err.Error(), http.StatusBadRequest)
  38. logger.Warning(fmt.Sprintf("read request body failed: %v\n", err.Error()))
  39. return
  40. }
  41. if len(data) == 0 {
  42. logger.Warning("回调参数为空,以至于无法正常解析,请检查原因")
  43. return
  44. }
  45. var msgObj = new(public.ReceiveMsg)
  46. err = json.Unmarshal(data, &msgObj)
  47. if err != nil {
  48. logger.Warning(fmt.Errorf("unmarshal request body failed: %v", err))
  49. }
  50. if msgObj.Text.Content == "" || msgObj.ChatbotUserID == "" {
  51. logger.Warning("从钉钉回调过来的内容为空,根据过往的经验,或许重新创建一下机器人,能解决这个问题")
  52. return
  53. }
  54. // TODO: 校验请求
  55. if len(msgObj.Text.Content) == 1 || strings.TrimSpace(msgObj.Text.Content) == "帮助" {
  56. // 欢迎信息
  57. _, err := msgObj.ReplyText(Welcome, msgObj.SenderStaffId)
  58. if err != nil {
  59. logger.Warning(fmt.Errorf("send message error: %v", err))
  60. }
  61. } else {
  62. logger.Info(fmt.Sprintf("dingtalk callback parameters: %#v", msgObj))
  63. err = ProcessRequest(*msgObj)
  64. if err != nil {
  65. logger.Warning(fmt.Errorf("process request failed: %v", err))
  66. }
  67. }
  68. }
  69. // 创建一个新的 HTTP 服务器
  70. server := &http.Server{
  71. Addr: ":8090",
  72. Handler: http.HandlerFunc(handler),
  73. }
  74. // 启动服务器
  75. logger.Info("Start Listen On ", server.Addr)
  76. err := server.ListenAndServe()
  77. if err != nil {
  78. logger.Danger(err)
  79. }
  80. }
  81. func ProcessRequest(rmsg public.ReceiveMsg) error {
  82. content := strings.TrimSpace(rmsg.Text.Content)
  83. switch content {
  84. case "单聊":
  85. public.UserService.SetUserMode(rmsg.SenderStaffId, content)
  86. _, err := rmsg.ReplyText(fmt.Sprintf("=====现在进入与👉%s👈单聊的模式 =====", rmsg.SenderNick), rmsg.SenderStaffId)
  87. if err != nil {
  88. logger.Warning(fmt.Errorf("send message error: %v", err))
  89. }
  90. case "串聊":
  91. public.UserService.SetUserMode(rmsg.SenderStaffId, content)
  92. _, err := rmsg.ReplyText(fmt.Sprintf("=====现在进入与👉%s👈串聊的模式 =====", rmsg.SenderNick), rmsg.SenderStaffId)
  93. if err != nil {
  94. logger.Warning(fmt.Errorf("send message error: %v", err))
  95. }
  96. case "重置":
  97. public.UserService.ClearUserMode(rmsg.SenderStaffId)
  98. public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
  99. _, err := rmsg.ReplyText(fmt.Sprintf("=====已重置与👉%s👈的对话模式,可以开始新的对话=====", rmsg.SenderNick), rmsg.SenderStaffId)
  100. if err != nil {
  101. logger.Warning(fmt.Errorf("send message error: %v", err))
  102. }
  103. case "余额":
  104. cacheMsg := public.UserService.GetUserMode("system_balance")
  105. if cacheMsg == "" {
  106. rst, err := public.GetBalance()
  107. if err != nil {
  108. logger.Warning(fmt.Errorf("get balance error: %v", err))
  109. return err
  110. }
  111. t1 := time.Unix(int64(rst.Grants.Data[0].EffectiveAt), 0)
  112. t2 := time.Unix(int64(rst.Grants.Data[0].ExpiresAt), 0)
  113. cacheMsg = fmt.Sprintf("💵 已用: 💲%v\n💵 剩余: 💲%v\n⏳ 有效时间: 从 %v 到 %v\n", fmt.Sprintf("%.2f", rst.TotalUsed), fmt.Sprintf("%.2f", rst.TotalAvailable), t1.Format("2006-01-02 15:04:05"), t2.Format("2006-01-02 15:04:05"))
  114. }
  115. _, err := rmsg.ReplyText(cacheMsg, rmsg.SenderStaffId)
  116. if err != nil {
  117. logger.Warning(fmt.Errorf("send message error: %v", err))
  118. }
  119. default:
  120. if public.FirstCheck(rmsg) {
  121. return Do("串聊", rmsg)
  122. } else {
  123. return Do("单聊", rmsg)
  124. }
  125. }
  126. return nil
  127. }
  128. func Do(mode string, rmsg public.ReceiveMsg) error {
  129. // 先把模式注入
  130. public.UserService.SetUserMode(rmsg.SenderStaffId, mode)
  131. switch mode {
  132. case "单聊":
  133. reply, err := SingleQa(rmsg.Text.Content, rmsg.SenderStaffId)
  134. if err != nil {
  135. logger.Info(fmt.Errorf("gpt request error: %v", err))
  136. if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
  137. public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
  138. _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err), rmsg.SenderStaffId)
  139. if err != nil {
  140. logger.Warning(fmt.Errorf("send message error: %v", err))
  141. return err
  142. }
  143. } else {
  144. _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v", err), rmsg.SenderStaffId)
  145. if err != nil {
  146. logger.Warning(fmt.Errorf("send message error: %v", err))
  147. return err
  148. }
  149. }
  150. }
  151. if reply == "" {
  152. logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
  153. return nil
  154. } else {
  155. reply = strings.TrimSpace(reply)
  156. reply = strings.Trim(reply, "\n")
  157. // 回复@我的用户
  158. // fmt.Println("单聊结果是:", reply)
  159. _, err = rmsg.ReplyText(reply, rmsg.SenderStaffId)
  160. if err != nil {
  161. logger.Warning(fmt.Errorf("send message error: %v", err))
  162. return err
  163. }
  164. }
  165. case "串聊":
  166. cli, reply, err := ContextQa(rmsg.Text.Content, rmsg.SenderStaffId)
  167. if err != nil {
  168. logger.Info(fmt.Sprintf("gpt request error: %v", err))
  169. if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
  170. public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
  171. _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err), rmsg.SenderStaffId)
  172. if err != nil {
  173. logger.Warning(fmt.Errorf("send message error: %v", err))
  174. return err
  175. }
  176. } else {
  177. _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v", err), rmsg.SenderStaffId)
  178. if err != nil {
  179. logger.Warning(fmt.Errorf("send message error: %v", err))
  180. return err
  181. }
  182. }
  183. }
  184. if reply == "" {
  185. logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
  186. return nil
  187. } else {
  188. reply = strings.TrimSpace(reply)
  189. reply = strings.Trim(reply, "\n")
  190. // 回复@我的用户
  191. _, err = rmsg.ReplyText(reply, rmsg.SenderStaffId)
  192. if err != nil {
  193. logger.Warning(fmt.Errorf("send message error: %v", err))
  194. return err
  195. }
  196. _ = cli.ChatContext.SaveConversation(rmsg.SenderStaffId)
  197. }
  198. default:
  199. }
  200. return nil
  201. }
  202. func SingleQa(question, userId string) (answer string, err error) {
  203. cfg := config.LoadConfig()
  204. chat := chatgpt.New(cfg.ApiKey, cfg.HttpProxy, userId, cfg.SessionTimeout)
  205. defer chat.Close()
  206. return chat.ChatWithContext(question)
  207. }
  208. func ContextQa(question, userId string) (chat *chatgpt.ChatGPT, answer string, err error) {
  209. cfg := config.LoadConfig()
  210. chat = chatgpt.New(cfg.ApiKey, cfg.HttpProxy, userId, cfg.SessionTimeout)
  211. if public.UserService.GetUserSessionContext(userId) != "" {
  212. err = chat.ChatContext.LoadConversation(userId)
  213. if err != nil {
  214. fmt.Printf("load station failed: %v\n", err)
  215. }
  216. }
  217. answer, err = chat.ChatWithContext(question)
  218. return
  219. }