2019-01-08 00:37:16 +00:00
|
|
|
package influxdb
|
2018-05-31 14:44:23 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-02-28 03:16:42 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2019-01-09 22:43:29 +00:00
|
|
|
"net/url"
|
2018-10-16 13:52:52 +00:00
|
|
|
"sort"
|
|
|
|
"time"
|
2018-05-31 14:44:23 +00:00
|
|
|
)
|
|
|
|
|
2018-12-11 18:38:24 +00:00
|
|
|
// ErrDashboardNotFound is the error msg for a missing dashboard.
|
|
|
|
const ErrDashboardNotFound = "dashboard not found"
|
|
|
|
|
|
|
|
// ErrCellNotFound is the error msg for a missing cell.
|
|
|
|
const ErrCellNotFound = "cell not found"
|
|
|
|
|
2019-02-28 03:16:42 +00:00
|
|
|
// ErrViewNotFound is the error msg for a missing View.
|
|
|
|
const ErrViewNotFound = "view not found"
|
|
|
|
|
2018-12-11 18:38:24 +00:00
|
|
|
// ops for dashboard service.
|
|
|
|
const (
|
2019-01-04 19:12:35 +00:00
|
|
|
OpFindDashboardByID = "FindDashboardByID"
|
|
|
|
OpFindDashboards = "FindDashboards"
|
|
|
|
OpCreateDashboard = "CreateDashboard"
|
|
|
|
OpUpdateDashboard = "UpdateDashboard"
|
|
|
|
OpAddDashboardCell = "AddDashboardCell"
|
|
|
|
OpRemoveDashboardCell = "RemoveDashboardCell"
|
|
|
|
OpUpdateDashboardCell = "UpdateDashboardCell"
|
|
|
|
OpGetDashboardCellView = "GetDashboardCellView"
|
|
|
|
OpUpdateDashboardCellView = "UpdateDashboardCellView"
|
|
|
|
OpDeleteDashboard = "DeleteDashboard"
|
|
|
|
OpReplaceDashboardCells = "ReplaceDashboardCells"
|
2018-12-11 18:38:24 +00:00
|
|
|
)
|
2018-08-07 20:10:05 +00:00
|
|
|
|
2018-05-31 14:44:23 +00:00
|
|
|
// DashboardService represents a service for managing dashboard data.
|
|
|
|
type DashboardService interface {
|
|
|
|
// FindDashboardByID returns a single dashboard by ID.
|
|
|
|
FindDashboardByID(ctx context.Context, id ID) (*Dashboard, error)
|
|
|
|
|
|
|
|
// FindDashboards returns a list of dashboards that match filter and the total count of matching dashboards.
|
|
|
|
// Additional options provide pagination & sorting.
|
2018-10-16 13:52:52 +00:00
|
|
|
FindDashboards(ctx context.Context, filter DashboardFilter, opts FindOptions) ([]*Dashboard, int, error)
|
2018-05-31 14:44:23 +00:00
|
|
|
|
|
|
|
// CreateDashboard creates a new dashboard and sets b.ID with the new identifier.
|
|
|
|
CreateDashboard(ctx context.Context, b *Dashboard) error
|
|
|
|
|
|
|
|
// UpdateDashboard updates a single dashboard with changeset.
|
|
|
|
// Returns the new dashboard state after update.
|
|
|
|
UpdateDashboard(ctx context.Context, id ID, upd DashboardUpdate) (*Dashboard, error)
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// AddDashboardCell adds a cell to a dashboard.
|
|
|
|
AddDashboardCell(ctx context.Context, id ID, c *Cell, opts AddDashboardCellOptions) error
|
2018-05-31 14:44:23 +00:00
|
|
|
|
2019-04-17 20:30:22 +00:00
|
|
|
// RemoveDashboardCell removes a dashboard.
|
2018-05-31 14:44:23 +00:00
|
|
|
RemoveDashboardCell(ctx context.Context, dashboardID, cellID ID) error
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// UpdateDashboardCell replaces the dashboard cell with the provided ID.
|
|
|
|
UpdateDashboardCell(ctx context.Context, dashboardID, cellID ID, upd CellUpdate) (*Cell, error)
|
2018-05-31 14:44:23 +00:00
|
|
|
|
2019-01-04 19:12:35 +00:00
|
|
|
// GetDashboardCellView retrieves a dashboard cells view.
|
|
|
|
GetDashboardCellView(ctx context.Context, dashboardID, cellID ID) (*View, error)
|
|
|
|
|
|
|
|
// UpdateDashboardCellView retrieves a dashboard cells view.
|
|
|
|
UpdateDashboardCellView(ctx context.Context, dashboardID, cellID ID, upd ViewUpdate) (*View, error)
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// DeleteDashboard removes a dashboard by ID.
|
|
|
|
DeleteDashboard(ctx context.Context, id ID) error
|
|
|
|
|
|
|
|
// ReplaceDashboardCells replaces all cells in a dashboard
|
|
|
|
ReplaceDashboardCells(ctx context.Context, id ID, c []*Cell) error
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-11-02 18:21:14 +00:00
|
|
|
// Dashboard represents all visual and query data for a dashboard.
|
2018-05-31 14:44:23 +00:00
|
|
|
type Dashboard struct {
|
2019-01-09 22:43:29 +00:00
|
|
|
ID ID `json:"id,omitempty"`
|
2019-01-14 18:53:17 +00:00
|
|
|
OrganizationID ID `json:"orgID,omitempty"`
|
2019-01-09 22:43:29 +00:00
|
|
|
Name string `json:"name"`
|
|
|
|
Description string `json:"description"`
|
|
|
|
Cells []*Cell `json:"cells"`
|
|
|
|
Meta DashboardMeta `json:"meta"`
|
2018-10-16 13:52:52 +00:00
|
|
|
}
|
|
|
|
|
2018-12-11 18:38:24 +00:00
|
|
|
// DashboardMeta contains meta information about dashboards
|
2018-10-16 13:52:52 +00:00
|
|
|
type DashboardMeta struct {
|
|
|
|
CreatedAt time.Time `json:"createdAt"`
|
|
|
|
UpdatedAt time.Time `json:"updatedAt"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultDashboardFindOptions are the default find options for dashboards
|
|
|
|
var DefaultDashboardFindOptions = FindOptions{
|
|
|
|
SortBy: "ID",
|
|
|
|
}
|
|
|
|
|
|
|
|
// SortDashboards sorts a slice of dashboards by a field.
|
2019-01-16 16:55:17 +00:00
|
|
|
func SortDashboards(opts FindOptions, ds []*Dashboard) {
|
2018-10-16 13:52:52 +00:00
|
|
|
var sorter func(i, j int) bool
|
2019-01-16 16:55:17 +00:00
|
|
|
switch opts.SortBy {
|
2018-10-16 13:52:52 +00:00
|
|
|
case "CreatedAt":
|
|
|
|
sorter = func(i, j int) bool {
|
|
|
|
return ds[j].Meta.CreatedAt.After(ds[i].Meta.CreatedAt)
|
|
|
|
}
|
|
|
|
case "UpdatedAt":
|
|
|
|
sorter = func(i, j int) bool {
|
|
|
|
return ds[j].Meta.UpdatedAt.After(ds[i].Meta.UpdatedAt)
|
|
|
|
}
|
|
|
|
case "Name":
|
|
|
|
sorter = func(i, j int) bool {
|
|
|
|
return ds[i].Name < ds[j].Name
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
sorter = func(i, j int) bool {
|
2019-01-16 16:55:17 +00:00
|
|
|
if opts.Descending {
|
|
|
|
return ds[i].ID > ds[j].ID
|
|
|
|
}
|
2018-10-16 13:52:52 +00:00
|
|
|
return ds[i].ID < ds[j].ID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Slice(ds, sorter)
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// Cell holds positional information about a cell on dashboard and a reference to a cell.
|
|
|
|
type Cell struct {
|
2019-04-16 22:06:39 +00:00
|
|
|
ID ID `json:"id,omitempty"`
|
|
|
|
CellProperty
|
|
|
|
}
|
|
|
|
|
|
|
|
// CellProperty contains the properties of a cell.
|
|
|
|
type CellProperty struct {
|
|
|
|
X int32 `json:"x"`
|
|
|
|
Y int32 `json:"y"`
|
|
|
|
W int32 `json:"w"`
|
|
|
|
H int32 `json:"h"`
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// DashboardFilter is a filter for dashboards.
|
|
|
|
type DashboardFilter struct {
|
2019-01-09 22:43:29 +00:00
|
|
|
IDs []*ID
|
|
|
|
OrganizationID *ID
|
|
|
|
Organization *string
|
|
|
|
}
|
|
|
|
|
|
|
|
// QueryParams turns a dashboard filter into query params
|
2019-01-16 16:55:17 +00:00
|
|
|
//
|
|
|
|
// It implements PagingFilter.
|
|
|
|
func (f DashboardFilter) QueryParams() map[string][]string {
|
2019-01-09 22:43:29 +00:00
|
|
|
qp := url.Values{}
|
|
|
|
for _, id := range f.IDs {
|
|
|
|
if id != nil {
|
|
|
|
qp.Add("id", id.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.OrganizationID != nil {
|
|
|
|
qp.Add("orgID", f.OrganizationID.String())
|
|
|
|
}
|
|
|
|
|
2019-01-16 16:55:17 +00:00
|
|
|
if f.Organization != nil {
|
|
|
|
qp.Add("org", *f.Organization)
|
|
|
|
}
|
|
|
|
|
2019-01-09 22:43:29 +00:00
|
|
|
return qp
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// DashboardUpdate is the patch structure for a dashboard.
|
|
|
|
type DashboardUpdate struct {
|
2018-10-23 01:25:18 +00:00
|
|
|
Name *string `json:"name"`
|
|
|
|
Description *string `json:"description"`
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// Apply applies an update to a dashboard.
|
|
|
|
func (u DashboardUpdate) Apply(d *Dashboard) error {
|
|
|
|
if u.Name != nil {
|
|
|
|
d.Name = *u.Name
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-10-23 01:25:18 +00:00
|
|
|
if u.Description != nil {
|
|
|
|
d.Description = *u.Description
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid returns an error if the dashboard update is invalid.
|
2018-12-11 18:38:24 +00:00
|
|
|
func (u DashboardUpdate) Valid() *Error {
|
2018-10-23 01:25:18 +00:00
|
|
|
if u.Name == nil && u.Description == nil {
|
2018-12-11 18:38:24 +00:00
|
|
|
return &Error{
|
|
|
|
Code: EInvalid,
|
|
|
|
Msg: "must update at least one attribute",
|
|
|
|
}
|
2018-10-23 01:25:18 +00:00
|
|
|
}
|
|
|
|
|
2018-05-31 14:44:23 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-10-23 01:25:18 +00:00
|
|
|
// AddDashboardCellOptions are options for adding a dashboard.
|
|
|
|
type AddDashboardCellOptions struct {
|
2019-01-07 21:47:16 +00:00
|
|
|
View *View
|
2018-10-23 01:25:18 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// CellUpdate is the patch structure for a cell.
|
|
|
|
type CellUpdate struct {
|
2019-01-04 19:12:35 +00:00
|
|
|
X *int32 `json:"x"`
|
|
|
|
Y *int32 `json:"y"`
|
|
|
|
W *int32 `json:"w"`
|
|
|
|
H *int32 `json:"h"`
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
// Apply applies an update to a Cell.
|
|
|
|
func (u CellUpdate) Apply(c *Cell) error {
|
|
|
|
if u.X != nil {
|
|
|
|
c.X = *u.X
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
if u.Y != nil {
|
|
|
|
c.Y = *u.Y
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
if u.W != nil {
|
|
|
|
c.W = *u.W
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
if u.H != nil {
|
|
|
|
c.H = *u.H
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
|
|
|
|
2018-08-07 20:10:05 +00:00
|
|
|
return nil
|
2018-05-31 14:44:23 +00:00
|
|
|
}
|
2018-10-25 23:05:12 +00:00
|
|
|
|
|
|
|
// Valid returns an error if the cell update is invalid.
|
2018-12-11 18:38:24 +00:00
|
|
|
func (u CellUpdate) Valid() *Error {
|
2019-01-04 19:12:35 +00:00
|
|
|
if u.H == nil && u.W == nil && u.Y == nil && u.X == nil {
|
2018-12-11 18:38:24 +00:00
|
|
|
return &Error{
|
|
|
|
Code: EInvalid,
|
|
|
|
Msg: "must update at least one attribute",
|
|
|
|
}
|
2018-10-25 23:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2019-02-28 03:16:42 +00:00
|
|
|
|
|
|
|
// ViewUpdate is a struct for updating Views.
|
|
|
|
type ViewUpdate struct {
|
|
|
|
ViewContentsUpdate
|
|
|
|
Properties ViewProperties
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid validates the update struct. It expects minimal values to be set.
|
|
|
|
func (u ViewUpdate) Valid() *Error {
|
|
|
|
_, ok := u.Properties.(EmptyViewProperties)
|
|
|
|
if u.Name == nil && ok {
|
|
|
|
return &Error{
|
|
|
|
Code: EInvalid,
|
|
|
|
Msg: "expected at least one attribute to be updated",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply updates a view with the view updates properties.
|
|
|
|
func (u ViewUpdate) Apply(v *View) error {
|
|
|
|
if err := u.Valid(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.Name != nil {
|
|
|
|
v.Name = *u.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.Properties != nil {
|
|
|
|
v.Properties = u.Properties
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ViewContentsUpdate is a struct for updating the non properties content of a View.
|
|
|
|
type ViewContentsUpdate struct {
|
|
|
|
Name *string `json:"name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ViewFilter represents a set of filter that restrict the returned results.
|
|
|
|
type ViewFilter struct {
|
|
|
|
ID *ID
|
|
|
|
Types []string
|
|
|
|
}
|
|
|
|
|
|
|
|
// View holds positional and visual information for a View.
|
|
|
|
type View struct {
|
|
|
|
ViewContents
|
|
|
|
Properties ViewProperties
|
|
|
|
}
|
|
|
|
|
2019-04-15 22:00:40 +00:00
|
|
|
// ViewContents is the id and name of a specific view.
|
2019-02-28 03:16:42 +00:00
|
|
|
type ViewContents struct {
|
2019-04-15 22:00:40 +00:00
|
|
|
ID ID `json:"id,omitempty"`
|
2019-02-28 03:16:42 +00:00
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ViewProperties is used to mark other structures as conforming to a View.
|
|
|
|
type ViewProperties interface {
|
|
|
|
viewProperties()
|
|
|
|
GetType() string
|
|
|
|
}
|
|
|
|
|
|
|
|
// EmptyViewProperties is visualization that has no values
|
|
|
|
type EmptyViewProperties struct{}
|
|
|
|
|
|
|
|
func (v EmptyViewProperties) viewProperties() {}
|
|
|
|
|
|
|
|
func (v EmptyViewProperties) GetType() string { return "" }
|
|
|
|
|
|
|
|
// UnmarshalViewPropertiesJSON unmarshals JSON bytes into a ViewProperties.
|
|
|
|
func UnmarshalViewPropertiesJSON(b []byte) (ViewProperties, error) {
|
|
|
|
var v struct {
|
|
|
|
B json.RawMessage `json:"properties"`
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(b, &v); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(v.B) == 0 {
|
|
|
|
// Then there wasn't any visualization field, so there's no need unmarshal it
|
|
|
|
return EmptyViewProperties{}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var t struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
Type string `json:"type"`
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(v.B, &t); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var vis ViewProperties
|
|
|
|
switch t.Shape {
|
|
|
|
case "chronograf-v2":
|
|
|
|
switch t.Type {
|
2019-08-12 21:48:24 +00:00
|
|
|
case "check":
|
|
|
|
var cv CheckViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &cv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = cv
|
2019-02-28 03:16:42 +00:00
|
|
|
case "xy":
|
|
|
|
var xyv XYViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &xyv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = xyv
|
|
|
|
case "single-stat":
|
|
|
|
var ssv SingleStatViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &ssv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = ssv
|
|
|
|
case "gauge":
|
|
|
|
var gv GaugeViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &gv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = gv
|
|
|
|
case "table":
|
|
|
|
var tv TableViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &tv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = tv
|
|
|
|
case "markdown":
|
|
|
|
var mv MarkdownViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &mv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = mv
|
|
|
|
case "log-viewer": // happens in log viewer stays in log viewer.
|
|
|
|
var lv LogViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &lv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = lv
|
|
|
|
case "line-plus-single-stat":
|
|
|
|
var lv LinePlusSingleStatProperties
|
|
|
|
if err := json.Unmarshal(v.B, &lv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = lv
|
|
|
|
case "histogram":
|
|
|
|
var hv HistogramViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &hv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = hv
|
2019-05-16 22:50:51 +00:00
|
|
|
case "heatmap":
|
|
|
|
var hv HeatmapViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &hv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = hv
|
2019-05-28 20:15:29 +00:00
|
|
|
case "scatter":
|
|
|
|
var sv ScatterViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &sv); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = sv
|
2019-02-28 03:16:42 +00:00
|
|
|
}
|
|
|
|
case "empty":
|
|
|
|
var ev EmptyViewProperties
|
|
|
|
if err := json.Unmarshal(v.B, &ev); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vis = ev
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("unknown type %v", t.Shape)
|
|
|
|
}
|
|
|
|
|
|
|
|
return vis, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalViewPropertiesJSON encodes a view into JSON bytes.
|
|
|
|
func MarshalViewPropertiesJSON(v ViewProperties) ([]byte, error) {
|
|
|
|
var s interface{}
|
|
|
|
switch vis := v.(type) {
|
|
|
|
case SingleStatViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
SingleStatViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
SingleStatViewProperties: vis,
|
|
|
|
}
|
|
|
|
case TableViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
TableViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
TableViewProperties: vis,
|
|
|
|
}
|
|
|
|
case GaugeViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
GaugeViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
GaugeViewProperties: vis,
|
|
|
|
}
|
|
|
|
case XYViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
XYViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
XYViewProperties: vis,
|
|
|
|
}
|
|
|
|
case LinePlusSingleStatProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
LinePlusSingleStatProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
LinePlusSingleStatProperties: vis,
|
|
|
|
}
|
|
|
|
case HistogramViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
HistogramViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
HistogramViewProperties: vis,
|
|
|
|
}
|
2019-05-16 22:50:51 +00:00
|
|
|
case HeatmapViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
HeatmapViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
HeatmapViewProperties: vis,
|
|
|
|
}
|
2019-05-28 20:15:29 +00:00
|
|
|
case ScatterViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
ScatterViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
ScatterViewProperties: vis,
|
|
|
|
}
|
2019-02-28 03:16:42 +00:00
|
|
|
case MarkdownViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
MarkdownViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
MarkdownViewProperties: vis,
|
|
|
|
}
|
|
|
|
case LogViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
LogViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
LogViewProperties: vis,
|
|
|
|
}
|
2019-08-12 21:48:24 +00:00
|
|
|
case CheckViewProperties:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
CheckViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "chronograf-v2",
|
|
|
|
|
|
|
|
CheckViewProperties: vis,
|
|
|
|
}
|
2019-02-28 03:16:42 +00:00
|
|
|
default:
|
|
|
|
s = struct {
|
|
|
|
Shape string `json:"shape"`
|
|
|
|
EmptyViewProperties
|
|
|
|
}{
|
|
|
|
Shape: "empty",
|
|
|
|
EmptyViewProperties: EmptyViewProperties{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return json.Marshal(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON encodes a view to JSON bytes.
|
|
|
|
func (c View) MarshalJSON() ([]byte, error) {
|
|
|
|
vis, err := MarshalViewPropertiesJSON(c.Properties)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(struct {
|
|
|
|
ViewContents
|
|
|
|
ViewProperties json.RawMessage `json:"properties"`
|
|
|
|
}{
|
|
|
|
ViewContents: c.ViewContents,
|
|
|
|
ViewProperties: vis,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON decodes JSON bytes into the corresponding view type (those that implement ViewProperties).
|
|
|
|
func (c *View) UnmarshalJSON(b []byte) error {
|
|
|
|
if err := json.Unmarshal(b, &c.ViewContents); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
v, err := UnmarshalViewPropertiesJSON(b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.Properties = v
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON decodes JSON bytes into the corresponding view update type (those that implement ViewProperties).
|
|
|
|
func (u *ViewUpdate) UnmarshalJSON(b []byte) error {
|
|
|
|
if err := json.Unmarshal(b, &u.ViewContentsUpdate); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
v, err := UnmarshalViewPropertiesJSON(b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
u.Properties = v
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON encodes a view to JSON bytes.
|
|
|
|
func (u ViewUpdate) MarshalJSON() ([]byte, error) {
|
|
|
|
vis, err := MarshalViewPropertiesJSON(u.Properties)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(struct {
|
|
|
|
ViewContentsUpdate
|
|
|
|
ViewProperties json.RawMessage `json:"properties,omitempty"`
|
|
|
|
}{
|
|
|
|
ViewContentsUpdate: u.ViewContentsUpdate,
|
|
|
|
ViewProperties: vis,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// LinePlusSingleStatProperties represents options for line plus single stat view in Chronograf
|
|
|
|
type LinePlusSingleStatProperties struct {
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
Axes map[string]Axis `json:"axes"`
|
|
|
|
Type string `json:"type"`
|
|
|
|
Legend Legend `json:"legend"`
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
Prefix string `json:"prefix"`
|
|
|
|
Suffix string `json:"suffix"`
|
|
|
|
DecimalPlaces DecimalPlaces `json:"decimalPlaces"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
2019-06-05 01:07:53 +00:00
|
|
|
XColumn string `json:"xColumn"`
|
|
|
|
YColumn string `json:"yColumn"`
|
2019-06-12 22:35:22 +00:00
|
|
|
ShadeBelow bool `json:"shadeBelow"`
|
2019-02-28 03:16:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// XYViewProperties represents options for line, bar, step, or stacked view in Chronograf
|
|
|
|
type XYViewProperties struct {
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
Axes map[string]Axis `json:"axes"`
|
|
|
|
Type string `json:"type"`
|
|
|
|
Legend Legend `json:"legend"`
|
|
|
|
Geom string `json:"geom"` // Either "line", "step", "stacked", or "bar"
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
2019-06-05 01:07:53 +00:00
|
|
|
XColumn string `json:"xColumn"`
|
|
|
|
YColumn string `json:"yColumn"`
|
2019-06-12 22:35:22 +00:00
|
|
|
ShadeBelow bool `json:"shadeBelow"`
|
2019-02-28 03:16:42 +00:00
|
|
|
}
|
|
|
|
|
2019-08-12 21:48:24 +00:00
|
|
|
// CheckViewProperties represents options for a view representing a check
|
|
|
|
type CheckViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
CheckID string `json:"checkID"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
ViewColors []string `json:"colors"`
|
|
|
|
}
|
|
|
|
|
2019-02-28 03:16:42 +00:00
|
|
|
// SingleStatViewProperties represents options for single stat view in Chronograf
|
|
|
|
type SingleStatViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
Prefix string `json:"prefix"`
|
|
|
|
Suffix string `json:"suffix"`
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
DecimalPlaces DecimalPlaces `json:"decimalPlaces"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// HistogramViewProperties represents options for histogram view in Chronograf
|
|
|
|
type HistogramViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
XColumn string `json:"xColumn"`
|
|
|
|
FillColumns []string `json:"fillColumns"`
|
2019-07-26 02:56:54 +00:00
|
|
|
XDomain []float64 `json:"xDomain,omitempty"`
|
2019-02-28 03:16:42 +00:00
|
|
|
XAxisLabel string `json:"xAxisLabel"`
|
|
|
|
Position string `json:"position"`
|
|
|
|
BinCount int `json:"binCount"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
2019-05-16 22:50:51 +00:00
|
|
|
// HeatmapViewProperties represents options for heatmap view in Chronograf
|
|
|
|
type HeatmapViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
ViewColors []string `json:"colors"`
|
|
|
|
BinSize int32 `json:"binSize"`
|
|
|
|
XColumn string `json:"xColumn"`
|
|
|
|
YColumn string `json:"yColumn"`
|
2019-07-26 02:56:54 +00:00
|
|
|
XDomain []float64 `json:"xDomain,omitempty"`
|
|
|
|
YDomain []float64 `json:"yDomain,omitempty"`
|
2019-05-16 22:50:51 +00:00
|
|
|
XAxisLabel string `json:"xAxisLabel"`
|
|
|
|
YAxisLabel string `json:"yAxisLabel"`
|
|
|
|
XPrefix string `json:"xPrefix"`
|
|
|
|
XSuffix string `json:"xSuffix"`
|
|
|
|
YPrefix string `json:"yPrefix"`
|
|
|
|
YSuffix string `json:"ySuffix"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
2019-05-28 20:15:29 +00:00
|
|
|
// ScatterViewProperties represents options for scatter view in Chronograf
|
|
|
|
type ScatterViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
ViewColors []string `json:"colors"`
|
|
|
|
FillColumns []string `json:"fillColumns"`
|
|
|
|
SymbolColumns []string `json:"symbolColumns"`
|
|
|
|
XColumn string `json:"xColumn"`
|
|
|
|
YColumn string `json:"yColumn"`
|
2019-07-26 02:56:54 +00:00
|
|
|
XDomain []float64 `json:"xDomain,omitempty"`
|
|
|
|
YDomain []float64 `json:"yDomain,omitempty"`
|
2019-05-28 20:15:29 +00:00
|
|
|
XAxisLabel string `json:"xAxisLabel"`
|
|
|
|
YAxisLabel string `json:"yAxisLabel"`
|
|
|
|
XPrefix string `json:"xPrefix"`
|
|
|
|
XSuffix string `json:"xSuffix"`
|
|
|
|
YPrefix string `json:"yPrefix"`
|
|
|
|
YSuffix string `json:"ySuffix"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
2019-02-28 03:16:42 +00:00
|
|
|
// GaugeViewProperties represents options for gauge view in Chronograf
|
|
|
|
type GaugeViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
Prefix string `json:"prefix"`
|
|
|
|
Suffix string `json:"suffix"`
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
DecimalPlaces DecimalPlaces `json:"decimalPlaces"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// TableViewProperties represents options for table view in Chronograf
|
|
|
|
type TableViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Queries []DashboardQuery `json:"queries"`
|
|
|
|
ViewColors []ViewColor `json:"colors"`
|
|
|
|
TableOptions TableOptions `json:"tableOptions"`
|
|
|
|
FieldOptions []RenamableField `json:"fieldOptions"`
|
|
|
|
TimeFormat string `json:"timeFormat"`
|
|
|
|
DecimalPlaces DecimalPlaces `json:"decimalPlaces"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
ShowNoteWhenEmpty bool `json:"showNoteWhenEmpty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type MarkdownViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Note string `json:"note"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// LogViewProperties represents options for log viewer in Chronograf.
|
|
|
|
type LogViewProperties struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Columns []LogViewerColumn `json:"columns"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// LogViewerColumn represents a specific column in a Log Viewer.
|
|
|
|
type LogViewerColumn struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
Position int32 `json:"position"`
|
|
|
|
Settings []LogColumnSetting `json:"settings"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// LogColumnSetting represent the settings for a specific column of a Log Viewer.
|
|
|
|
type LogColumnSetting struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
Value string `json:"value"`
|
|
|
|
Name string `json:"name,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (XYViewProperties) viewProperties() {}
|
|
|
|
func (LinePlusSingleStatProperties) viewProperties() {}
|
|
|
|
func (SingleStatViewProperties) viewProperties() {}
|
|
|
|
func (HistogramViewProperties) viewProperties() {}
|
2019-05-16 22:50:51 +00:00
|
|
|
func (HeatmapViewProperties) viewProperties() {}
|
2019-05-28 20:15:29 +00:00
|
|
|
func (ScatterViewProperties) viewProperties() {}
|
2019-02-28 03:16:42 +00:00
|
|
|
func (GaugeViewProperties) viewProperties() {}
|
|
|
|
func (TableViewProperties) viewProperties() {}
|
|
|
|
func (MarkdownViewProperties) viewProperties() {}
|
|
|
|
func (LogViewProperties) viewProperties() {}
|
2019-08-12 21:48:24 +00:00
|
|
|
func (CheckViewProperties) viewProperties() {}
|
2019-02-28 03:16:42 +00:00
|
|
|
|
|
|
|
func (v XYViewProperties) GetType() string { return v.Type }
|
|
|
|
func (v LinePlusSingleStatProperties) GetType() string { return v.Type }
|
|
|
|
func (v SingleStatViewProperties) GetType() string { return v.Type }
|
|
|
|
func (v HistogramViewProperties) GetType() string { return v.Type }
|
2019-05-16 22:50:51 +00:00
|
|
|
func (v HeatmapViewProperties) GetType() string { return v.Type }
|
2019-05-28 20:15:29 +00:00
|
|
|
func (v ScatterViewProperties) GetType() string { return v.Type }
|
2019-02-28 03:16:42 +00:00
|
|
|
func (v GaugeViewProperties) GetType() string { return v.Type }
|
|
|
|
func (v TableViewProperties) GetType() string { return v.Type }
|
|
|
|
func (v MarkdownViewProperties) GetType() string { return v.Type }
|
|
|
|
func (v LogViewProperties) GetType() string { return v.Type }
|
2019-08-12 21:48:24 +00:00
|
|
|
func (v CheckViewProperties) GetType() string { return v.Type }
|
2019-02-28 03:16:42 +00:00
|
|
|
|
|
|
|
/////////////////////////////
|
|
|
|
// Old Chronograf Types
|
|
|
|
/////////////////////////////
|
|
|
|
|
|
|
|
// DashboardQuery represents a query used in a dashboard cell
|
|
|
|
type DashboardQuery struct {
|
|
|
|
Text string `json:"text"`
|
|
|
|
EditMode string `json:"editMode"` // Either "builder" or "advanced"
|
|
|
|
Name string `json:"name"` // Term or phrase that refers to the query
|
|
|
|
BuilderConfig BuilderConfig `json:"builderConfig"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type BuilderConfig struct {
|
|
|
|
Buckets []string `json:"buckets"`
|
|
|
|
Tags []struct {
|
|
|
|
Key string `json:"key"`
|
|
|
|
Values []string `json:"values"`
|
|
|
|
} `json:"tags"`
|
|
|
|
Functions []struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
} `json:"functions"`
|
2019-05-13 22:14:05 +00:00
|
|
|
AggregateWindow struct {
|
|
|
|
Period string `json:"period"`
|
|
|
|
} `json:"aggregateWindow"`
|
2019-02-28 03:16:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Axis represents the visible extents of a visualization
|
|
|
|
type Axis struct {
|
|
|
|
Bounds []string `json:"bounds"` // bounds are an arbitrary list of client-defined strings that specify the viewport for a View
|
|
|
|
LegacyBounds [2]int64 `json:"-"` // legacy bounds are for testing a migration from an earlier version of axis
|
|
|
|
Label string `json:"label"` // label is a description of this Axis
|
|
|
|
Prefix string `json:"prefix"` // Prefix represents a label prefix for formatting axis values
|
|
|
|
Suffix string `json:"suffix"` // Suffix represents a label suffix for formatting axis values
|
|
|
|
Base string `json:"base"` // Base represents the radix for formatting axis values
|
|
|
|
Scale string `json:"scale"` // Scale is the axis formatting scale. Supported: "log", "linear"
|
|
|
|
}
|
|
|
|
|
|
|
|
// ViewColor represents the encoding of data into visualizations
|
|
|
|
type ViewColor struct {
|
|
|
|
ID string `json:"id"` // ID is the unique id of the View color
|
|
|
|
Type string `json:"type"` // Type is how the color is used. Accepted (min,max,threshold)
|
|
|
|
Hex string `json:"hex"` // Hex is the hex number of the color
|
|
|
|
Name string `json:"name"` // Name is the user-facing name of the hex color
|
|
|
|
Value float64 `json:"value"` // Value is the data value mapped to this color
|
|
|
|
}
|
|
|
|
|
|
|
|
// Legend represents the encoding of data into a legend
|
|
|
|
type Legend struct {
|
|
|
|
Type string `json:"type,omitempty"`
|
|
|
|
Orientation string `json:"orientation,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// TableOptions is a type of options for a DashboardView with type Table
|
|
|
|
type TableOptions struct {
|
|
|
|
VerticalTimeAxis bool `json:"verticalTimeAxis"`
|
|
|
|
SortBy RenamableField `json:"sortBy"`
|
|
|
|
Wrapping string `json:"wrapping"`
|
|
|
|
FixFirstColumn bool `json:"fixFirstColumn"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RenamableField is a column/row field in a DashboardView of type Table
|
|
|
|
type RenamableField struct {
|
|
|
|
InternalName string `json:"internalName"`
|
|
|
|
DisplayName string `json:"displayName"`
|
|
|
|
Visible bool `json:"visible"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecimalPlaces indicates whether decimal places should be enforced, and how many digits it should show.
|
|
|
|
type DecimalPlaces struct {
|
|
|
|
IsEnforced bool `json:"isEnforced"`
|
|
|
|
Digits int32 `json:"digits"`
|
|
|
|
}
|