qq_bot/handler/restart/restart.go
lixiangwuxian e2be7bd21d feat: 为Git仓库管理接口添加操作日志
为代码拉取、构建和重启处理程序添加日志记录,增强操作可追溯性和调试信息
2025-03-09 14:34:21 +08:00

196 lines
4.5 KiB
Go

package restart
import (
"errors"
"log"
"net/http"
"os"
"os/exec"
"time"
"git.lxtend.com/qqbot/constants"
"git.lxtend.com/qqbot/handler"
"git.lxtend.com/qqbot/model"
docker "git.lxtend.com/qqbot/service/exec"
"git.lxtend.com/qqbot/util"
"github.com/gin-gonic/gin"
)
var hasVaildBuild = true
func init() {
handler.RegisterHandler("/重启bot", restart, constants.LEVEL_ADMIN)
handler.RegisterHelpInform("/重启bot", "热更新", "重启bot")
handler.RegisterHandler("/构建bot", build, constants.LEVEL_ADMIN)
handler.RegisterHelpInform("/构建bot", "热更新", "构建bot")
handler.RegisterHandler("/构建重启", buildAndRestart, constants.LEVEL_ADMIN)
handler.RegisterHelpInform("/构建重启", "热更新", "构建并重启")
handler.RegisterHandler("/gitpull", pullCode, constants.LEVEL_USER)
handler.RegisterHelpInform("/gitpull", "热更新", "拉取最新代码")
handler.RegisterHandler("/shutdown", shutdown, constants.LEVEL_ADMIN)
handler.RegisterHelpInform("/shutdown", "热更新", "结束程序")
}
func restart(msg model.Message) model.Reply {
if !hasVaildBuild {
return model.Reply{
ReplyMsg: "上次构建失败,请先成功构建再部署",
ReferOriginMsg: true,
FromMsg: msg,
}
}
go restartProgram()
return model.Reply{
ReplyMsg: "重启中...",
ReferOriginMsg: true,
FromMsg: msg,
}
}
func shutdown(msg model.Message) model.Reply {
go shutdownProgram()
return model.Reply{
ReplyMsg: "关闭中...",
ReferOriginMsg: true,
FromMsg: msg,
}
}
func build(msg model.Message) model.Reply {
err := buildBot()
if err != nil {
return model.Reply{
ReplyMsg: "构建失败,报错如下\n" + err.Error(),
ReferOriginMsg: true,
FromMsg: msg,
}
}
return model.Reply{
ReplyMsg: "构建成功",
ReferOriginMsg: true,
FromMsg: msg,
}
}
func buildAndRestart(msg model.Message) model.Reply {
err := buildBot()
if err != nil {
return model.Reply{
ReplyMsg: "构建失败,报错如下\n" + err.Error(),
ReferOriginMsg: true,
FromMsg: msg,
}
}
if !hasVaildBuild {
return model.Reply{
ReplyMsg: "构建失败,请先成功构建再部署",
ReferOriginMsg: true,
FromMsg: msg,
}
}
go restartProgram()
return model.Reply{
ReplyMsg: "构建成功,重启中...",
ReferOriginMsg: true,
FromMsg: msg,
}
}
func pullCode(msg model.Message) model.Reply {
err := util.GitPull()
if err != nil {
return model.Reply{
ReplyMsg: "拉取代码失败,报错如下\n" + err.Error(),
ReferOriginMsg: true,
FromMsg: msg,
}
}
return model.Reply{
ReplyMsg: "拉取代码成功",
ReferOriginMsg: true,
FromMsg: msg,
}
}
func restartProgram() error {
log.Println("重启程序...")
docker.DockerContainer.RemoveContainer()
time.Sleep(500 * time.Millisecond)
path, err := os.Executable()
if err != nil {
return err
}
workDir, err := os.Getwd()
if err != nil {
return err
}
cmd := exec.Command(path)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
cmd.Dir = workDir
if err := cmd.Start(); err != nil {
return err
}
os.Exit(0)
return nil
}
func shutdownProgram() error {
log.Println("关闭程序...")
docker.DockerContainer.RemoveContainer()
time.Sleep(500 * time.Millisecond)
os.Exit(0)
return nil
}
func buildBot() error {
hasVaildBuild = false
workDir, err := os.Getwd()
if err != nil {
return err
}
cmd := exec.Command("go", "mod", "tidy")
cmd.Dir = workDir
output, err := cmd.CombinedOutput()
if err != nil {
return errors.New(string(output) + err.Error())
}
cmd = exec.Command("go", "build", "-o", "qqbot")
cmd.Dir = workDir
output, err = cmd.CombinedOutput()
if err != nil {
return errors.New(string(output) + err.Error())
}
hasVaildBuild = true
return nil
}
func PullCodeHandler(c *gin.Context) {
err := util.GitPull()
log.Println("拉取代码...")
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
}
c.JSON(http.StatusOK, gin.H{"message": "拉取代码成功"})
}
func BuildBotHandler(c *gin.Context) {
err := buildBot()
log.Println("构建程序...")
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
}
c.JSON(http.StatusOK, gin.H{"message": "构建成功"})
}
func RestartBotHandler(c *gin.Context) {
err := restartProgram()
log.Println("重启程序...")
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
}
c.JSON(http.StatusOK, gin.H{"message": "重启成功"})
}