123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230 |
- package main
- import (
- "encoding/json"
- "fmt"
- "io/ioutil"
- "net/http"
- "strings"
- "time"
- "github.com/eryajf/chatgpt-dingtalk/public"
- "github.com/eryajf/chatgpt-dingtalk/public/logger"
- "github.com/solywsh/chatgpt"
- )
- func init() {
- public.InitSvc()
- }
- func main() {
- Start()
- }
- var Welcome string = `Commands:
- =================================
- 🙋 单聊 👉 单独聊天
- 📣 串聊 👉 带上下文聊天
- 🔃 重置 👉 重置带上下文聊天
- 💵 余额 👉 查询剩余额度
- 🚀 帮助 👉 显示帮助信息
- =================================
- 🚜 例:@我发送 空 或 帮助 将返回此帮助信息
- 💪 Power By https://github.com/eryajf/chatgpt-dingtalk
- `
- // 💵 余额 👉 查看接口可调用额度
- func Start() {
- // 定义一个处理器函数
- handler := func(w http.ResponseWriter, r *http.Request) {
- data, err := ioutil.ReadAll(r.Body)
- if err != nil {
- http.Error(w, err.Error(), http.StatusBadRequest)
- logger.Warning(fmt.Sprintf("read request body failed: %v\n", err.Error()))
- return
- }
- if len(data) == 0 {
- logger.Warning("回调参数为空,以至于无法正常解析,请检查原因")
- return
- }
- var msgObj = new(public.ReceiveMsg)
- err = json.Unmarshal(data, &msgObj)
- if err != nil {
- logger.Warning(fmt.Errorf("unmarshal request body failed: %v", err))
- }
- if msgObj.Text.Content == "" || msgObj.ChatbotUserID == "" {
- logger.Warning("从钉钉回调过来的内容为空,根据过往的经验,或许重新创建一下机器人,能解决这个问题")
- return
- }
- // TODO: 校验请求
- if len(msgObj.Text.Content) == 1 || strings.TrimSpace(msgObj.Text.Content) == "帮助" {
- // 欢迎信息
- _, err := msgObj.ReplyText(Welcome, msgObj.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- }
- } else {
- logger.Info(fmt.Sprintf("dingtalk callback parameters: %#v", msgObj))
- err = ProcessRequest(*msgObj)
- if err != nil {
- logger.Warning(fmt.Errorf("process request failed: %v", err))
- }
- }
- }
- // 创建一个新的 HTTP 服务器
- server := &http.Server{
- Addr: ":8090",
- Handler: http.HandlerFunc(handler),
- }
- // 启动服务器
- logger.Info("Start Listen On ", server.Addr)
- err := server.ListenAndServe()
- if err != nil {
- logger.Danger(err)
- }
- }
- func ProcessRequest(rmsg public.ReceiveMsg) error {
- content := strings.TrimSpace(rmsg.Text.Content)
- switch content {
- case "单聊":
- public.UserService.SetUserMode(rmsg.SenderStaffId, content)
- _, err := rmsg.ReplyText(fmt.Sprintf("=====现在进入与👉%s👈单聊的模式 =====", rmsg.SenderNick), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- }
- case "串聊":
- public.UserService.SetUserMode(rmsg.SenderStaffId, content)
- _, err := rmsg.ReplyText(fmt.Sprintf("=====现在进入与👉%s👈串聊的模式 =====", rmsg.SenderNick), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- }
- case "重置":
- public.UserService.ClearUserMode(rmsg.SenderStaffId)
- public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
- _, err := rmsg.ReplyText(fmt.Sprintf("=====已重置与👉%s👈的对话模式,可以开始新的对话=====", rmsg.SenderNick), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- }
- case "余额":
- cacheMsg := public.UserService.GetUserMode("system_balance")
- if cacheMsg == "" {
- rst, err := public.GetBalance()
- if err != nil {
- logger.Warning(fmt.Errorf("get balance error: %v", err))
- return err
- }
- t1 := time.Unix(int64(rst.Grants.Data[0].EffectiveAt), 0)
- t2 := time.Unix(int64(rst.Grants.Data[0].ExpiresAt), 0)
- 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"))
- }
- _, err := rmsg.ReplyText(cacheMsg, rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- }
- default:
- if public.FirstCheck(rmsg) {
- return Do("串聊", rmsg)
- } else {
- return Do("单聊", rmsg)
- }
- }
- return nil
- }
- func Do(mode string, rmsg public.ReceiveMsg) error {
- // 先把模式注入
- public.UserService.SetUserMode(rmsg.SenderStaffId, mode)
- switch mode {
- case "单聊":
- reply, err := SingleQa(rmsg.Text.Content, rmsg.SenderStaffId)
- if err != nil {
- logger.Info(fmt.Errorf("gpt request error: %v", err))
- if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
- public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
- _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- } else {
- _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v", err), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- }
- }
- if reply == "" {
- logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
- return nil
- } else {
- reply = strings.TrimSpace(reply)
- reply = strings.Trim(reply, "\n")
- // 回复@我的用户
- // fmt.Println("单聊结果是:", reply)
- _, err = rmsg.ReplyText(reply, rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- }
- case "串聊":
- cli, reply, err := ContextQa(rmsg.Text.Content, rmsg.SenderStaffId)
- if err != nil {
- logger.Info(fmt.Sprintf("gpt request error: %v", err))
- if strings.Contains(fmt.Sprintf("%v", err), "maximum text length exceeded") {
- public.UserService.ClearUserSessionContext(rmsg.SenderStaffId)
- _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v,看起来是超过最大对话限制了,已自动重置您的对话", err), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- } else {
- _, err = rmsg.ReplyText(fmt.Sprintf("请求openai失败了,错误信息:%v", err), rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- }
- }
- if reply == "" {
- logger.Warning(fmt.Errorf("get gpt result falied: %v", err))
- return nil
- } else {
- reply = strings.TrimSpace(reply)
- reply = strings.Trim(reply, "\n")
- // 回复@我的用户
- _, err = rmsg.ReplyText(reply, rmsg.SenderStaffId)
- if err != nil {
- logger.Warning(fmt.Errorf("send message error: %v", err))
- return err
- }
- _ = cli.ChatContext.SaveConversation(rmsg.SenderStaffId)
- }
- default:
- }
- return nil
- }
- func SingleQa(question, userId string) (answer string, err error) {
- chat := chatgpt.New(public.Config.ApiKey, public.Config.HttpProxy, userId, public.Config.SessionTimeout)
- defer chat.Close()
- return chat.ChatWithContext(question)
- }
- func ContextQa(question, userId string) (chat *chatgpt.ChatGPT, answer string, err error) {
- chat = chatgpt.New(public.Config.ApiKey, public.Config.HttpProxy, userId, public.Config.SessionTimeout)
- if public.UserService.GetUserSessionContext(userId) != "" {
- err = chat.ChatContext.LoadConversation(userId)
- if err != nil {
- fmt.Printf("load station failed: %v\n", err)
- }
- }
- answer, err = chat.ChatWithContext(question)
- return
- }
|