main.go 6.8 KB

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