process_request.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. package process
  2. import (
  3. "fmt"
  4. "strings"
  5. "time"
  6. "github.com/eryajf/chatgpt-dingtalk/pkg/db"
  7. "github.com/eryajf/chatgpt-dingtalk/pkg/dingbot"
  8. "github.com/eryajf/chatgpt-dingtalk/pkg/logger"
  9. "github.com/eryajf/chatgpt-dingtalk/public"
  10. "github.com/solywsh/chatgpt"
  11. )
  12. // ProcessRequest 分析处理请求逻辑
  13. func ProcessRequest(rmsg *dingbot.ReceiveMsg) error {
  14. if public.CheckRequest(rmsg) {
  15. content := strings.TrimSpace(rmsg.Text.Content)
  16. switch content {
  17. case "单聊":
  18. public.UserService.SetUserMode(rmsg.GetSenderIdentifier(), content)
  19. _, err := rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("=====现在进入与👉%s👈单聊的模式 =====", rmsg.SenderNick))
  20. if err != nil {
  21. logger.Warning(fmt.Errorf("send message error: %v", err))
  22. }
  23. case "串聊":
  24. public.UserService.SetUserMode(rmsg.GetSenderIdentifier(), content)
  25. _, err := rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("=====现在进入与👉%s👈串聊的模式 =====", rmsg.SenderNick))
  26. if err != nil {
  27. logger.Warning(fmt.Errorf("send message error: %v", err))
  28. }
  29. case "重置":
  30. // 重置用户对话模式
  31. public.UserService.ClearUserMode(rmsg.GetSenderIdentifier())
  32. // 清空用户对话上下文
  33. public.UserService.ClearUserSessionContext(rmsg.GetSenderIdentifier())
  34. // 清空用户对话的答案ID
  35. public.UserService.ClearAnswerID(rmsg.SenderNick, rmsg.GetChatTitle())
  36. _, err := rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("=====已重置与👉%s👈的对话模式,可以开始新的对话=====", rmsg.SenderNick))
  37. if err != nil {
  38. logger.Warning(fmt.Errorf("send message error: %v", err))
  39. }
  40. case "模板":
  41. var title string
  42. for _, v := range *public.Prompt {
  43. title = title + v.Title + " | "
  44. }
  45. _, err := rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), fmt.Sprintf("%s 您好,当前程序内置集成了这些提示词:\n\n-----\n\n| %s \n\n-----\n\n您可以选择某个提示词作为对话内容的开头。\n\n以周报为例,可发送\"#周报 我本周用Go写了一个钉钉集成ChatGPT的聊天应用\",可将工作内容填充为一篇完整的周报。\n\n-----\n\n若您不清楚某个提示词的所代表的含义,您可以直接发送提示词,例如直接发送\"#周报\"", rmsg.SenderNick, title))
  46. if err != nil {
  47. logger.Warning(fmt.Errorf("send message error: %v", err))
  48. }
  49. case "图片":
  50. _, err := rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), "发送以 **#图片** 开头的内容,将会触发绘画能力,图片生成之后,将会保存在程序根目录下的 **images目录** \n 如果你绘图没有思路,可以在这两个网站寻找灵感。\n - [https://lexica.art/](https://lexica.art/)\n- [https://www.clickprompt.org/zh-CN/](https://www.clickprompt.org/zh-CN/)")
  51. if err != nil {
  52. logger.Warning(fmt.Errorf("send message error: %v", err))
  53. }
  54. case "余额":
  55. // cacheMsg := public.UserService.GetUserMode("system_balance")
  56. // if cacheMsg == "" {
  57. // rst, err := public.GetBalance()
  58. // if err != nil {
  59. // logger.Warning(fmt.Errorf("get balance error: %v", err))
  60. // return err
  61. // }
  62. // t1 := time.Unix(int64(rst.Grants.Data[0].EffectiveAt), 0)
  63. // t2 := time.Unix(int64(rst.Grants.Data[0].ExpiresAt), 0)
  64. // 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"))
  65. // }
  66. cacheMsg := "官方暂时改写了余额接口,因此暂不提供查询余额功能!2023-04-03"
  67. _, err := rmsg.ReplyToDingtalk(string(dingbot.TEXT), cacheMsg)
  68. if err != nil {
  69. logger.Warning(fmt.Errorf("send message error: %v", err))
  70. }
  71. case "查对话":
  72. msg := "使用如下指令进行查询:\n\n---\n\n**#查对话 username:张三**\n\n---\n\n需要注意格式必须严格与上边一致,否则会查询失败\n\n只有钉钉管理员,程序系统管理员,与查自己的情况下,才会被允许"
  73. _, err := rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), msg)
  74. if err != nil {
  75. logger.Warning(fmt.Errorf("send message error: %v", err))
  76. }
  77. default:
  78. if public.FirstCheck(rmsg) {
  79. return Do("串聊", rmsg)
  80. } else {
  81. return Do("单聊", rmsg)
  82. }
  83. }
  84. }
  85. return nil
  86. }
  87. // 执行处理请求
  88. func Do(mode string, rmsg *dingbot.ReceiveMsg) error {
  89. // 先把模式注入
  90. public.UserService.SetUserMode(rmsg.GetSenderIdentifier(), mode)
  91. switch mode {
  92. case "单聊":
  93. qObj := db.Chat{
  94. Username: rmsg.SenderNick,
  95. Source: rmsg.GetChatTitle(),
  96. ChatType: db.Q,
  97. ParentContent: 0,
  98. Content: rmsg.Text.Content,
  99. }
  100. qid, err := qObj.Add()
  101. if err != nil {
  102. logger.Error("往MySQL新增数据失败,错误信息:", err)
  103. }
  104. reply, err := chatgpt.SingleQa(rmsg.Text.Content, rmsg.GetSenderIdentifier())
  105. if err != nil {
  106. logger.Info(fmt.Errorf("gpt request error: %v", err))
  107. if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
  108. public.UserService.ClearUserSessionContext(rmsg.GetSenderIdentifier())
  109. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err))
  110. if err != nil {
  111. logger.Warning(fmt.Errorf("send message error: %v", err))
  112. return err
  113. }
  114. } else {
  115. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("请求openai失败了,错误信息:%v", err))
  116. if err != nil {
  117. logger.Warning(fmt.Errorf("send message error: %v", err))
  118. return err
  119. }
  120. }
  121. }
  122. if reply == "" {
  123. logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
  124. return nil
  125. } else {
  126. reply = strings.TrimSpace(reply)
  127. reply = strings.Trim(reply, "\n")
  128. aObj := db.Chat{
  129. Username: rmsg.SenderNick,
  130. Source: rmsg.GetChatTitle(),
  131. ChatType: db.A,
  132. ParentContent: qid,
  133. Content: reply,
  134. }
  135. _, err := aObj.Add()
  136. if err != nil {
  137. logger.Error("往MySQL新增数据失败,错误信息:", err)
  138. }
  139. logger.Info(fmt.Sprintf("🤖 %s得到的答案: %#v", rmsg.SenderNick, reply))
  140. // 回复@我的用户
  141. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), reply)
  142. if err != nil {
  143. logger.Warning(fmt.Errorf("send message error: %v", err))
  144. return err
  145. }
  146. }
  147. case "串聊":
  148. lastAid := public.UserService.GetAnswerID(rmsg.SenderNick, rmsg.GetChatTitle())
  149. qObj := db.Chat{
  150. Username: rmsg.SenderNick,
  151. Source: rmsg.GetChatTitle(),
  152. ChatType: db.Q,
  153. ParentContent: lastAid,
  154. Content: rmsg.Text.Content,
  155. }
  156. qid, err := qObj.Add()
  157. if err != nil {
  158. logger.Error("往MySQL新增数据失败,错误信息:", err)
  159. }
  160. cli, reply, err := chatgpt.ContextQa(rmsg.Text.Content, rmsg.GetSenderIdentifier())
  161. if err != nil {
  162. logger.Info(fmt.Sprintf("gpt request error: %v", err))
  163. if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
  164. public.UserService.ClearUserSessionContext(rmsg.GetSenderIdentifier())
  165. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err))
  166. if err != nil {
  167. logger.Warning(fmt.Errorf("send message error: %v", err))
  168. return err
  169. }
  170. } else {
  171. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("请求openai失败了,错误信息:%v", err))
  172. if err != nil {
  173. logger.Warning(fmt.Errorf("send message error: %v", err))
  174. return err
  175. }
  176. }
  177. }
  178. if reply == "" {
  179. logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
  180. return nil
  181. } else {
  182. reply = strings.TrimSpace(reply)
  183. reply = strings.Trim(reply, "\n")
  184. aObj := db.Chat{
  185. Username: rmsg.SenderNick,
  186. Source: rmsg.GetChatTitle(),
  187. ChatType: db.A,
  188. ParentContent: qid,
  189. Content: reply,
  190. }
  191. aid, err := aObj.Add()
  192. if err != nil {
  193. logger.Error("往MySQL新增数据失败,错误信息:", err)
  194. }
  195. // 将当前回答的ID放入缓存
  196. public.UserService.SetAnswerID(rmsg.SenderNick, rmsg.GetChatTitle(), aid)
  197. logger.Info(fmt.Sprintf("🤖 %s得到的答案: %#v", rmsg.SenderNick, reply))
  198. // 回复@我的用户
  199. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), reply)
  200. if err != nil {
  201. logger.Warning(fmt.Errorf("send message error: %v", err))
  202. return err
  203. }
  204. _ = cli.ChatContext.SaveConversation(rmsg.GetSenderIdentifier())
  205. }
  206. default:
  207. }
  208. return nil
  209. }
  210. func ImageGenerate(rmsg *dingbot.ReceiveMsg) error {
  211. qObj := db.Chat{
  212. Username: rmsg.SenderNick,
  213. Source: rmsg.GetChatTitle(),
  214. ChatType: db.Q,
  215. ParentContent: 0,
  216. Content: rmsg.Text.Content,
  217. }
  218. qid, err := qObj.Add()
  219. if err != nil {
  220. logger.Error("往MySQL新增数据失败,错误信息:", err)
  221. }
  222. reply, err := chatgpt.ImageQa(rmsg.Text.Content, rmsg.GetSenderIdentifier())
  223. if err != nil {
  224. logger.Info(fmt.Errorf("gpt request error: %v", err))
  225. _, err = rmsg.ReplyToDingtalk(string(dingbot.TEXT), fmt.Sprintf("请求openai失败了,错误信息:%v", err))
  226. if err != nil {
  227. logger.Error(fmt.Errorf("send message error: %v", err))
  228. return err
  229. }
  230. }
  231. if reply == "" {
  232. logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
  233. return nil
  234. } else {
  235. reply = strings.TrimSpace(reply)
  236. reply = strings.Trim(reply, "\n")
  237. reply = fmt.Sprintf(">点击图片可旋转或放大。\n![](%s)", reply)
  238. aObj := db.Chat{
  239. Username: rmsg.SenderNick,
  240. Source: rmsg.GetChatTitle(),
  241. ChatType: db.A,
  242. ParentContent: qid,
  243. Content: reply,
  244. }
  245. _, err := aObj.Add()
  246. if err != nil {
  247. logger.Error("往MySQL新增数据失败,错误信息:", err)
  248. }
  249. logger.Info(fmt.Sprintf("🤖 %s得到的答案: %#v", rmsg.SenderNick, reply))
  250. // 回复@我的用户
  251. _, err = rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), reply)
  252. if err != nil {
  253. logger.Error(fmt.Errorf("send message error: %v", err))
  254. return err
  255. }
  256. }
  257. return nil
  258. }
  259. func SelectHistory(rmsg *dingbot.ReceiveMsg) error {
  260. name := strings.TrimSpace(strings.Split(rmsg.Text.Content, ":")[1])
  261. if !rmsg.IsAdmin && name != rmsg.SenderNick && !public.JudgeAdminUsers(rmsg.SenderNick) {
  262. _, err := rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), "**🤷 抱歉,您没有权限查询其他人的对话记录!**")
  263. if err != nil {
  264. logger.Error(fmt.Errorf("send message error: %v", err))
  265. return err
  266. }
  267. return nil
  268. }
  269. // 获取数据列表
  270. var chat db.Chat
  271. if !chat.Exist(map[string]interface{}{"username": name}) {
  272. _, err := rmsg.ReplyToDingtalk(string(dingbot.TEXT), "用户名错误,这个用户不存在,请核实之后再进行查询")
  273. if err != nil {
  274. logger.Error(fmt.Errorf("send message error: %v", err))
  275. return err
  276. }
  277. return fmt.Errorf("用户名错误,这个用户不存在,请核实之后重新查询")
  278. }
  279. chats, err := chat.List(db.ChatListReq{
  280. Username: name,
  281. })
  282. if err != nil {
  283. return err
  284. }
  285. var rst string
  286. for _, chatTmp := range chats {
  287. ctime := chatTmp.CreatedAt.Format("2006-01-02 15:04:05")
  288. if chatTmp.ChatType == 1 {
  289. rst += fmt.Sprintf("## 🙋 %s 问\n\n**时间:** %v\n\n**问题为:** %s\n\n", chatTmp.Username, ctime, chatTmp.Content)
  290. } else {
  291. rst += fmt.Sprintf("## 🤖 机器人答\n\n**时间:** %v\n\n**回答如下:** \n\n%s\n\n", ctime, chatTmp.Content)
  292. }
  293. // TODO: 答案应该严格放在问题之后,目前只根据ID排序进行的陈列,当一个用户同时提出多个问题时,最终展示的可能会有点问题
  294. }
  295. fileName := time.Now().Format("20060102-150405") + ".md"
  296. // 写入文件
  297. if err = public.WriteToFile("./data/chatHistory/"+fileName, []byte(rst)); err != nil {
  298. return err
  299. }
  300. // 回复@我的用户
  301. reply := fmt.Sprintf("- 在线查看: [点我](%s)\n- 下载文件: [点我](%s)\n- 在线预览请安装插件:[Markdown Preview Plus](https://chrome.google.com/webstore/detail/markdown-preview-plus/febilkbfcbhebfnokafefeacimjdckgl)", public.Config.ServiceURL+"/history/"+fileName, public.Config.ServiceURL+"/download/"+fileName)
  302. logger.Info(fmt.Sprintf("🤖 %s得到的答案: %#v", rmsg.SenderNick, reply))
  303. _, err = rmsg.ReplyToDingtalk(string(dingbot.MARKDOWN), reply)
  304. if err != nil {
  305. logger.Error(fmt.Errorf("send message error: %v", err))
  306. return err
  307. }
  308. return nil
  309. }