mirror of
https://gitee.com/dromara/mayfly-go
synced 2026-01-06 06:35:47 +08:00
feat: 新增机器计划任务、数据物理删除调整为逻辑删除、支持记录登录ip归属地等
This commit is contained in:
266
server/internal/machine/application/machine_cronjob.go
Normal file
266
server/internal/machine/application/machine_cronjob.go
Normal file
@@ -0,0 +1,266 @@
|
||||
package application
|
||||
|
||||
import (
|
||||
"mayfly-go/internal/machine/domain/entity"
|
||||
"mayfly-go/internal/machine/domain/repository"
|
||||
"mayfly-go/pkg/global"
|
||||
"mayfly-go/pkg/model"
|
||||
"mayfly-go/pkg/scheduler"
|
||||
"mayfly-go/pkg/utils"
|
||||
"time"
|
||||
)
|
||||
|
||||
type MachineCronJob interface {
|
||||
// 分页获取机器任务列表信息
|
||||
GetPageList(condition *entity.MachineCronJob, pageParam *model.PageParam, toEntity any, orderBy ...string) *model.PageResult[any]
|
||||
|
||||
// 获取分页执行结果列表
|
||||
GetExecPageList(condition *entity.MachineCronJobExec, pageParam *model.PageParam, toEntity any, orderBy ...string) *model.PageResult[any]
|
||||
|
||||
// 根据id获取
|
||||
GetById(id uint64, cols ...string) *entity.MachineCronJob
|
||||
|
||||
Save(entity *entity.MachineCronJob) uint64
|
||||
|
||||
Delete(id uint64)
|
||||
|
||||
// 获取管理的机器列表id
|
||||
GetRelateMachineIds(cronJobId uint64) []uint64
|
||||
|
||||
// 获取机器关联的计划任务列表
|
||||
GetRelateCronJobIds(machineId uint64) []uint64
|
||||
|
||||
// 计划任务关联机器
|
||||
CronJobRelateMachines(cronJobId uint64, machineIds []uint64, la *model.LoginAccount)
|
||||
|
||||
// 机器关联计划任务
|
||||
MachineRelateCronJobs(machineId uint64, cronJobs []uint64, la *model.LoginAccount)
|
||||
|
||||
// 初始化计划任务
|
||||
InitCronJob()
|
||||
}
|
||||
|
||||
type machineCropJobAppImpl struct {
|
||||
machineCropJobRepo repository.MachineCronJob
|
||||
machineCropJobRelateRepo repository.MachineCronJobRelate
|
||||
machineCropJobExecRepo repository.MachineCronJobExec
|
||||
machineApp Machine
|
||||
}
|
||||
|
||||
func newMachineCronJobApp(
|
||||
machineCropJobRepo repository.MachineCronJob,
|
||||
machineCropJobRelateRepo repository.MachineCronJobRelate,
|
||||
machineCropJobExecRepo repository.MachineCronJobExec,
|
||||
machineApp Machine,
|
||||
) MachineCronJob {
|
||||
return &machineCropJobAppImpl{
|
||||
machineCropJobRepo: machineCropJobRepo,
|
||||
machineCropJobRelateRepo: machineCropJobRelateRepo,
|
||||
machineCropJobExecRepo: machineCropJobExecRepo,
|
||||
machineApp: machineApp,
|
||||
}
|
||||
}
|
||||
|
||||
// 分页获取机器脚本任务列表
|
||||
func (m *machineCropJobAppImpl) GetPageList(condition *entity.MachineCronJob, pageParam *model.PageParam, toEntity any, orderBy ...string) *model.PageResult[any] {
|
||||
return m.machineCropJobRepo.GetPageList(condition, pageParam, toEntity, orderBy...)
|
||||
}
|
||||
|
||||
// 获取分页执行结果列表
|
||||
func (m *machineCropJobAppImpl) GetExecPageList(condition *entity.MachineCronJobExec, pageParam *model.PageParam, toEntity any, orderBy ...string) *model.PageResult[any] {
|
||||
return m.machineCropJobExecRepo.GetPageList(condition, pageParam, toEntity, orderBy...)
|
||||
}
|
||||
|
||||
// 根据id获取
|
||||
func (m *machineCropJobAppImpl) GetById(id uint64, cols ...string) *entity.MachineCronJob {
|
||||
return m.machineCropJobRepo.GetById(id, cols...)
|
||||
}
|
||||
|
||||
// 保存机器任务信息
|
||||
func (m *machineCropJobAppImpl) Save(mcj *entity.MachineCronJob) uint64 {
|
||||
// 更新操作
|
||||
if mcj.Id != 0 {
|
||||
m.machineCropJobRepo.UpdateById(mcj)
|
||||
// 处理最新的计划任务
|
||||
m.addCronJob(m.GetById(mcj.Id))
|
||||
return mcj.Id
|
||||
}
|
||||
|
||||
m.addCronJob(mcj)
|
||||
m.machineCropJobRepo.Insert(mcj)
|
||||
return mcj.Id
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) Delete(id uint64) {
|
||||
m.machineCropJobRepo.Delete(id)
|
||||
m.machineCropJobExecRepo.Delete(&entity.MachineCronJobExec{CronJobId: id})
|
||||
m.machineCropJobRelateRepo.Delete(&entity.MachineCronJobRelate{CronJobId: id})
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) GetRelateMachineIds(cronJobId uint64) []uint64 {
|
||||
return m.machineCropJobRelateRepo.GetMachineIds(cronJobId)
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) GetRelateCronJobIds(machineId uint64) []uint64 {
|
||||
return m.machineCropJobRelateRepo.GetCronJobIds(machineId)
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) CronJobRelateMachines(cronJobId uint64, machineIds []uint64, la *model.LoginAccount) {
|
||||
oldMachineIds := m.machineCropJobRelateRepo.GetMachineIds(cronJobId)
|
||||
addIds, delIds, _ := utils.ArrayCompare[uint64](machineIds, oldMachineIds, func(u1, u2 uint64) bool { return u1 == u2 })
|
||||
addVals := make([]*entity.MachineCronJobRelate, 0)
|
||||
|
||||
now := time.Now()
|
||||
for _, addId := range addIds {
|
||||
addVals = append(addVals, &entity.MachineCronJobRelate{
|
||||
MachineId: addId,
|
||||
CronJobId: cronJobId,
|
||||
Creator: la.Username,
|
||||
CreatorId: la.Id,
|
||||
CreateTime: &now,
|
||||
})
|
||||
}
|
||||
m.machineCropJobRelateRepo.BatchInsert(addVals)
|
||||
|
||||
for _, delId := range delIds {
|
||||
m.machineCropJobRelateRepo.Delete(&entity.MachineCronJobRelate{CronJobId: cronJobId, MachineId: delId})
|
||||
}
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) MachineRelateCronJobs(machineId uint64, cronJobs []uint64, la *model.LoginAccount) {
|
||||
oldCronIds := m.machineCropJobRelateRepo.GetCronJobIds(machineId)
|
||||
addIds, delIds, _ := utils.ArrayCompare[uint64](cronJobs, oldCronIds, func(u1, u2 uint64) bool { return u1 == u2 })
|
||||
addVals := make([]*entity.MachineCronJobRelate, 0)
|
||||
|
||||
now := time.Now()
|
||||
for _, addId := range addIds {
|
||||
addVals = append(addVals, &entity.MachineCronJobRelate{
|
||||
MachineId: machineId,
|
||||
CronJobId: addId,
|
||||
Creator: la.Username,
|
||||
CreatorId: la.Id,
|
||||
CreateTime: &now,
|
||||
})
|
||||
}
|
||||
m.machineCropJobRelateRepo.BatchInsert(addVals)
|
||||
|
||||
for _, delId := range delIds {
|
||||
m.machineCropJobRelateRepo.Delete(&entity.MachineCronJobRelate{CronJobId: delId, MachineId: machineId})
|
||||
}
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) InitCronJob() {
|
||||
defer func() {
|
||||
if err := recover(); err != nil {
|
||||
global.Log.Errorf("机器计划任务初始化失败: %s", err.(error).Error())
|
||||
}
|
||||
}()
|
||||
|
||||
pageParam := &model.PageParam{
|
||||
PageSize: 2,
|
||||
PageNum: 1,
|
||||
}
|
||||
cond := new(entity.MachineCronJob)
|
||||
cond.Status = entity.MachineCronJobStatusEnable
|
||||
mcjs := new([]entity.MachineCronJob)
|
||||
|
||||
pr := m.GetPageList(cond, pageParam, mcjs)
|
||||
total := pr.Total
|
||||
add := 0
|
||||
|
||||
for {
|
||||
for _, mcj := range *mcjs {
|
||||
m.addCronJob(&mcj)
|
||||
add++
|
||||
}
|
||||
if add >= int(total) {
|
||||
return
|
||||
}
|
||||
|
||||
pageParam.PageNum = pageParam.PageNum + 1
|
||||
m.GetPageList(cond, pageParam, mcjs)
|
||||
}
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) addCronJob(mcj *entity.MachineCronJob) {
|
||||
var key string
|
||||
isDisable := mcj.Status == entity.MachineCronJobStatusDisable
|
||||
if mcj.Id == 0 {
|
||||
key = utils.RandString(16)
|
||||
mcj.Key = key
|
||||
if isDisable {
|
||||
return
|
||||
}
|
||||
} else {
|
||||
key = mcj.Key
|
||||
}
|
||||
|
||||
if isDisable {
|
||||
scheduler.RemoveByKey(key)
|
||||
return
|
||||
}
|
||||
|
||||
scheduler.AddFunByKey(key, mcj.Cron, func() {
|
||||
m.runCronJob(key)
|
||||
})
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) runCronJob(key string) {
|
||||
cronJob := new(entity.MachineCronJob)
|
||||
cronJob.Key = key
|
||||
err := m.machineCropJobRepo.GetBy(cronJob)
|
||||
// 不存在或禁用,则移除该任务
|
||||
if err != nil || cronJob.Status == entity.MachineCronJobStatusDisable {
|
||||
scheduler.RemoveByKey(key)
|
||||
}
|
||||
|
||||
machienIds := m.machineCropJobRelateRepo.GetMachineIds(cronJob.Id)
|
||||
for _, machineId := range machienIds {
|
||||
go m.runCronJob0(machineId, cronJob)
|
||||
}
|
||||
}
|
||||
|
||||
func (m *machineCropJobAppImpl) runCronJob0(mid uint64, cronJob *entity.MachineCronJob) {
|
||||
defer func() {
|
||||
if err := recover(); err != nil {
|
||||
res := err.(error).Error()
|
||||
m.machineCropJobExecRepo.Insert(&entity.MachineCronJobExec{
|
||||
MachineId: mid,
|
||||
CronJobId: cronJob.Id,
|
||||
ExecTime: time.Now(),
|
||||
Status: entity.MachineCronJobExecStatusError,
|
||||
Res: res,
|
||||
})
|
||||
global.Log.Errorf("机器:[%d]执行[%s]计划任务失败: %s", mid, cronJob.Name, res)
|
||||
}
|
||||
}()
|
||||
|
||||
res, err := m.machineApp.GetCli(uint64(mid)).Run(cronJob.Script)
|
||||
if err != nil {
|
||||
if res == "" {
|
||||
res = err.Error()
|
||||
}
|
||||
global.Log.Errorf("机器:[%d]执行[%s]计划任务失败: %s", mid, cronJob.Name, res)
|
||||
} else {
|
||||
global.Log.Debugf("机器:[%d]执行[%s]计划任务成功, 执行结果: %s", mid, cronJob.Name, res)
|
||||
}
|
||||
|
||||
if cronJob.SaveExecResType == entity.SaveExecResTypeNo ||
|
||||
(cronJob.SaveExecResType == entity.SaveExecResTypeOnError && err == nil) {
|
||||
return
|
||||
}
|
||||
|
||||
execRes := &entity.MachineCronJobExec{
|
||||
MachineId: mid,
|
||||
CronJobId: cronJob.Id,
|
||||
ExecTime: time.Now(),
|
||||
Res: res,
|
||||
}
|
||||
if err == nil {
|
||||
execRes.Status = entity.MachineCronJobExecStatusSuccess
|
||||
} else {
|
||||
execRes.Status = entity.MachineCronJobExecStatusError
|
||||
}
|
||||
// 保存执行记录
|
||||
m.machineCropJobExecRepo.Insert(execRes)
|
||||
}
|
||||
Reference in New Issue
Block a user