239 lines
5.7 KiB
Go
239 lines
5.7 KiB
Go
package restart
|
|
|
|
import (
|
|
"errors"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"os/exec"
|
|
"sync"
|
|
"time"
|
|
|
|
"git.lxtend.com/qqbot/config"
|
|
"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
|
|
|
|
// 避免被并发请求构建或重启
|
|
var stuffMutex sync.Mutex
|
|
|
|
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) {
|
|
if !stuffMutex.TryLock() {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": "有其他请求正在运行,请稍后再试"})
|
|
return
|
|
}
|
|
defer stuffMutex.Unlock()
|
|
err := util.GitPull()
|
|
log.Println("拉取代码...")
|
|
if err != nil {
|
|
log.Println(err)
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
c.Status(http.StatusOK)
|
|
// c.JSON(http.StatusOK, gin.H{"message": "拉取代码成功"})
|
|
}
|
|
|
|
func BuildBotHandler(c *gin.Context) {
|
|
if !stuffMutex.TryLock() {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": "有其他请求正在运行,请稍后再试"})
|
|
return
|
|
}
|
|
defer stuffMutex.Unlock()
|
|
err := buildBot()
|
|
log.Println("构建程序...")
|
|
if err != nil {
|
|
log.Println(err)
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
return
|
|
}
|
|
c.Status(http.StatusOK)
|
|
// c.JSON(http.StatusOK, gin.H{"message": "构建成功"})
|
|
}
|
|
|
|
func RestartBotHandler(c *gin.Context) {
|
|
token := c.Request.Header.Get("Authorization")
|
|
if token != config.ConfigManager.GetConfig().WebServer.Token {
|
|
c.JSON(http.StatusUnauthorized, gin.H{"error": "Unauthorized"})
|
|
return
|
|
}
|
|
if !stuffMutex.TryLock() {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": "有其他请求正在运行,请稍后再试"})
|
|
return
|
|
}
|
|
defer stuffMutex.Unlock()
|
|
err := restartProgram()
|
|
log.Println("重启程序...")
|
|
if err != nil {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
|
|
}
|
|
c.Status(http.StatusOK)
|
|
// c.JSON(http.StatusOK, gin.H{"message": "重启成功"})
|
|
}
|
|
|
|
func AllInOneHandler(c *gin.Context) {
|
|
PullCodeHandler(c)
|
|
BuildBotHandler(c)
|
|
RestartBotHandler(c)
|
|
if c.Writer.Status() != http.StatusOK {
|
|
c.JSON(http.StatusInternalServerError, gin.H{"error": "unauthorized"})
|
|
return
|
|
}
|
|
c.JSON(http.StatusOK, gin.H{"message": "全部操作成功"})
|
|
}
|