2024-10-26 11:57:47 -04:00
|
|
|
|
package controller
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"strconv"
|
|
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
"github.com/jinzhu/copier"
|
|
|
|
|
|
2024-11-28 06:38:54 -05:00
|
|
|
|
"github.com/nezhahq/nezha/model"
|
|
|
|
|
"github.com/nezhahq/nezha/service/singleton"
|
2024-10-26 11:57:47 -04:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// List schedule tasks
|
|
|
|
|
// @Summary List schedule tasks
|
|
|
|
|
// @Security BearerAuth
|
|
|
|
|
// @Schemes
|
|
|
|
|
// @Description List schedule tasks
|
|
|
|
|
// @Tags auth required
|
2024-12-24 10:23:01 -05:00
|
|
|
|
// @Param id query uint false "Resource ID"
|
2024-10-26 11:57:47 -04:00
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} model.CommonResponse[[]model.Cron]
|
|
|
|
|
// @Router /cron [get]
|
|
|
|
|
func listCron(c *gin.Context) ([]*model.Cron, error) {
|
|
|
|
|
singleton.CronLock.RLock()
|
|
|
|
|
defer singleton.CronLock.RUnlock()
|
|
|
|
|
|
|
|
|
|
var cr []*model.Cron
|
|
|
|
|
if err := copier.Copy(&cr, &singleton.CronList); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return cr, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create new schedule task
|
|
|
|
|
// @Summary Create new schedule task
|
|
|
|
|
// @Security BearerAuth
|
|
|
|
|
// @Schemes
|
|
|
|
|
// @Description Create new schedule task
|
|
|
|
|
// @Tags auth required
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @param request body model.CronForm true "CronForm"
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} model.CommonResponse[uint64]
|
|
|
|
|
// @Router /cron [post]
|
|
|
|
|
func createCron(c *gin.Context) (uint64, error) {
|
|
|
|
|
var cf model.CronForm
|
|
|
|
|
var cr model.Cron
|
|
|
|
|
|
|
|
|
|
if err := c.ShouldBindJSON(&cf); err != nil {
|
|
|
|
|
return 0, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 11:05:41 -05:00
|
|
|
|
singleton.ServerLock.RLock()
|
|
|
|
|
for _, sid := range cf.Servers {
|
|
|
|
|
if server, ok := singleton.ServerList[sid]; ok {
|
|
|
|
|
if !server.HasPermission(c) {
|
|
|
|
|
singleton.ServerLock.RUnlock()
|
|
|
|
|
return 0, singleton.Localizer.ErrorT("permission denied")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
singleton.ServerLock.RUnlock()
|
|
|
|
|
|
|
|
|
|
cr.UserID = getUid(c)
|
2024-10-26 11:57:47 -04:00
|
|
|
|
cr.TaskType = cf.TaskType
|
|
|
|
|
cr.Name = cf.Name
|
|
|
|
|
cr.Scheduler = cf.Scheduler
|
|
|
|
|
cr.Command = cf.Command
|
|
|
|
|
cr.Servers = cf.Servers
|
|
|
|
|
cr.PushSuccessful = cf.PushSuccessful
|
|
|
|
|
cr.NotificationGroupID = cf.NotificationGroupID
|
|
|
|
|
cr.Cover = cf.Cover
|
|
|
|
|
|
|
|
|
|
if cr.TaskType == model.CronTypeCronTask && cr.Cover == model.CronCoverAlertTrigger {
|
2024-10-31 17:07:04 -04:00
|
|
|
|
return 0, singleton.Localizer.ErrorT("scheduled tasks cannot be triggered by alarms")
|
2024-10-26 11:57:47 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 对于计划任务类型,需要更新CronJob
|
|
|
|
|
var err error
|
|
|
|
|
if cf.TaskType == model.CronTypeCronTask {
|
|
|
|
|
if cr.CronJobID, err = singleton.Cron.AddFunc(cr.Scheduler, singleton.CronTrigger(&cr)); err != nil {
|
|
|
|
|
return 0, err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = singleton.DB.Create(&cr).Error; err != nil {
|
|
|
|
|
return 0, newGormError("%v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
singleton.OnRefreshOrAddCron(&cr)
|
|
|
|
|
singleton.UpdateCronList()
|
|
|
|
|
return cr.ID, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update schedule task
|
|
|
|
|
// @Summary Update schedule task
|
|
|
|
|
// @Security BearerAuth
|
|
|
|
|
// @Schemes
|
|
|
|
|
// @Description Update schedule task
|
|
|
|
|
// @Tags auth required
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @param id path uint true "Task ID"
|
|
|
|
|
// @param request body model.CronForm true "CronForm"
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} model.CommonResponse[any]
|
|
|
|
|
// @Router /cron/{id} [patch]
|
|
|
|
|
func updateCron(c *gin.Context) (any, error) {
|
|
|
|
|
idStr := c.Param("id")
|
|
|
|
|
id, err := strconv.ParseUint(idStr, 10, 64)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var cf model.CronForm
|
|
|
|
|
if err := c.ShouldBindJSON(&cf); err != nil {
|
|
|
|
|
return 0, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 11:05:41 -05:00
|
|
|
|
singleton.ServerLock.RLock()
|
|
|
|
|
for _, sid := range cf.Servers {
|
|
|
|
|
if server, ok := singleton.ServerList[sid]; ok {
|
|
|
|
|
if !server.HasPermission(c) {
|
|
|
|
|
singleton.ServerLock.RUnlock()
|
|
|
|
|
return nil, singleton.Localizer.ErrorT("permission denied")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
singleton.ServerLock.RUnlock()
|
|
|
|
|
|
2024-10-26 11:57:47 -04:00
|
|
|
|
var cr model.Cron
|
|
|
|
|
if err := singleton.DB.First(&cr, id).Error; err != nil {
|
2024-12-21 11:05:41 -05:00
|
|
|
|
return nil, singleton.Localizer.ErrorT("task id %d does not exist", id)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !cr.HasPermission(c) {
|
|
|
|
|
return nil, singleton.Localizer.ErrorT("permission denied")
|
2024-10-26 11:57:47 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cr.TaskType = cf.TaskType
|
|
|
|
|
cr.Name = cf.Name
|
|
|
|
|
cr.Scheduler = cf.Scheduler
|
|
|
|
|
cr.Command = cf.Command
|
|
|
|
|
cr.Servers = cf.Servers
|
|
|
|
|
cr.PushSuccessful = cf.PushSuccessful
|
|
|
|
|
cr.NotificationGroupID = cf.NotificationGroupID
|
|
|
|
|
cr.Cover = cf.Cover
|
|
|
|
|
|
|
|
|
|
if cr.TaskType == model.CronTypeCronTask && cr.Cover == model.CronCoverAlertTrigger {
|
2024-10-31 17:07:04 -04:00
|
|
|
|
return nil, singleton.Localizer.ErrorT("scheduled tasks cannot be triggered by alarms")
|
2024-10-26 11:57:47 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 对于计划任务类型,需要更新CronJob
|
|
|
|
|
if cf.TaskType == model.CronTypeCronTask {
|
|
|
|
|
if cr.CronJobID, err = singleton.Cron.AddFunc(cr.Scheduler, singleton.CronTrigger(&cr)); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = singleton.DB.Save(&cr).Error; err != nil {
|
|
|
|
|
return nil, newGormError("%v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
singleton.OnRefreshOrAddCron(&cr)
|
|
|
|
|
singleton.UpdateCronList()
|
|
|
|
|
return nil, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Trigger schedule task
|
|
|
|
|
// @Summary Trigger schedule task
|
|
|
|
|
// @Security BearerAuth
|
|
|
|
|
// @Schemes
|
|
|
|
|
// @Description Trigger schedule task
|
|
|
|
|
// @Tags auth required
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @param id path uint true "Task ID"
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} model.CommonResponse[any]
|
|
|
|
|
// @Router /cron/{id}/manual [get]
|
|
|
|
|
func manualTriggerCron(c *gin.Context) (any, error) {
|
|
|
|
|
idStr := c.Param("id")
|
|
|
|
|
id, err := strconv.ParseUint(idStr, 10, 64)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 11:05:41 -05:00
|
|
|
|
singleton.CronLock.RLock()
|
|
|
|
|
cr, ok := singleton.Crons[id]
|
|
|
|
|
if !ok {
|
|
|
|
|
singleton.CronLock.RUnlock()
|
2024-10-31 17:07:04 -04:00
|
|
|
|
return nil, singleton.Localizer.ErrorT("task id %d does not exist", id)
|
2024-10-26 11:57:47 -04:00
|
|
|
|
}
|
2024-12-21 11:05:41 -05:00
|
|
|
|
singleton.CronLock.RUnlock()
|
2024-10-26 11:57:47 -04:00
|
|
|
|
|
2024-12-21 11:05:41 -05:00
|
|
|
|
if !cr.HasPermission(c) {
|
|
|
|
|
return nil, singleton.Localizer.ErrorT("permission denied")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
singleton.ManualTrigger(cr)
|
2024-10-26 11:57:47 -04:00
|
|
|
|
return nil, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Batch delete schedule tasks
|
|
|
|
|
// @Summary Batch delete schedule tasks
|
|
|
|
|
// @Security BearerAuth
|
|
|
|
|
// @Schemes
|
|
|
|
|
// @Description Batch delete schedule tasks
|
|
|
|
|
// @Tags auth required
|
|
|
|
|
// @Accept json
|
|
|
|
|
// @param request body []uint64 true "id list"
|
|
|
|
|
// @Produce json
|
|
|
|
|
// @Success 200 {object} model.CommonResponse[any]
|
|
|
|
|
// @Router /batch-delete/cron [post]
|
|
|
|
|
func batchDeleteCron(c *gin.Context) (any, error) {
|
|
|
|
|
var cr []uint64
|
|
|
|
|
if err := c.ShouldBindJSON(&cr); err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-21 11:05:41 -05:00
|
|
|
|
singleton.CronLock.RLock()
|
|
|
|
|
for _, crID := range cr {
|
|
|
|
|
if crn, ok := singleton.Crons[crID]; ok {
|
|
|
|
|
if !crn.HasPermission(c) {
|
|
|
|
|
singleton.CronLock.RUnlock()
|
|
|
|
|
return nil, singleton.Localizer.ErrorT("permission denied")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
singleton.CronLock.RUnlock()
|
|
|
|
|
|
2024-10-26 11:57:47 -04:00
|
|
|
|
if err := singleton.DB.Unscoped().Delete(&model.Cron{}, "id in (?)", cr).Error; err != nil {
|
|
|
|
|
return nil, newGormError("%v", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
singleton.OnDeleteCron(cr)
|
2024-11-19 10:34:56 -05:00
|
|
|
|
singleton.UpdateCronList()
|
2024-10-26 11:57:47 -04:00
|
|
|
|
return nil, nil
|
|
|
|
|
}
|