package api import ( "encoding/base64" "fmt" "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/mcm" tagapp "mayfly-go/internal/tag/application" "mayfly-go/pkg/biz" "mayfly-go/pkg/config" "mayfly-go/pkg/errorx" "mayfly-go/pkg/ginx" "mayfly-go/pkg/model" "mayfly-go/pkg/req" "mayfly-go/pkg/utils/anyx" "mayfly-go/pkg/utils/collx" "mayfly-go/pkg/utils/stringx" "mayfly-go/pkg/ws" "os" "path" "sort" "strconv" "strings" "time" "github.com/gin-gonic/gin" "github.com/gorilla/websocket" ) type Machine struct { MachineApp application.Machine TagApp tagapp.TagTree } func (m *Machine) Machines(rc *req.Ctx) { condition, pageParam := ginx.BindQueryAndPage(rc.GinCtx, new(entity.MachineQuery)) // 不存在可访问标签id,即没有可操作数据 tagIds := m.TagApp.ListTagIdByAccountId(rc.LoginAccount.Id) if len(tagIds) == 0 { rc.ResData = model.EmptyPageResult[any]() return } condition.TagIds = tagIds res, err := m.MachineApp.GetMachineList(condition, pageParam, new([]*vo.MachineVO)) biz.ErrIsNil(err) if res.Total == 0 { rc.ResData = res return } for _, mv := range *res.List { mv.HasCli = mcm.HasCli(mv.Id) } rc.ResData = res } func (m *Machine) MachineTags(rc *req.Ctx) { rc.ResData = m.TagApp.ListTagByAccountIdAndResource(rc.LoginAccount.Id, new(entity.Machine)) } func (m *Machine) MachineStats(rc *req.Ctx) { cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx)) biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s") rc.ResData = cli.GetAllStats() } // 保存机器信息 func (m *Machine) SaveMachine(rc *req.Ctx) { machineForm := new(form.MachineForm) me := ginx.BindJsonAndCopyTo(rc.GinCtx, machineForm, new(entity.Machine)) machineForm.Password = "******" rc.ReqParam = machineForm me.SetBaseInfo(rc.LoginAccount) biz.ErrIsNil(m.MachineApp.Save(me)) } func (m *Machine) TestConn(rc *req.Ctx) { me := ginx.BindJsonAndCopyTo(rc.GinCtx, new(form.MachineForm), new(entity.Machine)) // 测试连接 biz.ErrIsNilAppendErr(m.MachineApp.TestConn(me), "该机器无法连接: %s") } func (m *Machine) ChangeStatus(rc *req.Ctx) { g := rc.GinCtx id := uint64(ginx.PathParamInt(g, "machineId")) status := int8(ginx.PathParamInt(g, "status")) rc.ReqParam = collx.Kvs("id", id, "status", status) biz.ErrIsNil(m.MachineApp.ChangeStatus(id, status)) } func (m *Machine) DeleteMachine(rc *req.Ctx) { 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)) } } // 关闭机器客户端 func (m *Machine) CloseCli(rc *req.Ctx) { mcm.DeleteCli(GetMachineId(rc.GinCtx)) } // 获取进程列表信息 func (m *Machine) GetProcess(rc *req.Ctx) { 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) } count := ginx.QueryInt(g, "count", 10) cmd += "| head -n " + fmt.Sprintf("%d", count) cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx)) biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s") biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.Info.TagPath), "%s") res, err := cli.Run(cmd) biz.ErrIsNilAppendErr(err, "获取进程信息失败: %s") rc.ResData = res } // 终止进程 func (m *Machine) KillProcess(rc *req.Ctx) { pid := rc.GinCtx.Query("pid") biz.NotEmpty(pid, "进程id不能为空") cli, err := m.MachineApp.GetCli(GetMachineId(rc.GinCtx)) biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s") biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.Info.TagPath), "%s") res, err := cli.Run("sudo kill -9 " + pid) biz.ErrIsNil(err, "终止进程失败: %s", res) } func (m *Machine) WsSSH(g *gin.Context) { wsConn, err := ws.Upgrader.Upgrade(g.Writer, g.Request, nil) defer func() { if wsConn != nil { if err := recover(); err != nil { wsConn.WriteMessage(websocket.TextMessage, []byte(anyx.ToString(err))) } wsConn.Close() } }() biz.ErrIsNilAppendErr(err, "升级websocket失败: %s") // 权限校验 rc := req.NewCtxWithGin(g).WithRequiredPermission(req.NewPermission("machine:terminal")) if err = req.PermissionHandler(rc); err != nil { panic(errorx.NewBiz("\033[1;31m您没有权限操作该机器终端,请重新登录后再试~\033[0m")) } cli, err := m.MachineApp.GetCli(GetMachineId(g)) biz.ErrIsNilAppendErr(err, "获取客户端连接失败: %s") biz.ErrIsNilAppendErr(m.TagApp.CanAccess(rc.LoginAccount.Id, cli.Info.TagPath), "%s") cols := ginx.QueryInt(g, "cols", 80) rows := ginx.QueryInt(g, "rows", 40) var recorder *mcm.Recorder if cli.Info.EnableRecorder == 1 { now := time.Now() // 回放文件路径为: 基础配置路径/机器id/操作日期/操作者账号/操作时间.cast recPath := fmt.Sprintf("%s/%d/%s/%s", config.Conf.Server.GetMachineRecPath(), cli.Info.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 = mcm.NewRecorder(f) } mts, err := mcm.NewTerminalSession(stringx.Rand(16), wsConn, cli, rows, cols, recorder) biz.ErrIsNilAppendErr(err, "\033[1;31m连接失败: %s\033[0m") // 记录系统操作日志 rc.WithLog(req.NewLogSave("机器-终端操作")) rc.ReqParam = cli.Info req.LogHandler(rc) mts.Start() defer mts.Stop() } // 获取机器终端回放记录的相应文件夹名或文件内容 func (m *Machine) MachineRecDirNames(rc *req.Ctx) { 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 } func GetMachineId(g *gin.Context) uint64 { machineId, _ := strconv.Atoi(g.Param("machineId")) biz.IsTrue(machineId != 0, "machineId错误") return uint64(machineId) }