177 lines
4.3 KiB
Go
177 lines
4.3 KiB
Go
package bolt
|
|
|
|
import (
|
|
"context"
|
|
"strconv"
|
|
|
|
"github.com/boltdb/bolt"
|
|
"github.com/influxdata/chronograf"
|
|
"github.com/influxdata/chronograf/bolt/internal"
|
|
)
|
|
|
|
// Ensure DashboardsStore implements chronograf.DashboardsStore.
|
|
var _ chronograf.DashboardsStore = &DashboardsStore{}
|
|
|
|
// DashboardsBucket is the bolt bucket dashboards are stored in
|
|
var DashboardsBucket = []byte("Dashoard")
|
|
|
|
// DashboardsStore is the bolt implementation of storing dashboards
|
|
type DashboardsStore struct {
|
|
client *Client
|
|
IDs chronograf.ID
|
|
}
|
|
|
|
// AddIDs is a migration function that adds ID information to existing dashboards
|
|
func (d *DashboardsStore) AddIDs(ctx context.Context, boards []chronograf.Dashboard) error {
|
|
for _, board := range boards {
|
|
update := false
|
|
for i, cell := range board.Cells {
|
|
// If there are is no id set, we generate one and update the dashboard
|
|
if cell.ID == "" {
|
|
id, err := d.IDs.Generate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cell.ID = id
|
|
board.Cells[i] = cell
|
|
update = true
|
|
}
|
|
}
|
|
if !update {
|
|
continue
|
|
}
|
|
if err := d.Update(ctx, board); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Migrate updates the dashboards at runtime
|
|
func (d *DashboardsStore) Migrate(ctx context.Context) error {
|
|
// 1. Add UUIDs to cells without one
|
|
boards, err := d.All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return d.AddIDs(ctx, boards)
|
|
}
|
|
|
|
// All returns all known dashboards
|
|
func (d *DashboardsStore) All(ctx context.Context) ([]chronograf.Dashboard, error) {
|
|
var srcs []chronograf.Dashboard
|
|
if err := d.client.db.View(func(tx *bolt.Tx) error {
|
|
if err := tx.Bucket(DashboardsBucket).ForEach(func(k, v []byte) error {
|
|
var src chronograf.Dashboard
|
|
if err := internal.UnmarshalDashboard(v, &src); err != nil {
|
|
return err
|
|
}
|
|
srcs = append(srcs, src)
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return srcs, nil
|
|
}
|
|
|
|
// Add creates a new Dashboard in the DashboardsStore
|
|
func (d *DashboardsStore) Add(ctx context.Context, src chronograf.Dashboard) (chronograf.Dashboard, error) {
|
|
if err := d.client.db.Update(func(tx *bolt.Tx) error {
|
|
b := tx.Bucket(DashboardsBucket)
|
|
id, _ := b.NextSequence()
|
|
|
|
src.ID = chronograf.DashboardID(id)
|
|
strID := strconv.Itoa(int(id))
|
|
for i, cell := range src.Cells {
|
|
cid, err := d.IDs.Generate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cell.ID = cid
|
|
src.Cells[i] = cell
|
|
}
|
|
if v, err := internal.MarshalDashboard(src); err != nil {
|
|
return err
|
|
} else if err := b.Put([]byte(strID), v); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return chronograf.Dashboard{}, err
|
|
}
|
|
|
|
return src, nil
|
|
}
|
|
|
|
// Get returns a Dashboard if the id exists.
|
|
func (d *DashboardsStore) Get(ctx context.Context, id chronograf.DashboardID) (chronograf.Dashboard, error) {
|
|
var src chronograf.Dashboard
|
|
if err := d.client.db.View(func(tx *bolt.Tx) error {
|
|
strID := strconv.Itoa(int(id))
|
|
if v := tx.Bucket(DashboardsBucket).Get([]byte(strID)); v == nil {
|
|
return chronograf.ErrDashboardNotFound
|
|
} else if err := internal.UnmarshalDashboard(v, &src); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return chronograf.Dashboard{}, err
|
|
}
|
|
|
|
return src, nil
|
|
}
|
|
|
|
// Delete the dashboard from DashboardsStore
|
|
func (d *DashboardsStore) Delete(ctx context.Context, dash chronograf.Dashboard) error {
|
|
if err := d.client.db.Update(func(tx *bolt.Tx) error {
|
|
strID := strconv.Itoa(int(dash.ID))
|
|
if err := tx.Bucket(DashboardsBucket).Delete([]byte(strID)); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Update the dashboard in DashboardsStore
|
|
func (d *DashboardsStore) Update(ctx context.Context, dash chronograf.Dashboard) error {
|
|
if err := d.client.db.Update(func(tx *bolt.Tx) error {
|
|
// Get an existing dashboard with the same ID.
|
|
b := tx.Bucket(DashboardsBucket)
|
|
strID := strconv.Itoa(int(dash.ID))
|
|
if v := b.Get([]byte(strID)); v == nil {
|
|
return chronograf.ErrDashboardNotFound
|
|
}
|
|
|
|
for i, cell := range dash.Cells {
|
|
if cell.ID != "" {
|
|
continue
|
|
}
|
|
cid, err := d.IDs.Generate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cell.ID = cid
|
|
dash.Cells[i] = cell
|
|
}
|
|
if v, err := internal.MarshalDashboard(dash); err != nil {
|
|
return err
|
|
} else if err := b.Put([]byte(strID), v); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|