2016-10-20 14:38:23 +00:00
|
|
|
package chronograf
|
2016-09-12 19:43:01 +00:00
|
|
|
|
|
|
|
import (
|
2016-10-25 15:20:06 +00:00
|
|
|
"context"
|
|
|
|
"net/http"
|
2016-09-12 19:43:01 +00:00
|
|
|
"time"
|
2016-10-25 15:20:06 +00:00
|
|
|
)
|
2016-09-12 19:43:01 +00:00
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// General errors.
|
|
|
|
const (
|
2017-02-01 21:05:06 +00:00
|
|
|
ErrUpstreamTimeout = Error("request to backend timed out")
|
|
|
|
ErrSourceNotFound = Error("source not found")
|
|
|
|
ErrServerNotFound = Error("server not found")
|
|
|
|
ErrLayoutNotFound = Error("layout not found")
|
|
|
|
ErrDashboardNotFound = Error("dashboard not found")
|
|
|
|
ErrUserNotFound = Error("user not found")
|
|
|
|
ErrLayoutInvalid = Error("layout is invalid")
|
|
|
|
ErrAlertNotFound = Error("alert not found")
|
|
|
|
ErrAuthentication = Error("user not authenticated")
|
2017-02-17 16:11:55 +00:00
|
|
|
ErrUninitialized = Error("client uninitialized. Call Open() method")
|
2016-09-12 19:43:01 +00:00
|
|
|
)
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// Error is a domain error encountered while processing chronograf requests
|
|
|
|
type Error string
|
|
|
|
|
|
|
|
func (e Error) Error() string {
|
|
|
|
return string(e)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Logger represents an abstracted structured logging implementation. It
|
|
|
|
// provides methods to trigger log messages at various alert levels and a
|
|
|
|
// WithField method to set keys for a structured log message.
|
|
|
|
type Logger interface {
|
2016-10-24 17:08:36 +00:00
|
|
|
Debug(...interface{})
|
2016-10-25 15:20:06 +00:00
|
|
|
Info(...interface{})
|
|
|
|
Warn(...interface{})
|
|
|
|
Error(...interface{})
|
2016-10-24 17:08:36 +00:00
|
|
|
Fatal(...interface{})
|
|
|
|
Panic(...interface{})
|
2016-10-25 15:20:06 +00:00
|
|
|
|
|
|
|
WithField(string, interface{}) Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assets returns a handler to serve the website.
|
|
|
|
type Assets interface {
|
|
|
|
Handler() http.Handler
|
|
|
|
}
|
|
|
|
|
|
|
|
// TimeSeries represents a queryable time series database.
|
|
|
|
type TimeSeries interface {
|
|
|
|
// Query retrieves time series data from the database.
|
|
|
|
Query(context.Context, Query) (Response, error)
|
|
|
|
// Connect will connect to the time series using the information in `Source`.
|
|
|
|
Connect(context.Context, *Source) error
|
2017-02-17 19:37:00 +00:00
|
|
|
// UsersStore represents the user accounts within the TimeSeries database
|
|
|
|
Users(context.Context) UsersStore
|
2016-10-25 15:20:06 +00:00
|
|
|
}
|
|
|
|
|
2016-11-29 21:04:54 +00:00
|
|
|
// Range represents an upper and lower bound for data
|
|
|
|
type Range struct {
|
2016-11-30 21:22:35 +00:00
|
|
|
Upper int64 `json:"upper"` // Upper is the upper bound
|
|
|
|
Lower int64 `json:"lower"` // Lower is the lower bound
|
2016-11-29 21:04:54 +00:00
|
|
|
}
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// Query retrieves a Response from a TimeSeries.
|
|
|
|
type Query struct {
|
2016-11-29 21:04:54 +00:00
|
|
|
Command string `json:"query"` // Command is the query itself
|
|
|
|
DB string `json:"db,omitempty"` // DB is optional and if empty will not be used.
|
|
|
|
RP string `json:"rp,omitempty"` // RP is a retention policy and optional; if empty will not be used.
|
|
|
|
Wheres []string `json:"wheres,omitempty"` // Wheres restricts the query to certain attributes
|
|
|
|
GroupBys []string `json:"groupbys,omitempty"` // GroupBys collate the query by these tags
|
|
|
|
Label string `json:"label,omitempty"` // Label is the Y-Axis label for the data
|
|
|
|
Range *Range `json:"range,omitempty"` // Range is the default Y-Axis range for the data
|
2016-10-25 15:20:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Response is the result of a query against a TimeSeries
|
|
|
|
type Response interface {
|
|
|
|
MarshalJSON() ([]byte, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Source is connection information to a time-series data store.
|
|
|
|
type Source struct {
|
2017-01-05 01:35:07 +00:00
|
|
|
ID int `json:"id,omitempty,string"` // ID is the unique ID of the source
|
|
|
|
Name string `json:"name"` // Name is the user-defined name for the source
|
|
|
|
Type string `json:"type,omitempty"` // Type specifies which kinds of source (enterprise vs oss)
|
|
|
|
Username string `json:"username,omitempty"` // Username is the username to connect to the source
|
|
|
|
Password string `json:"password,omitempty"` // Password is in CLEARTEXT
|
|
|
|
URL string `json:"url"` // URL are the connections to the source
|
|
|
|
InsecureSkipVerify bool `json:"insecureSkipVerify,omitempty"` // InsecureSkipVerify as true means any certificate presented by the source is accepted.
|
|
|
|
Default bool `json:"default"` // Default specifies the default source for the application
|
|
|
|
Telegraf string `json:"telegraf"` // Telegraf is the db telegraf is written to. By default it is "telegraf"
|
2016-10-25 15:20:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// SourcesStore stores connection information for a `TimeSeries`
|
|
|
|
type SourcesStore interface {
|
|
|
|
// All returns all sources in the store
|
|
|
|
All(context.Context) ([]Source, error)
|
|
|
|
// Add creates a new source in the SourcesStore and returns Source with ID
|
|
|
|
Add(context.Context, Source) (Source, error)
|
|
|
|
// Delete the Source from the store
|
|
|
|
Delete(context.Context, Source) error
|
|
|
|
// Get retrieves Source if `ID` exists
|
|
|
|
Get(ctx context.Context, ID int) (Source, error)
|
|
|
|
// Update the Source in the store.
|
|
|
|
Update(context.Context, Source) error
|
|
|
|
}
|
|
|
|
|
2016-11-03 00:59:25 +00:00
|
|
|
// AlertRule represents rules for building a tickscript alerting task
|
|
|
|
type AlertRule struct {
|
2017-02-09 06:10:23 +00:00
|
|
|
ID string `json:"id,omitempty"` // ID is the unique ID of the alert
|
|
|
|
Query QueryConfig `json:"query"` // Query is the filter of data for the alert.
|
|
|
|
Every string `json:"every"` // Every how often to check for the alerting criteria
|
|
|
|
Alerts []string `json:"alerts"` // Alerts name all the services to notify (e.g. pagerduty)
|
|
|
|
AlertNodes []KapacitorNode `json:"alertNodes,omitempty"` // AlertNodes define additional arguments to alerts
|
|
|
|
Message string `json:"message"` // Message included with alert
|
|
|
|
Details string `json:"details"` // Details is generally used for the Email alert. If empty will not be added.
|
|
|
|
Trigger string `json:"trigger"` // Trigger is a type that defines when to trigger the alert
|
|
|
|
TriggerValues TriggerValues `json:"values"` // Defines the values that cause the alert to trigger
|
|
|
|
Name string `json:"name"` // Name is the user-defined name for the alert
|
2016-11-03 00:59:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AlertRulesStore stores rules for building tickscript alerting tasks
|
|
|
|
type AlertRulesStore interface {
|
2016-11-04 01:56:42 +00:00
|
|
|
// All returns all rules in the store for the given source and kapacitor id
|
|
|
|
All(ctx context.Context, sourceID, kapaID int) ([]AlertRule, error)
|
|
|
|
// Add creates a new rule in the AlertRulesStore and returns AlertRule with ID for a given source and kapacitor id
|
|
|
|
Add(ctx context.Context, sourceID, kapaID int, rule AlertRule) (AlertRule, error)
|
|
|
|
// Delete the AlertRule from the store for a given source and kapacitor ID
|
|
|
|
Delete(ctx context.Context, sourceID, kapaID int, rule AlertRule) error
|
|
|
|
// Get retrieves AlertRule if `ID` exists within a given source and kapacitor id
|
|
|
|
Get(ctx context.Context, sourceID, kapaID int, ID string) (AlertRule, error)
|
|
|
|
// Update the AlertRule in the store within a given source and kapacitor id
|
|
|
|
Update(ctx context.Context, sourceID, kapaID int, rule AlertRule) error
|
2016-11-03 00:59:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// TICKScript task to be used by kapacitor
|
|
|
|
type TICKScript string
|
2016-11-01 00:19:32 +00:00
|
|
|
|
2016-11-03 00:59:25 +00:00
|
|
|
// Ticker generates tickscript tasks for kapacitor
|
|
|
|
type Ticker interface {
|
|
|
|
// Generate will create the tickscript to be used as a kapacitor task
|
|
|
|
Generate(AlertRule) (TICKScript, error)
|
2016-11-01 00:19:32 +00:00
|
|
|
}
|
|
|
|
|
2016-11-04 00:44:28 +00:00
|
|
|
// TriggerValues specifies the alerting logic for a specific trigger type
|
|
|
|
type TriggerValues struct {
|
2017-01-05 04:20:12 +00:00
|
|
|
Change string `json:"change,omitempty"` // Change specifies if the change is a percent or absolute
|
|
|
|
Period string `json:"period,omitempty"` // Period length of time before deadman is alerted
|
|
|
|
Shift string `json:"shift,omitempty"` // Shift is the amount of time to look into the past for the alert to compare to the present
|
|
|
|
Operator string `json:"operator,omitempty"` // Operator for alert comparison
|
|
|
|
Value string `json:"value,omitempty"` // Value is the boundary value when alert goes critical
|
|
|
|
RangeValue string `json:"rangeValue,omitempty"` // RangeValue is an optional value for range comparisons
|
2016-11-10 17:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Field represent influxql fields and functions from the UI
|
|
|
|
type Field struct {
|
|
|
|
Field string `json:"field"`
|
|
|
|
Funcs []string `json:"funcs"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// GroupBy represents influxql group by tags from the UI
|
|
|
|
type GroupBy struct {
|
|
|
|
Time string `json:"time"`
|
|
|
|
Tags []string `json:"tags"`
|
2016-11-03 22:27:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// QueryConfig represents UI query from the data explorer
|
|
|
|
type QueryConfig struct {
|
2016-11-10 17:27:42 +00:00
|
|
|
ID string `json:"id,omitempty"`
|
|
|
|
Database string `json:"database"`
|
|
|
|
Measurement string `json:"measurement"`
|
|
|
|
RetentionPolicy string `json:"retentionPolicy"`
|
|
|
|
Fields []Field `json:"fields"`
|
|
|
|
Tags map[string][]string `json:"tags"`
|
|
|
|
GroupBy GroupBy `json:"groupBy"`
|
|
|
|
AreTagsAccepted bool `json:"areTagsAccepted"`
|
|
|
|
RawText string `json:"rawText,omitempty"`
|
2016-11-03 22:27:58 +00:00
|
|
|
}
|
|
|
|
|
2017-02-09 04:18:23 +00:00
|
|
|
// KapacitorNode adds arguments and properties to an alert
|
|
|
|
type KapacitorNode struct {
|
2017-02-09 06:10:23 +00:00
|
|
|
Name string `json:"name"`
|
|
|
|
Args []string `json:"args"`
|
|
|
|
Properties []KapacitorProperty `json:"properties"`
|
2017-02-09 04:18:23 +00:00
|
|
|
// In the future we could add chaining methods here.
|
|
|
|
}
|
|
|
|
|
|
|
|
// KapacitorProperty modifies the node they are called on
|
|
|
|
type KapacitorProperty struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Args []string `json:"args"`
|
|
|
|
}
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// Server represents a proxy connection to an HTTP server
|
|
|
|
type Server struct {
|
|
|
|
ID int // ID is the unique ID of the server
|
|
|
|
SrcID int // SrcID of the data source
|
|
|
|
Name string // Name is the user-defined name for the server
|
|
|
|
Username string // Username is the username to connect to the server
|
2016-11-10 18:09:14 +00:00
|
|
|
Password string // Password is in CLEARTEXT
|
2016-10-25 15:20:06 +00:00
|
|
|
URL string // URL are the connections to the server
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServersStore stores connection information for a `Server`
|
|
|
|
type ServersStore interface {
|
|
|
|
// All returns all servers in the store
|
|
|
|
All(context.Context) ([]Server, error)
|
|
|
|
// Add creates a new source in the ServersStore and returns Server with ID
|
|
|
|
Add(context.Context, Server) (Server, error)
|
|
|
|
// Delete the Server from the store
|
|
|
|
Delete(context.Context, Server) error
|
|
|
|
// Get retrieves Server if `ID` exists
|
|
|
|
Get(ctx context.Context, ID int) (Server, error)
|
|
|
|
// Update the Server in the store.
|
|
|
|
Update(context.Context, Server) error
|
|
|
|
}
|
|
|
|
|
|
|
|
// ID creates uniq ID string
|
|
|
|
type ID interface {
|
|
|
|
// Generate creates a unique ID string
|
|
|
|
Generate() (string, error)
|
|
|
|
}
|
|
|
|
|
2017-02-17 22:03:49 +00:00
|
|
|
// Permission is a specific allowance for User or Role bound to a
|
|
|
|
// scope of the data source
|
|
|
|
type Permission struct {
|
|
|
|
Scope string `json:"scope"`
|
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
Allowed []string `json:"allowed"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Permissions represent the entire set of permissions a User or Role may have
|
|
|
|
type Permissions []Permission
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// User represents an authenticated user.
|
2016-09-12 19:43:01 +00:00
|
|
|
type User struct {
|
2017-02-17 22:03:49 +00:00
|
|
|
Name string `json:"username"`
|
|
|
|
Passwd string `json:"password"`
|
|
|
|
Permissions Permissions `json:"permissions,omitempty"`
|
2016-11-17 23:57:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UsersStore is the Storage and retrieval of authentication information
|
|
|
|
type UsersStore interface {
|
2017-02-17 21:13:51 +00:00
|
|
|
// All lists all users from the UsersStore
|
|
|
|
All(context.Context) ([]User, error)
|
2016-11-17 23:57:46 +00:00
|
|
|
// Create a new User in the UsersStore
|
|
|
|
Add(context.Context, *User) (*User, error)
|
|
|
|
// Delete the User from the UsersStore
|
|
|
|
Delete(context.Context, *User) error
|
2017-02-17 19:37:00 +00:00
|
|
|
// Get retrieves a user if name exists.
|
|
|
|
Get(ctx context.Context, name string) (*User, error)
|
2016-11-17 23:57:46 +00:00
|
|
|
// Update the user's permissions or roles
|
|
|
|
Update(context.Context, *User) error
|
2016-09-12 19:43:01 +00:00
|
|
|
}
|
|
|
|
|
2016-12-08 00:31:22 +00:00
|
|
|
// DashboardID is the dashboard ID
|
|
|
|
type DashboardID int
|
|
|
|
|
|
|
|
// Dashboard represents all visual and query data for a dashboard
|
|
|
|
type Dashboard struct {
|
2016-12-14 20:12:20 +00:00
|
|
|
ID DashboardID `json:"id"`
|
2016-12-08 00:31:22 +00:00
|
|
|
Cells []DashboardCell `json:"cells"`
|
2016-12-14 20:12:20 +00:00
|
|
|
Name string `json:"name"`
|
2016-12-08 00:31:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DashboardCell holds visual and query information for a cell
|
|
|
|
type DashboardCell struct {
|
2016-09-22 16:10:40 +00:00
|
|
|
X int32 `json:"x"`
|
|
|
|
Y int32 `json:"y"`
|
|
|
|
W int32 `json:"w"`
|
|
|
|
H int32 `json:"h"`
|
|
|
|
Name string `json:"name"`
|
2017-01-27 12:51:31 +00:00
|
|
|
Queries []Query `json:"queries"`
|
2016-09-22 16:10:40 +00:00
|
|
|
Type string `json:"type"`
|
2016-12-08 00:31:22 +00:00
|
|
|
}
|
|
|
|
|
2016-12-09 03:28:40 +00:00
|
|
|
// DashboardsStore is the storage and retrieval of dashboards
|
|
|
|
type DashboardsStore interface {
|
2016-12-20 20:22:53 +00:00
|
|
|
// All lists all dashboards from the DashboardStore
|
2016-12-14 07:56:26 +00:00
|
|
|
All(context.Context) ([]Dashboard, error)
|
2016-12-08 00:31:22 +00:00
|
|
|
// Create a new Dashboard in the DashboardStore
|
2016-12-15 21:37:11 +00:00
|
|
|
Add(context.Context, Dashboard) (Dashboard, error)
|
2016-12-14 06:57:52 +00:00
|
|
|
// Delete the Dashboard from the DashboardStore if `ID` exists.
|
2016-12-15 21:37:11 +00:00
|
|
|
Delete(context.Context, Dashboard) error
|
2016-12-08 00:31:22 +00:00
|
|
|
// Get retrieves a dashboard if `ID` exists.
|
2016-12-15 21:37:11 +00:00
|
|
|
Get(ctx context.Context, id DashboardID) (Dashboard, error)
|
2016-12-08 00:31:22 +00:00
|
|
|
// Update replaces the dashboard information
|
2016-12-15 21:37:11 +00:00
|
|
|
Update(context.Context, Dashboard) error
|
2016-12-08 00:31:22 +00:00
|
|
|
}
|
|
|
|
|
2016-09-12 19:43:01 +00:00
|
|
|
// Cell is a rectangle and multiple time series queries to visualize.
|
|
|
|
type Cell struct {
|
2016-11-29 21:04:54 +00:00
|
|
|
X int32 `json:"x"`
|
|
|
|
Y int32 `json:"y"`
|
|
|
|
W int32 `json:"w"`
|
|
|
|
H int32 `json:"h"`
|
|
|
|
I string `json:"i"`
|
|
|
|
Name string `json:"name"`
|
|
|
|
Queries []Query `json:"queries"`
|
2016-12-14 20:12:20 +00:00
|
|
|
Type string `json:"type"`
|
2016-09-12 19:43:01 +00:00
|
|
|
}
|
|
|
|
|
2016-10-06 04:26:39 +00:00
|
|
|
// Layout is a collection of Cells for visualization
|
|
|
|
type Layout struct {
|
2016-10-28 23:08:05 +00:00
|
|
|
ID string `json:"id"`
|
2016-10-10 15:45:35 +00:00
|
|
|
Application string `json:"app"`
|
2016-10-11 17:48:32 +00:00
|
|
|
Measurement string `json:"measurement"`
|
2016-11-15 18:11:32 +00:00
|
|
|
Autoflow bool `json:"autoflow"`
|
2016-10-10 15:45:35 +00:00
|
|
|
Cells []Cell `json:"cells"`
|
2016-09-12 19:43:01 +00:00
|
|
|
}
|
|
|
|
|
2016-10-06 04:26:39 +00:00
|
|
|
// LayoutStore stores dashboards and associated Cells
|
|
|
|
type LayoutStore interface {
|
2016-09-29 22:07:35 +00:00
|
|
|
// All returns all dashboards in the store
|
2016-10-06 04:26:39 +00:00
|
|
|
All(context.Context) ([]Layout, error)
|
|
|
|
// Add creates a new dashboard in the LayoutStore
|
|
|
|
Add(context.Context, Layout) (Layout, error)
|
2016-09-12 19:43:01 +00:00
|
|
|
// Delete the dashboard from the store
|
2016-10-06 04:26:39 +00:00
|
|
|
Delete(context.Context, Layout) error
|
|
|
|
// Get retrieves Layout if `ID` exists
|
2016-10-10 22:00:27 +00:00
|
|
|
Get(ctx context.Context, ID string) (Layout, error)
|
2016-09-12 19:43:01 +00:00
|
|
|
// Update the dashboard in the store.
|
2016-10-06 04:26:39 +00:00
|
|
|
Update(context.Context, Layout) error
|
2016-09-29 22:07:35 +00:00
|
|
|
}
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// Principal is any entity that can be authenticated
|
|
|
|
type Principal string
|
2016-09-29 22:07:35 +00:00
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// PrincipalKey is used to pass principal
|
|
|
|
// via context.Context to request-scoped
|
|
|
|
// functions.
|
|
|
|
const PrincipalKey Principal = "principal"
|
2016-09-30 23:46:28 +00:00
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// Authenticator represents a service for authenticating users.
|
|
|
|
type Authenticator interface {
|
|
|
|
// Authenticate returns User associated with token if successful.
|
|
|
|
Authenticate(ctx context.Context, token string) (Principal, error)
|
|
|
|
// Token generates a valid token for Principal lasting a duration
|
|
|
|
Token(context.Context, Principal, time.Duration) (string, error)
|
2016-09-30 23:46:28 +00:00
|
|
|
}
|
|
|
|
|
2016-10-25 15:20:06 +00:00
|
|
|
// TokenExtractor extracts tokens from http requests
|
|
|
|
type TokenExtractor interface {
|
|
|
|
// Extract will return the token or an error.
|
|
|
|
Extract(r *http.Request) (string, error)
|
2016-09-30 23:46:28 +00:00
|
|
|
}
|