2016-11-03 06:10:02 +00:00
|
|
|
package kapacitor
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/influxdata/chronograf"
|
|
|
|
client "github.com/influxdata/kapacitor/client/v1"
|
|
|
|
)
|
|
|
|
|
2016-11-03 22:27:58 +00:00
|
|
|
// Client communicates to kapacitor
|
2016-11-03 06:10:02 +00:00
|
|
|
type Client struct {
|
|
|
|
URL string
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
ID chronograf.ID
|
|
|
|
Ticker chronograf.Ticker
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
2016-11-03 22:27:58 +00:00
|
|
|
// Prefix is prepended to the ID of all alerts
|
|
|
|
Prefix = "chronograf-v1-"
|
2016-11-03 06:10:02 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Task represents a running kapacitor task
|
|
|
|
type Task struct {
|
|
|
|
ID string // Kapacitor ID
|
|
|
|
Href string // Kapacitor relative URI
|
2016-11-10 17:27:42 +00:00
|
|
|
HrefOutput string // Kapacitor relative URI to HTTPOutNode
|
2016-11-03 06:10:02 +00:00
|
|
|
TICKScript chronograf.TICKScript // TICKScript is the running script
|
|
|
|
}
|
|
|
|
|
2016-11-03 22:27:58 +00:00
|
|
|
// Href returns the link to a kapacitor task given an id
|
2016-11-03 06:42:52 +00:00
|
|
|
func (c *Client) Href(ID string) string {
|
|
|
|
return fmt.Sprintf("/kapacitor/v1/tasks/%s", ID)
|
|
|
|
}
|
|
|
|
|
2016-11-10 18:56:34 +00:00
|
|
|
// HrefOutput returns the link to a kapacitor task httpOut Node given an id
|
|
|
|
func (c *Client) HrefOutput(ID string) string {
|
|
|
|
return fmt.Sprintf("/kapacitor/v1/tasks/%s/%s", ID, HTTPEndpoint)
|
|
|
|
}
|
|
|
|
|
2016-11-03 22:27:58 +00:00
|
|
|
// Create builds and POSTs a tickscript to kapacitor
|
2016-11-03 06:10:02 +00:00
|
|
|
func (c *Client) Create(ctx context.Context, rule chronograf.AlertRule) (*Task, error) {
|
|
|
|
kapa, err := c.kapaClient(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
id, err := c.ID.Generate()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
script, err := c.Ticker.Generate(rule)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
kapaID := Prefix + id
|
|
|
|
task, err := kapa.CreateTask(client.CreateTaskOptions{
|
|
|
|
ID: kapaID,
|
2016-11-03 22:27:58 +00:00
|
|
|
Type: toTask(rule.Query),
|
2016-11-03 06:10:02 +00:00
|
|
|
DBRPs: []client.DBRP{{Database: rule.Query.Database, RetentionPolicy: rule.Query.RetentionPolicy}},
|
|
|
|
TICKscript: string(script),
|
|
|
|
Status: client.Enabled,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Task{
|
|
|
|
ID: kapaID,
|
|
|
|
Href: task.Link.Href,
|
2016-11-10 18:56:34 +00:00
|
|
|
HrefOutput: c.HrefOutput(kapaID),
|
2016-11-03 06:10:02 +00:00
|
|
|
TICKScript: script,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2016-11-03 22:27:58 +00:00
|
|
|
// Delete removes tickscript task from kapacitor
|
2016-11-03 06:10:02 +00:00
|
|
|
func (c *Client) Delete(ctx context.Context, href string) error {
|
|
|
|
kapa, err := c.kapaClient(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return kapa.DeleteTask(client.Link{Href: href})
|
|
|
|
}
|
|
|
|
|
2016-11-10 18:56:34 +00:00
|
|
|
func (c *Client) updateStatus(ctx context.Context, href string, status client.TaskStatus) (*Task, error) {
|
|
|
|
kapa, err := c.kapaClient(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
opts := client.UpdateTaskOptions{
|
|
|
|
Status: status,
|
|
|
|
}
|
|
|
|
|
|
|
|
task, err := kapa.UpdateTask(client.Link{Href: href}, opts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Task{
|
|
|
|
ID: task.ID,
|
|
|
|
Href: task.Link.Href,
|
|
|
|
HrefOutput: c.HrefOutput(task.ID),
|
|
|
|
TICKScript: chronograf.TICKScript(task.TICKscript),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Disable changes the tickscript status to disabled for a given href.
|
|
|
|
func (c *Client) Disable(ctx context.Context, href string) (*Task, error) {
|
|
|
|
return c.updateStatus(ctx, href, client.Disabled)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable changes the tickscript status to disabled for a given href.
|
|
|
|
func (c *Client) Enable(ctx context.Context, href string) (*Task, error) {
|
|
|
|
return c.updateStatus(ctx, href, client.Enabled)
|
|
|
|
}
|
|
|
|
|
2016-11-03 22:27:58 +00:00
|
|
|
// Update changes the tickscript of a given id.
|
2016-11-03 06:10:02 +00:00
|
|
|
func (c *Client) Update(ctx context.Context, href string, rule chronograf.AlertRule) (*Task, error) {
|
|
|
|
kapa, err := c.kapaClient(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
script, err := c.Ticker.Generate(rule)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-11-10 18:56:34 +00:00
|
|
|
// We need to disable the kapacitor task followed by enabling it during update.
|
2016-11-03 06:10:02 +00:00
|
|
|
opts := client.UpdateTaskOptions{
|
|
|
|
TICKscript: string(script),
|
2016-11-10 18:56:34 +00:00
|
|
|
Status: client.Disabled,
|
2016-11-03 22:27:58 +00:00
|
|
|
Type: toTask(rule.Query),
|
2016-11-03 06:10:02 +00:00
|
|
|
DBRPs: []client.DBRP{
|
|
|
|
{
|
|
|
|
Database: rule.Query.Database,
|
|
|
|
RetentionPolicy: rule.Query.RetentionPolicy,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
task, err := kapa.UpdateTask(client.Link{Href: href}, opts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-11-10 18:56:34 +00:00
|
|
|
// Now enable the task.
|
|
|
|
if _, err := c.Enable(ctx, href); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-11-03 06:10:02 +00:00
|
|
|
return &Task{
|
|
|
|
ID: task.ID,
|
|
|
|
Href: task.Link.Href,
|
2016-11-10 18:56:34 +00:00
|
|
|
HrefOutput: c.HrefOutput(task.ID),
|
2016-11-03 06:10:02 +00:00
|
|
|
TICKScript: script,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) kapaClient(ctx context.Context) (*client.Client, error) {
|
|
|
|
var creds *client.Credentials
|
|
|
|
if c.Username != "" {
|
|
|
|
creds = &client.Credentials{
|
|
|
|
Method: client.UserAuthentication,
|
|
|
|
Username: c.Username,
|
|
|
|
Password: c.Password,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return client.New(client.Config{
|
|
|
|
URL: c.URL,
|
|
|
|
Credentials: creds,
|
|
|
|
})
|
|
|
|
}
|
2016-11-03 22:27:58 +00:00
|
|
|
|
|
|
|
func toTask(q chronograf.QueryConfig) client.TaskType {
|
|
|
|
if q.RawText == "" {
|
|
|
|
return client.StreamTask
|
2016-11-03 06:10:02 +00:00
|
|
|
}
|
2016-11-03 22:27:58 +00:00
|
|
|
return client.BatchTask
|
2016-11-03 06:10:02 +00:00
|
|
|
}
|