2021-09-11 14:04:09 +08:00
|
|
|
|
package api
|
2021-05-08 18:00:33 +08:00
|
|
|
|
|
|
|
|
|
|
import (
|
2022-08-29 21:43:24 +08:00
|
|
|
|
"encoding/base64"
|
2022-01-16 21:45:00 +08:00
|
|
|
|
"fmt"
|
2022-09-09 18:26:08 +08:00
|
|
|
|
"mayfly-go/internal/machine/api/form"
|
|
|
|
|
|
"mayfly-go/internal/machine/api/vo"
|
|
|
|
|
|
"mayfly-go/internal/machine/application"
|
|
|
|
|
|
"mayfly-go/internal/machine/domain/entity"
|
|
|
|
|
|
"mayfly-go/internal/machine/infrastructure/machine"
|
2022-10-26 20:49:29 +08:00
|
|
|
|
tagapp "mayfly-go/internal/tag/application"
|
2022-06-02 17:41:11 +08:00
|
|
|
|
"mayfly-go/pkg/biz"
|
2022-08-29 21:43:24 +08:00
|
|
|
|
"mayfly-go/pkg/config"
|
2023-10-26 17:15:49 +08:00
|
|
|
|
"mayfly-go/pkg/errorx"
|
2022-06-02 17:41:11 +08:00
|
|
|
|
"mayfly-go/pkg/ginx"
|
2022-10-26 20:49:29 +08:00
|
|
|
|
"mayfly-go/pkg/model"
|
2023-01-14 16:29:52 +08:00
|
|
|
|
"mayfly-go/pkg/req"
|
2023-10-26 17:15:49 +08:00
|
|
|
|
"mayfly-go/pkg/utils/anyx"
|
2023-10-12 12:14:56 +08:00
|
|
|
|
"mayfly-go/pkg/utils/collx"
|
2023-07-21 17:07:04 +08:00
|
|
|
|
"mayfly-go/pkg/utils/stringx"
|
2022-06-02 17:41:11 +08:00
|
|
|
|
"mayfly-go/pkg/ws"
|
2022-08-29 21:43:24 +08:00
|
|
|
|
"os"
|
|
|
|
|
|
"path"
|
|
|
|
|
|
"sort"
|
2021-05-08 18:00:33 +08:00
|
|
|
|
"strconv"
|
2023-07-01 14:34:42 +08:00
|
|
|
|
"strings"
|
2022-08-29 21:43:24 +08:00
|
|
|
|
"time"
|
2021-05-08 18:00:33 +08:00
|
|
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
|
"github.com/gorilla/websocket"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type Machine struct {
|
2021-07-28 18:03:19 +08:00
|
|
|
|
MachineApp application.Machine
|
2022-10-26 20:49:29 +08:00
|
|
|
|
TagApp tagapp.TagTree
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) Machines(rc *req.Ctx) {
|
2023-07-08 20:05:55 +08:00
|
|
|
|
condition, pageParam := ginx.BindQueryAndPage(rc.GinCtx, new(entity.MachineQuery))
|
2022-10-26 20:49:29 +08:00
|
|
|
|
|
|
|
|
|
|
// 不存在可访问标签id,即没有可操作数据
|
|
|
|
|
|
tagIds := m.TagApp.ListTagIdByAccountId(rc.LoginAccount.Id)
|
|
|
|
|
|
if len(tagIds) == 0 {
|
2023-07-01 14:34:42 +08:00
|
|
|
|
rc.ResData = model.EmptyPageResult[any]()
|
2022-10-26 20:49:29 +08:00
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
condition.TagIds = tagIds
|
2021-12-11 11:19:47 +08:00
|
|
|
|
|
2023-10-26 17:15:49 +08:00
|
|
|
|
res, err := m.MachineApp.GetMachineList(condition, pageParam, new([]*vo.MachineVO))
|
|
|
|
|
|
biz.ErrIsNil(err)
|
2021-12-11 11:19:47 +08:00
|
|
|
|
if res.Total == 0 {
|
|
|
|
|
|
rc.ResData = res
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-07-01 14:34:42 +08:00
|
|
|
|
for _, mv := range *res.List {
|
2023-03-06 16:59:57 +08:00
|
|
|
|
mv.HasCli = machine.HasCli(mv.Id)
|
2021-09-08 17:55:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
rc.ResData = res
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-07-20 22:41:13 +08:00
|
|
|
|
func (m *Machine) MachineTags(rc *req.Ctx) {
|
|
|
|
|
|
rc.ResData = m.TagApp.ListTagByAccountIdAndResource(rc.LoginAccount.Id, new(entity.Machine))
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) MachineStats(rc *req.Ctx) {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx))
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s")
|
|
|
|
|
|
rc.ResData = cli.GetAllStats()
|
2021-11-25 14:34:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-06 16:59:57 +08:00
|
|
|
|
// 保存机器信息
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) SaveMachine(rc *req.Ctx) {
|
2021-05-08 18:00:33 +08:00
|
|
|
|
machineForm := new(form.MachineForm)
|
2023-07-08 20:05:55 +08:00
|
|
|
|
me := ginx.BindJsonAndCopyTo(rc.GinCtx, machineForm, new(entity.Machine))
|
2022-07-20 23:25:52 +08:00
|
|
|
|
|
2023-03-06 16:59:57 +08:00
|
|
|
|
machineForm.Password = "******"
|
2022-07-14 11:39:12 +08:00
|
|
|
|
rc.ReqParam = machineForm
|
|
|
|
|
|
|
2022-07-20 23:25:52 +08:00
|
|
|
|
me.SetBaseInfo(rc.LoginAccount)
|
2023-10-26 17:15:49 +08:00
|
|
|
|
biz.ErrIsNil(m.MachineApp.Save(me))
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-03-06 16:59:57 +08:00
|
|
|
|
func (m *Machine) TestConn(rc *req.Ctx) {
|
2023-09-09 23:34:26 +08:00
|
|
|
|
me := ginx.BindJsonAndCopyTo(rc.GinCtx, new(form.MachineForm), new(entity.Machine))
|
2023-10-26 17:15:49 +08:00
|
|
|
|
// 测试连接
|
|
|
|
|
|
biz.ErrIsNilAppendErr(m.MachineApp.TestConn(me), "该机器无法连接: %s")
|
2022-08-02 21:44:01 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) ChangeStatus(rc *req.Ctx) {
|
2022-04-27 10:59:02 +08:00
|
|
|
|
g := rc.GinCtx
|
|
|
|
|
|
id := uint64(ginx.PathParamInt(g, "machineId"))
|
|
|
|
|
|
status := int8(ginx.PathParamInt(g, "status"))
|
2023-10-12 12:14:56 +08:00
|
|
|
|
rc.ReqParam = collx.Kvs("id", id, "status", status)
|
2023-10-26 17:15:49 +08:00
|
|
|
|
biz.ErrIsNil(m.MachineApp.ChangeStatus(id, status))
|
2022-04-27 10:59:02 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) DeleteMachine(rc *req.Ctx) {
|
2023-07-01 14:34:42 +08:00
|
|
|
|
idsStr := ginx.PathParam(rc.GinCtx, "machineId")
|
|
|
|
|
|
rc.ReqParam = idsStr
|
|
|
|
|
|
ids := strings.Split(idsStr, ",")
|
|
|
|
|
|
|
|
|
|
|
|
for _, v := range ids {
|
|
|
|
|
|
value, err := strconv.Atoi(v)
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "string类型转换为int异常: %s")
|
|
|
|
|
|
m.MachineApp.Delete(uint64(value))
|
|
|
|
|
|
}
|
2021-07-28 18:03:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2021-09-08 17:55:57 +08:00
|
|
|
|
// 关闭机器客户端
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) CloseCli(rc *req.Ctx) {
|
2021-09-08 17:55:57 +08:00
|
|
|
|
machine.DeleteCli(GetMachineId(rc.GinCtx))
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-01-16 21:45:00 +08:00
|
|
|
|
// 获取进程列表信息
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) GetProcess(rc *req.Ctx) {
|
2022-01-16 21:45:00 +08:00
|
|
|
|
g := rc.GinCtx
|
|
|
|
|
|
cmd := "ps -aux "
|
|
|
|
|
|
sortType := g.Query("sortType")
|
|
|
|
|
|
if sortType == "2" {
|
|
|
|
|
|
cmd += "--sort -pmem "
|
|
|
|
|
|
} else {
|
|
|
|
|
|
cmd += "--sort -pcpu "
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pname := g.Query("name")
|
|
|
|
|
|
if pname != "" {
|
|
|
|
|
|
cmd += fmt.Sprintf("| grep %s ", pname)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-09-05 12:49:12 +08:00
|
|
|
|
count := ginx.QueryInt(g, "count", 10)
|
|
|
|
|
|
cmd += "| head -n " + fmt.Sprintf("%d", count)
|
2022-04-22 17:49:21 +08:00
|
|
|
|
|
2023-10-26 17:15:49 +08:00
|
|
|
|
cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx))
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s")
|
2022-10-26 20:49:29 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.GetMachine().TagPath), "%s")
|
2022-04-22 17:49:21 +08:00
|
|
|
|
|
|
|
|
|
|
res, err := cli.Run(cmd)
|
2022-01-16 21:45:00 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(err, "获取进程信息失败: %s")
|
|
|
|
|
|
rc.ResData = res
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 终止进程
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) KillProcess(rc *req.Ctx) {
|
2022-01-16 21:45:00 +08:00
|
|
|
|
pid := rc.GinCtx.Query("pid")
|
|
|
|
|
|
biz.NotEmpty(pid, "进程id不能为空")
|
2022-04-22 17:49:21 +08:00
|
|
|
|
|
2023-10-26 17:15:49 +08:00
|
|
|
|
cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx))
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s")
|
2022-10-26 20:49:29 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.GetMachine().TagPath), "%s")
|
2022-04-22 17:49:21 +08:00
|
|
|
|
|
2023-09-08 22:24:45 +08:00
|
|
|
|
res, err := cli.Run("sudo kill -9 " + pid)
|
|
|
|
|
|
biz.ErrIsNil(err, "终止进程失败: %s", res)
|
2022-01-16 21:45:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2021-05-08 18:00:33 +08:00
|
|
|
|
func (m *Machine) WsSSH(g *gin.Context) {
|
2021-11-11 15:56:02 +08:00
|
|
|
|
wsConn, err := ws.Upgrader.Upgrade(g.Writer, g.Request, nil)
|
2021-05-08 18:00:33 +08:00
|
|
|
|
defer func() {
|
2022-08-29 21:43:24 +08:00
|
|
|
|
if wsConn != nil {
|
|
|
|
|
|
if err := recover(); err != nil {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
wsConn.WriteMessage(websocket.TextMessage, []byte(anyx.ToString(err)))
|
2022-08-29 21:43:24 +08:00
|
|
|
|
}
|
2021-05-08 18:00:33 +08:00
|
|
|
|
wsConn.Close()
|
|
|
|
|
|
}
|
|
|
|
|
|
}()
|
|
|
|
|
|
|
2022-08-29 21:43:24 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(err, "升级websocket失败: %s")
|
2021-05-08 18:00:33 +08:00
|
|
|
|
// 权限校验
|
2023-01-14 16:29:52 +08:00
|
|
|
|
rc := req.NewCtxWithGin(g).WithRequiredPermission(req.NewPermission("machine:terminal"))
|
|
|
|
|
|
if err = req.PermissionHandler(rc); err != nil {
|
2023-10-26 17:15:49 +08:00
|
|
|
|
panic(errorx.NewBiz("\033[1;31m您没有权限操作该机器终端,请重新登录后再试~\033[0m"))
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2023-10-26 17:15:49 +08:00
|
|
|
|
cli, err := m.MachineApp.GetCli(GetMachineId(g))
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s")
|
2022-10-26 20:49:29 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.GetMachine().TagPath), "%s")
|
2022-04-22 17:49:21 +08:00
|
|
|
|
|
2022-08-13 19:31:16 +08:00
|
|
|
|
cols := ginx.QueryInt(g, "cols", 80)
|
|
|
|
|
|
rows := ginx.QueryInt(g, "rows", 40)
|
2021-05-08 18:00:33 +08:00
|
|
|
|
|
2022-08-29 21:43:24 +08:00
|
|
|
|
var recorder *machine.Recorder
|
|
|
|
|
|
if cli.GetMachine().EnableRecorder == 1 {
|
|
|
|
|
|
now := time.Now()
|
|
|
|
|
|
// 回放文件路径为: 基础配置路径/机器id/操作日期/操作者账号/操作时间.cast
|
|
|
|
|
|
recPath := fmt.Sprintf("%s/%d/%s/%s", config.Conf.Server.GetMachineRecPath(), cli.GetMachine().Id, now.Format("20060102"), rc.LoginAccount.Username)
|
|
|
|
|
|
os.MkdirAll(recPath, 0766)
|
|
|
|
|
|
fileName := path.Join(recPath, fmt.Sprintf("%s.cast", now.Format("20060102_150405")))
|
|
|
|
|
|
f, err := os.OpenFile(fileName, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0766)
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "创建终端回放记录文件失败: %s")
|
|
|
|
|
|
defer f.Close()
|
|
|
|
|
|
recorder = machine.NewRecorder(f)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2023-07-21 17:07:04 +08:00
|
|
|
|
mts, err := machine.NewTerminalSession(stringx.Rand(16), wsConn, cli, rows, cols, recorder)
|
2022-08-13 19:31:16 +08:00
|
|
|
|
biz.ErrIsNilAppendErr(err, "\033[1;31m连接失败: %s\033[0m")
|
2022-11-18 17:52:30 +08:00
|
|
|
|
|
|
|
|
|
|
// 记录系统操作日志
|
2023-07-08 20:05:55 +08:00
|
|
|
|
rc.WithLog(req.NewLogSave("机器-终端操作"))
|
2023-09-09 23:34:26 +08:00
|
|
|
|
rc.ReqParam = cli.GetMachine()
|
2023-01-14 16:29:52 +08:00
|
|
|
|
req.LogHandler(rc)
|
2022-11-18 17:52:30 +08:00
|
|
|
|
|
2022-08-13 19:31:16 +08:00
|
|
|
|
mts.Start()
|
|
|
|
|
|
defer mts.Stop()
|
2021-05-08 18:00:33 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2022-08-29 21:43:24 +08:00
|
|
|
|
// 获取机器终端回放记录的相应文件夹名或文件内容
|
2023-01-14 16:29:52 +08:00
|
|
|
|
func (m *Machine) MachineRecDirNames(rc *req.Ctx) {
|
2022-08-29 21:43:24 +08:00
|
|
|
|
readPath := rc.GinCtx.Query("path")
|
|
|
|
|
|
biz.NotEmpty(readPath, "path不能为空")
|
|
|
|
|
|
path_ := path.Join(config.Conf.Server.GetMachineRecPath(), readPath)
|
|
|
|
|
|
|
|
|
|
|
|
// 如果是读取文件内容,则读取对应回放记录文件内容,否则读取文件夹名列表。小小偷懒一会不想再加个接口
|
|
|
|
|
|
isFile := rc.GinCtx.Query("isFile")
|
|
|
|
|
|
if isFile == "1" {
|
|
|
|
|
|
bytes, err := os.ReadFile(path_)
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "还未有相应终端操作记录: %s")
|
|
|
|
|
|
rc.ResData = base64.StdEncoding.EncodeToString(bytes)
|
|
|
|
|
|
return
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
files, err := os.ReadDir(path_)
|
|
|
|
|
|
biz.ErrIsNilAppendErr(err, "还未有相应终端操作记录: %s")
|
|
|
|
|
|
var names []string
|
|
|
|
|
|
for _, f := range files {
|
|
|
|
|
|
names = append(names, f.Name())
|
|
|
|
|
|
}
|
|
|
|
|
|
sort.Sort(sort.Reverse(sort.StringSlice(names)))
|
|
|
|
|
|
rc.ResData = names
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2021-05-08 18:00:33 +08:00
|
|
|
|
func GetMachineId(g *gin.Context) uint64 {
|
|
|
|
|
|
machineId, _ := strconv.Atoi(g.Param("machineId"))
|
|
|
|
|
|
biz.IsTrue(machineId != 0, "machineId错误")
|
|
|
|
|
|
return uint64(machineId)
|
|
|
|
|
|
}
|