package main import ( "context" "fmt" "net/http" "os" "os/signal" "path/filepath" "strings" "time" "github.com/eryajf/chatgpt-dingtalk/pkg/dingbot" "github.com/eryajf/chatgpt-dingtalk/pkg/logger" "github.com/eryajf/chatgpt-dingtalk/pkg/process" "github.com/eryajf/chatgpt-dingtalk/public" "github.com/xgfone/ship/v5" ) func init() { public.InitSvc() logger.InitLogger(public.Config.LogLevel) } func main() { Start() } func Start() { app := ship.Default() app.Route("/").POST(func(c *ship.Context) error { var msgObj dingbot.ReceiveMsg err := c.Bind(&msgObj) if err != nil { return ship.ErrBadRequest.New(fmt.Errorf("bind to receivemsg failed : %v", err)) } // 先校验回调是否合法 if !public.CheckRequest(c.GetReqHeader("timestamp"), c.GetReqHeader("sign")) { logger.Warning("该请求不合法,可能是其他企业或者未经允许的应用调用所致,请知悉!") return nil } // 再校验回调参数是否有价值 if msgObj.Text.Content == "" || msgObj.ChatbotUserID == "" { logger.Warning("从钉钉回调过来的内容为空,根据过往的经验,或许重新创建一下机器人,能解决这个问题") return ship.ErrBadRequest.New(fmt.Errorf("从钉钉回调过来的内容为空,根据过往的经验,或许重新创建一下机器人,能解决这个问题")) } // 去除问题的前后空格 msgObj.Text.Content = strings.TrimSpace(msgObj.Text.Content) // 打印钉钉回调过来的请求明细,调试时打开 fmt.Println("=======", logger.Logger.GetLevel().String()) logger.Debug(fmt.Sprintf("dingtalk callback parameters: %#v", msgObj)) if public.Config.ChatType != "0" && msgObj.ConversationType != public.Config.ChatType { _, err = msgObj.ReplyToDingtalk(string(dingbot.TEXT), "抱歉,管理员禁用了这种聊天方式,请选择其他聊天方式与机器人对话!") if err != nil { logger.Warning(fmt.Errorf("send message error: %v", err)) return err } return nil } if !public.JudgeGroup(msgObj.GetChatTitle()) && !public.JudgeUsers(msgObj.SenderNick) && !public.JudgeAdminUsers(msgObj.SenderNick) { _, err = msgObj.ReplyToDingtalk(string(dingbot.TEXT), "抱歉,您不在该机器人对话功能的白名单当中!") if err != nil { logger.Warning(fmt.Errorf("send message error: %v", err)) return err } return nil } if len(msgObj.Text.Content) == 1 || msgObj.Text.Content == "帮助" { // 欢迎信息 _, err := msgObj.ReplyToDingtalk(string(dingbot.MARKDOWN), public.Welcome) if err != nil { logger.Warning(fmt.Errorf("send message error: %v", err)) return ship.ErrBadRequest.New(fmt.Errorf("send message error: %v", err)) } } else { logger.Info(fmt.Sprintf("🙋 %s发起的问题: %#v", msgObj.SenderNick, msgObj.Text.Content)) // 除去帮助之外的逻辑分流在这里处理 switch { case strings.HasPrefix(msgObj.Text.Content, "#图片"): return process.ImageGenerate(&msgObj) case strings.HasPrefix(msgObj.Text.Content, "#查对话"): return process.SelectHistory(&msgObj) case strings.HasPrefix(msgObj.Text.Content, "#域名"): return process.DomainMsg(&msgObj) case strings.HasPrefix(msgObj.Text.Content, "#证书"): return process.DomainCertMsg(&msgObj) default: msgObj.Text.Content, err = process.GeneratePrompt(msgObj.Text.Content) // err不为空:提示词之后没有文本 -> 直接返回提示词所代表的内容 if err != nil { _, err = msgObj.ReplyToDingtalk(string(dingbot.TEXT), msgObj.Text.Content) if err != nil { logger.Warning(fmt.Errorf("send message error: %v", err)) return err } return nil } return process.ProcessRequest(&msgObj) } } return nil }) // 解析生成后的图片 app.Route("/images/:filename").GET(func(c *ship.Context) error { filename := c.Param("filename") root := "./data/images/" return c.File(filepath.Join(root, filename)) }) // 解析生成后的历史聊天 app.Route("/history/:filename").GET(func(c *ship.Context) error { filename := c.Param("filename") root := "./data/chatHistory/" return c.File(filepath.Join(root, filename)) }) // 直接下载文件 app.Route("/download/:filename").GET(func(c *ship.Context) error { filename := c.Param("filename") root := "./data/chatHistory/" return c.Attachment(filepath.Join(root, filename), "") }) port := ":" + public.Config.Port srv := &http.Server{ Addr: port, Handler: app, } // Initializing the server in a goroutine so that // it won't block the graceful shutdown handling below go func() { logger.Info("🚀 The HTTP Server is running on", port) if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { logger.Fatal("listen: %s\n", err) } }() // Wait for interrupt signal to gracefully shutdown the server with // a timeout of 5 seconds. quit := make(chan os.Signal, 1) // kill (no param) default send syscall.SIGTERM // kill -2 is syscall.SIGINT // kill -9 is syscall.SIGKILL but can't be catch, so don't need add it // signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) signal.Notify(quit, os.Interrupt) <-quit logger.Info("Shutting down server...") // 5秒后强制退出 ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() if err := srv.Shutdown(ctx); err != nil { logger.Fatal("Server forced to shutdown:", err) } logger.Info("Server exiting!") }