2019-01-08 00:37:16 +00:00
|
|
|
package influxdb
|
2018-06-05 23:47:32 +00:00
|
|
|
|
2018-11-20 21:53:52 +00:00
|
|
|
import (
|
|
|
|
"context"
|
2018-12-17 23:37:51 +00:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2019-02-25 19:18:39 +00:00
|
|
|
"fmt"
|
2019-03-06 04:57:50 +00:00
|
|
|
"strconv"
|
2019-04-09 22:52:54 +00:00
|
|
|
"time"
|
2018-12-17 23:37:51 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/flux/ast"
|
|
|
|
"github.com/influxdata/flux/ast/edit"
|
|
|
|
"github.com/influxdata/flux/parser"
|
|
|
|
"github.com/influxdata/influxdb/task/options"
|
2018-11-20 21:53:52 +00:00
|
|
|
)
|
2018-06-05 23:47:32 +00:00
|
|
|
|
2018-11-29 16:28:17 +00:00
|
|
|
const (
|
|
|
|
TaskDefaultPageSize = 100
|
|
|
|
TaskMaxPageSize = 500
|
2019-02-24 06:25:24 +00:00
|
|
|
|
|
|
|
TaskStatusActive = "active"
|
|
|
|
TaskStatusInactive = "inactive"
|
2019-10-11 14:53:38 +00:00
|
|
|
)
|
2019-08-06 16:27:52 +00:00
|
|
|
|
2019-10-11 14:53:38 +00:00
|
|
|
var (
|
|
|
|
// TaskSystemType is the type set in tasks' for all crud requests
|
|
|
|
TaskSystemType = "system"
|
2018-11-29 16:28:17 +00:00
|
|
|
)
|
|
|
|
|
2018-06-05 23:47:32 +00:00
|
|
|
// Task is a task. 🎊
|
|
|
|
type Task struct {
|
2019-10-11 14:53:38 +00:00
|
|
|
ID ID `json:"id"`
|
|
|
|
Type string `json:"type,omitempty"`
|
|
|
|
OrganizationID ID `json:"orgID"`
|
|
|
|
Organization string `json:"org"`
|
|
|
|
AuthorizationID ID `json:"-"`
|
|
|
|
Authorization *Authorization `json:"-"`
|
|
|
|
OwnerID ID `json:"ownerID"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Description string `json:"description,omitempty"`
|
|
|
|
Status string `json:"status"`
|
|
|
|
Flux string `json:"flux"`
|
|
|
|
Every string `json:"every,omitempty"`
|
|
|
|
Cron string `json:"cron,omitempty"`
|
|
|
|
Offset string `json:"offset,omitempty"`
|
|
|
|
LatestCompleted string `json:"latestCompleted,omitempty"`
|
|
|
|
CreatedAt string `json:"createdAt,omitempty"`
|
|
|
|
UpdatedAt string `json:"updatedAt,omitempty"`
|
|
|
|
Metadata map[string]interface{} `json:"metadata,omitempty"`
|
2018-06-05 23:47:32 +00:00
|
|
|
}
|
|
|
|
|
2019-03-28 22:52:25 +00:00
|
|
|
// EffectiveCron returns the effective cron string of the options.
|
|
|
|
// If the cron option was specified, it is returned.
|
|
|
|
// If the every option was specified, it is converted into a cron string using "@every".
|
|
|
|
// Otherwise, the empty string is returned.
|
|
|
|
// The value of the offset option is not considered.
|
|
|
|
func (t *Task) EffectiveCron() string {
|
|
|
|
if t.Cron != "" {
|
|
|
|
return t.Cron
|
|
|
|
}
|
|
|
|
if t.Every != "" {
|
|
|
|
return "@every " + t.Every
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2019-10-21 19:54:13 +00:00
|
|
|
// TaskEffectiveCron returns the effective cron string of the options
|
|
|
|
// If the cron option was specified, it is returned.
|
|
|
|
// If the every option was specified, it is converted into a cron string using by taking the largest duration and setting it to run at that interval
|
|
|
|
// Otherwise, the empty string is returned.
|
|
|
|
// The value of the offset option is not considered.
|
|
|
|
// TODO(docmerlin): when we switch completely to the new scheduler, rename this as EffectiveCron and delete the current
|
|
|
|
// EffectiveCron method
|
|
|
|
func (t *Task) TaskEffectiveCron() (string, error) {
|
|
|
|
if t.Cron != "" {
|
|
|
|
return t.Cron, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if t.Every != "" {
|
|
|
|
dur := &options.Duration{}
|
|
|
|
err := dur.Parse(t.Every)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
ts := struct {
|
|
|
|
hours int64
|
|
|
|
minutes int64
|
|
|
|
seconds int64
|
|
|
|
}{1, 1, 1}
|
|
|
|
|
|
|
|
for _, x := range dur.Node.Values {
|
|
|
|
if x.Magnitude < 0 {
|
|
|
|
return "", errors.New("we do not support negative durations")
|
|
|
|
}
|
|
|
|
switch x.Unit {
|
|
|
|
case "h":
|
|
|
|
if x.Magnitude > 24 {
|
|
|
|
return "", errors.New("we do not support everys with hours units over 24 ")
|
|
|
|
}
|
|
|
|
ts.hours = x.Magnitude
|
|
|
|
return fmt.Sprintf("0 0 */%d * * * *", ts.hours), nil
|
|
|
|
case "m":
|
|
|
|
if x.Magnitude > 60 {
|
|
|
|
return "", errors.New("we do not support everys with seconds units over 60 ")
|
|
|
|
}
|
|
|
|
ts.minutes = x.Magnitude
|
|
|
|
return fmt.Sprintf("0 */%d */%d * * * *", ts.minutes, ts.hours), nil
|
|
|
|
|
|
|
|
case "s":
|
|
|
|
if x.Magnitude > 60 {
|
|
|
|
return "", errors.New("we do not support everys with seconds units over 60 ")
|
|
|
|
}
|
|
|
|
ts.seconds = x.Magnitude
|
|
|
|
return fmt.Sprintf("*/%d */%d */%d * * * *", ts.seconds, ts.minutes, ts.hours), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("*/%d */%d */%d * * * *", ts.seconds, ts.minutes, ts.hours), nil
|
|
|
|
}
|
|
|
|
return "", errors.New("either every or cron must be set")
|
|
|
|
}
|
|
|
|
|
2019-09-26 20:55:23 +00:00
|
|
|
// LatestCompletedTime gives the time.Time that the task was last queued to be run in RFC3339 format.
|
|
|
|
func (t *Task) LatestCompletedTime() (time.Time, error) {
|
|
|
|
tm := t.LatestCompleted
|
|
|
|
if tm == "" {
|
|
|
|
tm = t.CreatedAt
|
|
|
|
}
|
|
|
|
return time.Parse(time.RFC3339, tm)
|
|
|
|
}
|
|
|
|
|
|
|
|
// OffsetDuration gives the time.Duration of the Task's Offset property, which represents a delay before execution
|
|
|
|
func (t *Task) OffsetDuration() (time.Duration, error) {
|
|
|
|
if t.Offset == "" {
|
|
|
|
return time.Duration(0), nil
|
|
|
|
}
|
|
|
|
return time.ParseDuration(t.Offset)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run is a record createId when a run of a task is scheduled.
|
2018-06-08 23:41:26 +00:00
|
|
|
type Run struct {
|
2019-10-18 00:23:45 +00:00
|
|
|
ID ID `json:"id,omitempty"`
|
|
|
|
TaskID ID `json:"taskID"`
|
|
|
|
Status string `json:"status"`
|
|
|
|
ScheduledFor time.Time `json:"scheduledFor"` // ScheduledFor is the time the task is scheduled to run at
|
|
|
|
StartedAt time.Time `json:"startedAt,omitempty"` // StartedAt is the time the executor begins running the task
|
|
|
|
FinishedAt time.Time `json:"finishedAt,omitempty"` // FinishedAt is the time the executor finishes running the task
|
|
|
|
RequestedAt time.Time `json:"requestedAt,omitempty"` // RequestedAt is the time the coordinator told the scheduler to schedule the task
|
|
|
|
Log []Log `json:"log,omitempty"`
|
2019-04-09 22:52:54 +00:00
|
|
|
}
|
|
|
|
|
2018-06-08 23:41:26 +00:00
|
|
|
// Log represents a link to a log resource
|
2019-03-05 22:58:33 +00:00
|
|
|
type Log struct {
|
2019-06-05 17:53:44 +00:00
|
|
|
RunID ID `json:"runID,omitempty"`
|
2019-03-05 22:58:33 +00:00
|
|
|
Time string `json:"time"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l Log) String() string {
|
|
|
|
return l.Time + ": " + l.Message
|
|
|
|
}
|
2018-06-08 23:41:26 +00:00
|
|
|
|
2018-06-05 23:47:32 +00:00
|
|
|
// TaskService represents a service for managing one-off and recurring tasks.
|
|
|
|
type TaskService interface {
|
2018-10-29 19:10:33 +00:00
|
|
|
// FindTaskByID returns a single task
|
2018-06-07 18:45:43 +00:00
|
|
|
FindTaskByID(ctx context.Context, id ID) (*Task, error)
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// FindTasks returns a list of tasks that match a filter (limit 100) and the total count
|
2018-06-05 23:47:32 +00:00
|
|
|
// of matching tasks.
|
|
|
|
FindTasks(ctx context.Context, filter TaskFilter) ([]*Task, int, error)
|
2018-06-07 18:45:43 +00:00
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// CreateTask creates a new task.
|
2019-02-09 01:34:44 +00:00
|
|
|
// The owner of the task is inferred from the authorizer associated with ctx.
|
|
|
|
CreateTask(ctx context.Context, t TaskCreate) (*Task, error)
|
2018-06-07 18:45:43 +00:00
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// UpdateTask updates a single task with changeset.
|
2018-06-07 18:45:43 +00:00
|
|
|
UpdateTask(ctx context.Context, id ID, upd TaskUpdate) (*Task, error)
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// DeleteTask removes a task by ID and purges all associated data and scheduled runs.
|
2018-06-07 18:45:43 +00:00
|
|
|
DeleteTask(ctx context.Context, id ID) error
|
2018-06-08 23:41:26 +00:00
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// FindLogs returns logs for a run.
|
2018-06-08 23:41:26 +00:00
|
|
|
FindLogs(ctx context.Context, filter LogFilter) ([]*Log, int, error)
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// FindRuns returns a list of runs that match a filter and the total count of returned runs.
|
2018-06-08 23:41:26 +00:00
|
|
|
FindRuns(ctx context.Context, filter RunFilter) ([]*Run, int, error)
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// FindRunByID returns a single run.
|
2018-10-25 23:43:11 +00:00
|
|
|
FindRunByID(ctx context.Context, taskID, runID ID) (*Run, error)
|
2018-06-08 23:41:26 +00:00
|
|
|
|
2018-10-04 17:12:43 +00:00
|
|
|
// CancelRun cancels a currently running run.
|
|
|
|
CancelRun(ctx context.Context, taskID, runID ID) error
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// RetryRun creates and returns a new run (which is a retry of another run).
|
2018-11-27 22:32:25 +00:00
|
|
|
RetryRun(ctx context.Context, taskID, runID ID) (*Run, error)
|
2018-12-21 17:04:08 +00:00
|
|
|
|
|
|
|
// ForceRun forces a run to occur with unix timestamp scheduledFor, to be executed as soon as possible.
|
|
|
|
// The value of scheduledFor may or may not align with the task's schedule.
|
|
|
|
ForceRun(ctx context.Context, taskID ID, scheduledFor int64) (*Run, error)
|
2018-06-07 18:45:43 +00:00
|
|
|
}
|
|
|
|
|
2019-02-09 01:34:44 +00:00
|
|
|
// TaskCreate is the set of values to create a task.
|
|
|
|
type TaskCreate struct {
|
2019-10-11 14:53:38 +00:00
|
|
|
Type string `json:"type,omitempty"`
|
|
|
|
Flux string `json:"flux"`
|
|
|
|
Description string `json:"description,omitempty"`
|
|
|
|
Status string `json:"status,omitempty"`
|
|
|
|
OrganizationID ID `json:"orgID,omitempty"`
|
|
|
|
Organization string `json:"org,omitempty"`
|
|
|
|
OwnerID ID `json:"-"`
|
|
|
|
Metadata map[string]interface{} `json:"-"` // not to be set through a web request but rather used by a http service using tasks backend.
|
2019-02-09 01:34:44 +00:00
|
|
|
}
|
|
|
|
|
2019-02-24 06:25:24 +00:00
|
|
|
func (t TaskCreate) Validate() error {
|
|
|
|
switch {
|
|
|
|
case t.Flux == "":
|
|
|
|
return errors.New("missing flux")
|
|
|
|
case !t.OrganizationID.Valid() && t.Organization == "":
|
|
|
|
return errors.New("missing orgID and org")
|
|
|
|
case t.Status != "" && t.Status != TaskStatusActive && t.Status != TaskStatusInactive:
|
|
|
|
return fmt.Errorf("invalid task status: %q", t.Status)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
// TaskUpdate represents updates to a task. Options updates override any options set in the Flux field.
|
2018-06-07 18:45:43 +00:00
|
|
|
type TaskUpdate struct {
|
2019-05-08 18:16:20 +00:00
|
|
|
Flux *string `json:"flux,omitempty"`
|
|
|
|
Status *string `json:"status,omitempty"`
|
|
|
|
Description *string `json:"description,omitempty"`
|
2019-04-19 14:37:59 +00:00
|
|
|
|
|
|
|
// LatestCompleted us to set latest completed on startup to skip task catchup
|
2019-10-11 14:53:38 +00:00
|
|
|
LatestCompleted *string `json:"-"`
|
|
|
|
Metadata map[string]interface{} `json:"-"` // not to be set through a web request but rather used by a http service using tasks backend.
|
2019-04-19 14:37:59 +00:00
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
// Options gets unmarshalled from json as if it was flat, with the same level as Flux and Status.
|
|
|
|
Options options.Options // when we unmarshal this gets unmarshalled from flat key-values
|
2019-10-11 14:53:38 +00:00
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TaskUpdate) UnmarshalJSON(data []byte) error {
|
|
|
|
// this is a type so we can marshal string into durations nicely
|
|
|
|
jo := struct {
|
2019-05-08 18:16:20 +00:00
|
|
|
Flux *string `json:"flux,omitempty"`
|
|
|
|
Status *string `json:"status,omitempty"`
|
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Description *string `json:"description,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
|
|
|
// Cron is a cron style time schedule that can be used in place of Every.
|
|
|
|
Cron string `json:"cron,omitempty"`
|
|
|
|
|
|
|
|
// Every represents a fixed period to repeat execution.
|
|
|
|
// It gets marshalled from a string duration, i.e.: "10s" is 10 seconds
|
2019-03-27 20:24:53 +00:00
|
|
|
Every options.Duration `json:"every,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
|
|
|
// Offset represents a delay before execution.
|
|
|
|
// It gets marshalled from a string duration, i.e.: "10s" is 10 seconds
|
2019-03-27 20:24:53 +00:00
|
|
|
Offset *options.Duration `json:"offset,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
2019-03-01 20:09:31 +00:00
|
|
|
Concurrency *int64 `json:"concurrency,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
2019-03-01 20:09:31 +00:00
|
|
|
Retry *int64 `json:"retry,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
}{}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(data, &jo); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
t.Options.Name = jo.Name
|
2019-05-08 18:16:20 +00:00
|
|
|
t.Description = jo.Description
|
2018-12-17 23:37:51 +00:00
|
|
|
t.Options.Cron = jo.Cron
|
2019-03-27 20:24:53 +00:00
|
|
|
t.Options.Every = jo.Every
|
2019-03-01 20:09:31 +00:00
|
|
|
if jo.Offset != nil {
|
2019-03-27 20:24:53 +00:00
|
|
|
offset := *jo.Offset
|
2019-03-01 20:09:31 +00:00
|
|
|
t.Options.Offset = &offset
|
|
|
|
}
|
2018-12-17 23:37:51 +00:00
|
|
|
t.Options.Concurrency = jo.Concurrency
|
|
|
|
t.Options.Retry = jo.Retry
|
|
|
|
t.Flux = jo.Flux
|
|
|
|
t.Status = jo.Status
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t TaskUpdate) MarshalJSON() ([]byte, error) {
|
|
|
|
jo := struct {
|
2019-05-08 18:16:20 +00:00
|
|
|
Flux *string `json:"flux,omitempty"`
|
|
|
|
Status *string `json:"status,omitempty"`
|
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Description *string `json:"description,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
|
|
|
// Cron is a cron style time schedule that can be used in place of Every.
|
|
|
|
Cron string `json:"cron,omitempty"`
|
|
|
|
|
|
|
|
// Every represents a fixed period to repeat execution.
|
2019-03-27 20:24:53 +00:00
|
|
|
Every options.Duration `json:"every,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
|
|
|
// Offset represents a delay before execution.
|
2019-03-27 20:24:53 +00:00
|
|
|
Offset *options.Duration `json:"offset,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
2019-03-01 20:09:31 +00:00
|
|
|
Concurrency *int64 `json:"concurrency,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
|
2019-03-01 20:09:31 +00:00
|
|
|
Retry *int64 `json:"retry,omitempty"`
|
2018-12-17 23:37:51 +00:00
|
|
|
}{}
|
|
|
|
jo.Name = t.Options.Name
|
|
|
|
jo.Cron = t.Options.Cron
|
2019-03-27 20:24:53 +00:00
|
|
|
jo.Every = t.Options.Every
|
2019-05-08 18:16:20 +00:00
|
|
|
jo.Description = t.Description
|
2019-03-01 20:09:31 +00:00
|
|
|
if t.Options.Offset != nil {
|
2019-03-27 20:24:53 +00:00
|
|
|
offset := *t.Options.Offset
|
2019-03-01 20:09:31 +00:00
|
|
|
jo.Offset = &offset
|
|
|
|
}
|
2018-12-17 23:37:51 +00:00
|
|
|
jo.Concurrency = t.Options.Concurrency
|
|
|
|
jo.Retry = t.Options.Retry
|
|
|
|
jo.Flux = t.Flux
|
|
|
|
jo.Status = t.Status
|
|
|
|
return json.Marshal(jo)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t TaskUpdate) Validate() error {
|
|
|
|
switch {
|
2019-03-27 20:24:53 +00:00
|
|
|
case !t.Options.Every.IsZero() && t.Options.Cron != "":
|
2018-12-17 23:37:51 +00:00
|
|
|
return errors.New("cannot specify both every and cron")
|
2019-09-17 21:44:00 +00:00
|
|
|
case !t.Options.Every.IsZero():
|
|
|
|
if _, err := time.ParseDuration(t.Options.Every.String()); err != nil {
|
|
|
|
return fmt.Errorf("every: %s is invalid", err)
|
|
|
|
}
|
|
|
|
case t.Options.Offset != nil && !t.Options.Offset.IsZero():
|
|
|
|
if _, err := time.ParseDuration(t.Options.Offset.String()); err != nil {
|
|
|
|
return fmt.Errorf("offset: %s, %s is invalid", t.Options.Offset.String(), err)
|
|
|
|
}
|
2019-08-20 14:42:40 +00:00
|
|
|
case t.Flux == nil && t.Status == nil && t.Options.IsZero():
|
2018-12-17 23:37:51 +00:00
|
|
|
return errors.New("cannot update task without content")
|
2019-02-24 06:25:24 +00:00
|
|
|
case t.Status != nil && *t.Status != TaskStatusActive && *t.Status != TaskStatusInactive:
|
|
|
|
return fmt.Errorf("invalid task status: %q", *t.Status)
|
2018-12-17 23:37:51 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-09-10 19:56:15 +00:00
|
|
|
// safeParseSource calls the Flux parser.ParseSource function
|
|
|
|
// and is guaranteed not to panic.
|
|
|
|
func safeParseSource(f string) (pkg *ast.Package, err error) {
|
|
|
|
defer func() {
|
|
|
|
if r := recover(); r != nil {
|
|
|
|
err = &Error{
|
|
|
|
Code: EInternal,
|
|
|
|
Msg: "internal error in flux engine; unable to parse",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
pkg = parser.ParseSource(f)
|
|
|
|
return pkg, err
|
|
|
|
}
|
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
// UpdateFlux updates the TaskUpdate to go from updating options to updating a flux string, that now has those updated options in it
|
|
|
|
// It zeros the options in the TaskUpdate.
|
2019-09-10 19:56:15 +00:00
|
|
|
func (t *TaskUpdate) UpdateFlux(oldFlux string) (err error) {
|
2019-03-01 14:44:42 +00:00
|
|
|
if t.Flux != nil && *t.Flux != "" {
|
|
|
|
oldFlux = *t.Flux
|
2018-12-17 23:37:51 +00:00
|
|
|
}
|
2019-03-01 20:09:31 +00:00
|
|
|
toDelete := map[string]struct{}{}
|
2019-09-10 19:56:15 +00:00
|
|
|
parsedPKG, err := safeParseSource(oldFlux)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
if ast.Check(parsedPKG) > 0 {
|
|
|
|
return ast.GetError(parsedPKG)
|
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
parsed := parsedPKG.Files[0]
|
2019-03-27 20:24:53 +00:00
|
|
|
if !t.Options.Every.IsZero() && t.Options.Cron != "" {
|
|
|
|
return errors.New("cannot specify both cron and every")
|
2018-12-17 23:37:51 +00:00
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
op := make(map[string]ast.Expression, 4)
|
2018-12-17 23:37:51 +00:00
|
|
|
|
2019-02-25 19:18:39 +00:00
|
|
|
if t.Options.Name != "" {
|
|
|
|
op["name"] = &ast.StringLiteral{Value: t.Options.Name}
|
|
|
|
}
|
2019-03-27 20:24:53 +00:00
|
|
|
if !t.Options.Every.IsZero() {
|
|
|
|
op["every"] = &t.Options.Every.Node
|
2019-02-25 19:18:39 +00:00
|
|
|
}
|
|
|
|
if t.Options.Cron != "" {
|
|
|
|
op["cron"] = &ast.StringLiteral{Value: t.Options.Cron}
|
|
|
|
}
|
2019-03-01 20:09:31 +00:00
|
|
|
if t.Options.Offset != nil {
|
2019-03-27 20:24:53 +00:00
|
|
|
if !t.Options.Offset.IsZero() {
|
|
|
|
op["offset"] = &t.Options.Offset.Node
|
2019-03-01 20:09:31 +00:00
|
|
|
} else {
|
|
|
|
toDelete["offset"] = struct{}{}
|
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
}
|
2019-03-01 20:09:31 +00:00
|
|
|
if len(op) > 0 || len(toDelete) > 0 {
|
2019-02-25 19:18:39 +00:00
|
|
|
editFunc := func(opt *ast.OptionStatement) (ast.Expression, error) {
|
|
|
|
a, ok := opt.Assignment.(*ast.VariableAssignment)
|
|
|
|
if !ok {
|
|
|
|
return nil, errors.New("option assignment must be variable assignment")
|
|
|
|
}
|
|
|
|
obj, ok := a.Init.(*ast.ObjectExpression)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("value is is %s, not an object expression", a.Init.Type())
|
|
|
|
}
|
|
|
|
// modify in the keys and values that already are in the ast
|
2019-03-01 20:09:31 +00:00
|
|
|
for i, p := range obj.Properties {
|
2019-02-25 19:18:39 +00:00
|
|
|
k := p.Key.Key()
|
2019-03-01 20:09:31 +00:00
|
|
|
if _, ok := toDelete[k]; ok {
|
|
|
|
obj.Properties = append(obj.Properties[:i], obj.Properties[i+1:]...)
|
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
switch k {
|
|
|
|
case "name":
|
|
|
|
if name, ok := op["name"]; ok && t.Options.Name != "" {
|
|
|
|
delete(op, "name")
|
|
|
|
p.Value = name
|
|
|
|
}
|
|
|
|
case "offset":
|
2019-03-01 20:09:31 +00:00
|
|
|
if offset, ok := op["offset"]; ok && t.Options.Offset != nil {
|
2019-02-25 19:18:39 +00:00
|
|
|
delete(op, "offset")
|
2019-03-27 20:24:53 +00:00
|
|
|
p.Value = offset.Copy().(*ast.DurationLiteral)
|
2019-02-25 19:18:39 +00:00
|
|
|
}
|
|
|
|
case "every":
|
2019-03-27 20:24:53 +00:00
|
|
|
if every, ok := op["every"]; ok && !t.Options.Every.IsZero() {
|
|
|
|
p.Value = every.Copy().(*ast.DurationLiteral)
|
2019-02-25 19:18:39 +00:00
|
|
|
delete(op, "every")
|
|
|
|
} else if cron, ok := op["cron"]; ok && t.Options.Cron != "" {
|
|
|
|
delete(op, "cron")
|
|
|
|
p.Value = cron
|
|
|
|
p.Key = &ast.Identifier{Name: "cron"}
|
|
|
|
}
|
|
|
|
case "cron":
|
|
|
|
if cron, ok := op["cron"]; ok && t.Options.Cron != "" {
|
|
|
|
delete(op, "cron")
|
|
|
|
p.Value = cron
|
2019-03-27 20:24:53 +00:00
|
|
|
} else if every, ok := op["every"]; ok && !t.Options.Every.IsZero() {
|
2019-02-25 19:18:39 +00:00
|
|
|
delete(op, "every")
|
|
|
|
p.Key = &ast.Identifier{Name: "every"}
|
2019-03-27 20:24:53 +00:00
|
|
|
p.Value = every.Copy().(*ast.DurationLiteral)
|
2019-02-25 19:18:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// add in new keys and values to the ast
|
|
|
|
for k := range op {
|
|
|
|
obj.Properties = append(obj.Properties, &ast.Property{
|
|
|
|
Key: &ast.Identifier{Name: k},
|
|
|
|
Value: op[k],
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return nil, nil
|
2018-12-17 23:37:51 +00:00
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
|
|
|
|
ok, err := edit.Option(parsed, "task", editFunc)
|
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
return errors.New("unable to edit option")
|
|
|
|
}
|
2019-02-25 19:18:39 +00:00
|
|
|
|
2018-12-17 23:37:51 +00:00
|
|
|
t.Options.Clear()
|
|
|
|
s := ast.Format(parsed)
|
|
|
|
t.Flux = &s
|
|
|
|
}
|
|
|
|
return nil
|
2018-06-05 23:47:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TaskFilter represents a set of filters that restrict the returned results
|
|
|
|
type TaskFilter struct {
|
2019-08-06 16:27:52 +00:00
|
|
|
Type *string
|
2019-08-16 18:16:46 +00:00
|
|
|
Name *string
|
2019-02-27 15:56:32 +00:00
|
|
|
After *ID
|
|
|
|
OrganizationID *ID
|
|
|
|
Organization string
|
|
|
|
User *ID
|
|
|
|
Limit int
|
2018-06-05 23:47:32 +00:00
|
|
|
}
|
2018-06-08 23:41:26 +00:00
|
|
|
|
2019-03-06 04:57:50 +00:00
|
|
|
// QueryParams Converts TaskFilter fields to url query params.
|
|
|
|
func (f TaskFilter) QueryParams() map[string][]string {
|
|
|
|
qp := map[string][]string{}
|
|
|
|
if f.After != nil {
|
|
|
|
qp["after"] = []string{f.After.String()}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.OrganizationID != nil {
|
|
|
|
qp["orgID"] = []string{f.OrganizationID.String()}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.Organization != "" {
|
|
|
|
qp["org"] = []string{f.Organization}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.User != nil {
|
|
|
|
qp["user"] = []string{f.User.String()}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.Limit > 0 {
|
|
|
|
qp["limit"] = []string{strconv.Itoa(f.Limit)}
|
|
|
|
}
|
|
|
|
|
|
|
|
return qp
|
|
|
|
}
|
|
|
|
|
2018-06-08 23:41:26 +00:00
|
|
|
// RunFilter represents a set of filters that restrict the returned results
|
|
|
|
type RunFilter struct {
|
2019-02-16 00:04:54 +00:00
|
|
|
// Task ID is required for listing runs.
|
|
|
|
Task ID
|
|
|
|
|
2018-06-08 23:41:26 +00:00
|
|
|
After *ID
|
|
|
|
Limit int
|
|
|
|
AfterTime string
|
|
|
|
BeforeTime string
|
|
|
|
}
|
|
|
|
|
2019-02-16 00:04:54 +00:00
|
|
|
// LogFilter represents a set of filters that restrict the returned log results.
|
2018-06-08 23:41:26 +00:00
|
|
|
type LogFilter struct {
|
2019-02-16 00:04:54 +00:00
|
|
|
// Task ID is required.
|
|
|
|
Task ID
|
|
|
|
|
|
|
|
// The optional Run ID limits logs to a single run.
|
|
|
|
Run *ID
|
2018-06-08 23:41:26 +00:00
|
|
|
}
|