mirror of https://github.com/portainer/portainer
176 lines
5.3 KiB
Go
176 lines
5.3 KiB
Go
package schedules
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"errors"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/asaskevich/govalidator"
|
|
httperror "github.com/portainer/libhttp/error"
|
|
"github.com/portainer/libhttp/request"
|
|
"github.com/portainer/libhttp/response"
|
|
"github.com/portainer/portainer/api"
|
|
"github.com/portainer/portainer/api/cron"
|
|
)
|
|
|
|
type scheduleUpdatePayload struct {
|
|
Name *string
|
|
Image *string
|
|
CronExpression *string
|
|
Recurring *bool
|
|
Endpoints []portainer.EndpointID
|
|
FileContent *string
|
|
RetryCount *int
|
|
RetryInterval *int
|
|
}
|
|
|
|
func (payload *scheduleUpdatePayload) Validate(r *http.Request) error {
|
|
if payload.Name != nil && !govalidator.Matches(*payload.Name, `^[a-zA-Z0-9][a-zA-Z0-9_.-]+$`) {
|
|
return errors.New("Invalid schedule name format. Allowed characters are: [a-zA-Z0-9_.-]")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (handler *Handler) scheduleUpdate(w http.ResponseWriter, r *http.Request) *httperror.HandlerError {
|
|
settings, err := handler.SettingsService.Settings()
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusServiceUnavailable, "Unable to retrieve settings", err}
|
|
}
|
|
if !settings.EnableHostManagementFeatures {
|
|
return &httperror.HandlerError{http.StatusServiceUnavailable, "Host management features are disabled", portainer.ErrHostManagementFeaturesDisabled}
|
|
}
|
|
|
|
scheduleID, err := request.RetrieveNumericRouteVariableValue(r, "id")
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusBadRequest, "Invalid schedule identifier route variable", err}
|
|
}
|
|
|
|
var payload scheduleUpdatePayload
|
|
err = request.DecodeAndValidateJSONPayload(r, &payload)
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusBadRequest, "Invalid request payload", err}
|
|
}
|
|
|
|
schedule, err := handler.ScheduleService.Schedule(portainer.ScheduleID(scheduleID))
|
|
if err == portainer.ErrObjectNotFound {
|
|
return &httperror.HandlerError{http.StatusNotFound, "Unable to find a schedule with the specified identifier inside the database", err}
|
|
} else if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to find a schedule with the specified identifier inside the database", err}
|
|
}
|
|
|
|
updateJobSchedule := false
|
|
if schedule.EdgeSchedule != nil {
|
|
err := handler.updateEdgeSchedule(schedule, &payload)
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to update Edge schedule", err}
|
|
}
|
|
} else {
|
|
updateJobSchedule = updateSchedule(schedule, &payload)
|
|
}
|
|
|
|
if payload.FileContent != nil {
|
|
_, err := handler.FileService.StoreScheduledJobFileFromBytes(strconv.Itoa(scheduleID), []byte(*payload.FileContent))
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to persist script file changes on the filesystem", err}
|
|
}
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if updateJobSchedule {
|
|
jobContext := cron.NewScriptExecutionJobContext(handler.JobService, handler.EndpointService, handler.FileService)
|
|
jobRunner := cron.NewScriptExecutionJobRunner(schedule, jobContext)
|
|
err := handler.JobScheduler.UpdateJobSchedule(jobRunner)
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to update job scheduler", err}
|
|
}
|
|
}
|
|
|
|
err = handler.ScheduleService.UpdateSchedule(portainer.ScheduleID(scheduleID), schedule)
|
|
if err != nil {
|
|
return &httperror.HandlerError{http.StatusInternalServerError, "Unable to persist schedule changes inside the database", err}
|
|
}
|
|
|
|
return response.JSON(w, schedule)
|
|
}
|
|
|
|
func (handler *Handler) updateEdgeSchedule(schedule *portainer.Schedule, payload *scheduleUpdatePayload) error {
|
|
if payload.Name != nil {
|
|
schedule.Name = *payload.Name
|
|
}
|
|
|
|
if payload.Endpoints != nil {
|
|
|
|
edgeEndpointIDs := make([]portainer.EndpointID, 0)
|
|
|
|
for _, ID := range payload.Endpoints {
|
|
endpoint, err := handler.EndpointService.Endpoint(ID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if endpoint.Type == portainer.EdgeAgentEnvironment {
|
|
edgeEndpointIDs = append(edgeEndpointIDs, endpoint.ID)
|
|
}
|
|
}
|
|
|
|
schedule.EdgeSchedule.Endpoints = edgeEndpointIDs
|
|
}
|
|
|
|
if payload.CronExpression != nil {
|
|
schedule.EdgeSchedule.CronExpression = *payload.CronExpression
|
|
schedule.EdgeSchedule.Version++
|
|
}
|
|
|
|
if payload.FileContent != nil {
|
|
schedule.EdgeSchedule.Script = base64.RawStdEncoding.EncodeToString([]byte(*payload.FileContent))
|
|
schedule.EdgeSchedule.Version++
|
|
}
|
|
|
|
for _, endpointID := range schedule.EdgeSchedule.Endpoints {
|
|
handler.ReverseTunnelService.AddSchedule(endpointID, schedule.EdgeSchedule)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func updateSchedule(schedule *portainer.Schedule, payload *scheduleUpdatePayload) bool {
|
|
updateJobSchedule := false
|
|
|
|
if payload.Name != nil {
|
|
schedule.Name = *payload.Name
|
|
}
|
|
|
|
if payload.Endpoints != nil {
|
|
schedule.ScriptExecutionJob.Endpoints = payload.Endpoints
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if payload.CronExpression != nil {
|
|
schedule.CronExpression = *payload.CronExpression
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if payload.Recurring != nil {
|
|
schedule.Recurring = *payload.Recurring
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if payload.Image != nil {
|
|
schedule.ScriptExecutionJob.Image = *payload.Image
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if payload.RetryCount != nil {
|
|
schedule.ScriptExecutionJob.RetryCount = *payload.RetryCount
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
if payload.RetryInterval != nil {
|
|
schedule.ScriptExecutionJob.RetryInterval = *payload.RetryInterval
|
|
updateJobSchedule = true
|
|
}
|
|
|
|
return updateJobSchedule
|
|
}
|