influxdb/kv/dashboard.go

977 lines
22 KiB
Go

package kv
import (
"bytes"
"context"
"encoding/json"
"time"
"go.uber.org/zap"
influxdb "github.com/influxdata/influxdb"
icontext "github.com/influxdata/influxdb/context"
)
var (
dashboardBucket = []byte("dashboardsv2")
orgDashboardIndex = []byte("orgsdashboardsv1")
dashboardCellViewBucket = []byte("dashboardcellviewsv1")
)
// TODO(desa): what do we want these to be?
const (
dashboardCreatedEvent = "Dashboard Created"
dashboardUpdatedEvent = "Dashboard Updated"
dashboardRemovedEvent = "Dashboard Removed"
dashboardCellsReplacedEvent = "Dashboard Cells Replaced"
dashboardCellAddedEvent = "Dashboard Cell Added"
dashboardCellRemovedEvent = "Dashboard Cell Removed"
dashboardCellUpdatedEvent = "Dashboard Cell Updated"
)
var _ influxdb.DashboardService = (*Service)(nil)
var _ influxdb.DashboardOperationLogService = (*Service)(nil)
func (s *Service) initializeDashboards(ctx context.Context, tx Tx) error {
if _, err := tx.Bucket(dashboardBucket); err != nil {
return err
}
if _, err := tx.Bucket(orgDashboardIndex); err != nil {
return err
}
if _, err := tx.Bucket(dashboardCellViewBucket); err != nil {
return err
}
return nil
}
// FindDashboardByID retrieves a dashboard by id.
func (s *Service) FindDashboardByID(ctx context.Context, id influxdb.ID) (*influxdb.Dashboard, error) {
var d *influxdb.Dashboard
err := s.kv.View(ctx, func(tx Tx) error {
dash, err := s.findDashboardByID(ctx, tx, id)
if err != nil {
return err
}
d = dash
return nil
})
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return d, nil
}
func (s *Service) findDashboardByID(ctx context.Context, tx Tx, id influxdb.ID) (*influxdb.Dashboard, error) {
encodedID, err := id.Encode()
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
b, err := tx.Bucket(dashboardBucket)
if err != nil {
return nil, err
}
v, err := b.Get(encodedID)
if IsNotFound(err) {
return nil, &influxdb.Error{
Code: influxdb.ENotFound,
Msg: influxdb.ErrDashboardNotFound,
}
}
if err != nil {
return nil, err
}
var d influxdb.Dashboard
if err := json.Unmarshal(v, &d); err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return &d, nil
}
// FindDashboard retrieves a dashboard using an arbitrary dashboard filter.
func (s *Service) FindDashboard(ctx context.Context, filter influxdb.DashboardFilter, opts ...influxdb.FindOptions) (*influxdb.Dashboard, error) {
if len(filter.IDs) == 1 {
return s.FindDashboardByID(ctx, *filter.IDs[0])
}
var d *influxdb.Dashboard
err := s.kv.View(ctx, func(tx Tx) error {
filterFn := filterDashboardsFn(filter)
return s.forEachDashboard(ctx, tx, opts[0].Descending, func(dash *influxdb.Dashboard) bool {
if filterFn(dash) {
d = dash
return false
}
return true
})
})
if err != nil {
return nil, err
}
if d == nil {
return nil, &influxdb.Error{
Code: influxdb.ENotFound,
Msg: influxdb.ErrDashboardNotFound,
}
}
return d, nil
}
func filterDashboardsFn(filter influxdb.DashboardFilter) func(d *influxdb.Dashboard) bool {
if len(filter.IDs) > 0 {
m := map[string]struct{}{}
for _, id := range filter.IDs {
m[id.String()] = struct{}{}
}
return func(d *influxdb.Dashboard) bool {
_, ok := m[d.ID.String()]
return ok
}
}
return func(d *influxdb.Dashboard) bool { return true }
}
// FindDashboards retrives all dashboards that match an arbitrary dashboard filter.
func (s *Service) FindDashboards(ctx context.Context, filter influxdb.DashboardFilter, opts influxdb.FindOptions) ([]*influxdb.Dashboard, int, error) {
ds := []*influxdb.Dashboard{}
if len(filter.IDs) == 1 {
d, err := s.FindDashboardByID(ctx, *filter.IDs[0])
if err != nil && influxdb.ErrorCode(err) != influxdb.ENotFound {
return ds, 0, &influxdb.Error{
Err: err,
}
}
if d == nil {
return ds, 0, nil
}
return []*influxdb.Dashboard{d}, 1, nil
}
err := s.kv.View(ctx, func(tx Tx) error {
dashs, err := s.findDashboards(ctx, tx, filter, opts)
if err != nil && influxdb.ErrorCode(err) != influxdb.ENotFound {
return err
}
ds = dashs
return nil
})
if err != nil {
return nil, 0, &influxdb.Error{
Err: err,
}
}
influxdb.SortDashboards(opts, ds)
return ds, len(ds), nil
}
func (s *Service) findOrganizationDashboards(ctx context.Context, tx Tx, orgID influxdb.ID) ([]*influxdb.Dashboard, error) {
idx, err := tx.Bucket(orgDashboardIndex)
if err != nil {
return nil, err
}
// TODO(desa): support find options.
cur, err := idx.Cursor()
if err != nil {
return nil, err
}
prefix, err := orgID.Encode()
if err != nil {
return nil, err
}
ds := []*influxdb.Dashboard{}
for k, _ := cur.Seek(prefix); bytes.HasPrefix(k, prefix); k, _ = cur.Next() {
_, id, err := decodeOrgDashboardIndexKey(k)
if err != nil {
return nil, err
}
d, err := s.findDashboardByID(ctx, tx, id)
if err != nil {
return nil, err
}
ds = append(ds, d)
}
return ds, nil
}
func decodeOrgDashboardIndexKey(indexKey []byte) (orgID influxdb.ID, dashID influxdb.ID, err error) {
if len(indexKey) != 2*influxdb.IDLength {
return 0, 0, &influxdb.Error{Code: influxdb.EInternal, Msg: "malformed org dashboard index key (please report this error)"}
}
if err := (&orgID).Decode(indexKey[:influxdb.IDLength]); err != nil {
return 0, 0, &influxdb.Error{Code: influxdb.EInternal, Msg: "bad org id", Err: influxdb.ErrInvalidID}
}
if err := (&dashID).Decode(indexKey[influxdb.IDLength:]); err != nil {
return 0, 0, &influxdb.Error{Code: influxdb.EInternal, Msg: "bad dashboard id", Err: influxdb.ErrInvalidID}
}
return orgID, dashID, nil
}
func (s *Service) findDashboards(ctx context.Context, tx Tx, filter influxdb.DashboardFilter, opts ...influxdb.FindOptions) ([]*influxdb.Dashboard, error) {
if filter.OrganizationID != nil {
return s.findOrganizationDashboards(ctx, tx, *filter.OrganizationID)
}
if filter.Organization != nil {
o, err := s.findOrganizationByName(ctx, tx, *filter.Organization)
if err != nil {
return nil, err
}
return s.findOrganizationDashboards(ctx, tx, o.ID)
}
var offset, limit, count int
var descending bool
if len(opts) > 0 {
offset = opts[0].Offset
limit = opts[0].Limit
descending = opts[0].Descending
}
ds := []*influxdb.Dashboard{}
filterFn := filterDashboardsFn(filter)
err := s.forEachDashboard(ctx, tx, descending, func(d *influxdb.Dashboard) bool {
if filterFn(d) {
if count >= offset {
ds = append(ds, d)
}
count++
}
if limit > 0 && len(ds) >= limit {
return false
}
return true
})
if err != nil {
return nil, err
}
return ds, nil
}
// CreateDashboard creates a influxdb dashboard and sets d.ID.
func (s *Service) CreateDashboard(ctx context.Context, d *influxdb.Dashboard) error {
err := s.kv.Update(ctx, func(tx Tx) error {
d.ID = s.IDGenerator.ID()
for _, cell := range d.Cells {
cell.ID = s.IDGenerator.ID()
if err := s.createCellView(ctx, tx, d.ID, cell.ID, nil); err != nil {
return err
}
}
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardCreatedEvent); err != nil {
return err
}
if err := s.putOrganizationDashboardIndex(ctx, tx, d); err != nil {
return err
}
d.Meta.CreatedAt = s.Now()
d.Meta.UpdatedAt = s.Now()
if err := s.putDashboardWithMeta(ctx, tx, d); err != nil {
return err
}
if err := s.addDashboardOwner(ctx, tx, d.ID); err != nil {
s.Logger.Info("failed to make user owner of organization", zap.Error(err))
}
return nil
})
if err != nil {
return &influxdb.Error{
Err: err,
}
}
return nil
}
// addDashboardOwner attempts to create a user resource mapping for the user on the
// authorizer found on context. If no authorizer is found on context if returns an error.
func (s *Service) addDashboardOwner(ctx context.Context, tx Tx, orgID influxdb.ID) error {
return s.addResourceOwner(ctx, tx, influxdb.DashboardsResourceType, orgID)
}
func (s *Service) createCellView(ctx context.Context, tx Tx, dashID, cellID influxdb.ID, view *influxdb.View) error {
if view == nil {
// If not view exists create the view
view = &influxdb.View{}
}
// TODO: this is temporary until we can fully remove the view service.
view.ID = cellID
return s.putDashboardCellView(ctx, tx, dashID, cellID, view)
}
// ReplaceDashboardCells updates the positions of each cell in a dashboard concurrently.
func (s *Service) ReplaceDashboardCells(ctx context.Context, id influxdb.ID, cs []*influxdb.Cell) error {
err := s.kv.Update(ctx, func(tx Tx) error {
d, err := s.findDashboardByID(ctx, tx, id)
if err != nil {
return err
}
ids := map[string]*influxdb.Cell{}
for _, cell := range d.Cells {
ids[cell.ID.String()] = cell
}
for _, cell := range cs {
if !cell.ID.Valid() {
return &influxdb.Error{
Code: influxdb.EInvalid,
Msg: "cannot provide empty cell id",
}
}
if _, ok := ids[cell.ID.String()]; !ok {
return &influxdb.Error{
Code: influxdb.EConflict,
Msg: "cannot replace cells that were not already present",
}
}
}
d.Cells = cs
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardCellsReplacedEvent); err != nil {
return err
}
return s.putDashboardWithMeta(ctx, tx, d)
})
if err != nil {
return &influxdb.Error{
Err: err,
}
}
return nil
}
func (s *Service) addDashboardCell(ctx context.Context, tx Tx, id influxdb.ID, cell *influxdb.Cell, opts influxdb.AddDashboardCellOptions) error {
d, err := s.findDashboardByID(ctx, tx, id)
if err != nil {
return err
}
cell.ID = s.IDGenerator.ID()
if err := s.createCellView(ctx, tx, id, cell.ID, opts.View); err != nil {
return err
}
d.Cells = append(d.Cells, cell)
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardCellAddedEvent); err != nil {
return err
}
return s.putDashboardWithMeta(ctx, tx, d)
}
// AddDashboardCell adds a cell to a dashboard and sets the cells ID.
func (s *Service) AddDashboardCell(ctx context.Context, id influxdb.ID, cell *influxdb.Cell, opts influxdb.AddDashboardCellOptions) error {
err := s.kv.Update(ctx, func(tx Tx) error {
return s.addDashboardCell(ctx, tx, id, cell, opts)
})
if err != nil {
return &influxdb.Error{
Err: err,
}
}
return nil
}
// RemoveDashboardCell removes a cell from a dashboard.
func (s *Service) RemoveDashboardCell(ctx context.Context, dashboardID, cellID influxdb.ID) error {
return s.kv.Update(ctx, func(tx Tx) error {
d, err := s.findDashboardByID(ctx, tx, dashboardID)
if err != nil {
return &influxdb.Error{
Err: err,
}
}
idx := -1
for i, cell := range d.Cells {
if cell.ID == cellID {
idx = i
break
}
}
if idx == -1 {
return &influxdb.Error{
Code: influxdb.ENotFound,
Msg: influxdb.ErrCellNotFound,
}
}
if err := s.deleteDashboardCellView(ctx, tx, d.ID, d.Cells[idx].ID); err != nil {
return &influxdb.Error{
Err: err,
}
}
d.Cells = append(d.Cells[:idx], d.Cells[idx+1:]...)
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardCellRemovedEvent); err != nil {
return &influxdb.Error{
Err: err,
}
}
if err := s.putDashboardWithMeta(ctx, tx, d); err != nil {
return &influxdb.Error{
Err: err,
}
}
return nil
})
}
// GetDashboardCellView retrieves the view for a dashboard cell.
func (s *Service) GetDashboardCellView(ctx context.Context, dashboardID, cellID influxdb.ID) (*influxdb.View, error) {
var v *influxdb.View
err := s.kv.View(ctx, func(tx Tx) error {
view, err := s.findDashboardCellView(ctx, tx, dashboardID, cellID)
if err != nil {
return err
}
v = view
return nil
})
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return v, nil
}
func (s *Service) findDashboardCellView(ctx context.Context, tx Tx, dashboardID, cellID influxdb.ID) (*influxdb.View, error) {
k, err := encodeDashboardCellViewID(dashboardID, cellID)
if err != nil {
return nil, influxdb.NewError(influxdb.WithErrorErr(err))
}
vb, err := tx.Bucket(dashboardCellViewBucket)
if err != nil {
return nil, err
}
v, err := vb.Get(k)
if IsNotFound(err) {
return nil, influxdb.NewError(influxdb.WithErrorCode(influxdb.ENotFound), influxdb.WithErrorMsg(influxdb.ErrViewNotFound))
}
if err != nil {
return nil, err
}
view := &influxdb.View{}
if err := json.Unmarshal(v, view); err != nil {
return nil, influxdb.NewError(influxdb.WithErrorErr(err))
}
return view, nil
}
func (s *Service) deleteDashboardCellView(ctx context.Context, tx Tx, dashboardID, cellID influxdb.ID) error {
k, err := encodeDashboardCellViewID(dashboardID, cellID)
if err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
vb, err := tx.Bucket(dashboardCellViewBucket)
if err != nil {
return err
}
if err := vb.Delete(k); err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
return nil
}
func (s *Service) putDashboardCellView(ctx context.Context, tx Tx, dashboardID, cellID influxdb.ID, view *influxdb.View) error {
k, err := encodeDashboardCellViewID(dashboardID, cellID)
if err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
v, err := json.Marshal(view)
if err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
vb, err := tx.Bucket(dashboardCellViewBucket)
if err != nil {
return err
}
if err := vb.Put(k, v); err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
return nil
}
func encodeDashboardCellViewID(dashID, cellID influxdb.ID) ([]byte, error) {
did, err := dashID.Encode()
if err != nil {
return nil, err
}
cid, err := cellID.Encode()
if err != nil {
return nil, err
}
buf := bytes.NewBuffer(nil)
if _, err := buf.Write(did); err != nil {
return nil, err
}
if _, err := buf.Write(cid); err != nil {
return nil, err
}
return buf.Bytes(), nil
}
// UpdateDashboardCellView updates the view for a dashboard cell.
func (s *Service) UpdateDashboardCellView(ctx context.Context, dashboardID, cellID influxdb.ID, upd influxdb.ViewUpdate) (*influxdb.View, error) {
var v *influxdb.View
err := s.kv.Update(ctx, func(tx Tx) error {
view, err := s.findDashboardCellView(ctx, tx, dashboardID, cellID)
if err != nil {
return err
}
if err := upd.Apply(view); err != nil {
return err
}
if err := s.putDashboardCellView(ctx, tx, dashboardID, cellID, view); err != nil {
return err
}
v = view
return nil
})
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return v, nil
}
// UpdateDashboardCell udpates a cell on a dashboard.
func (s *Service) UpdateDashboardCell(ctx context.Context, dashboardID, cellID influxdb.ID, upd influxdb.CellUpdate) (*influxdb.Cell, error) {
if err := upd.Valid(); err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
var cell *influxdb.Cell
err := s.kv.Update(ctx, func(tx Tx) error {
d, err := s.findDashboardByID(ctx, tx, dashboardID)
if err != nil {
return err
}
idx := -1
for i, cell := range d.Cells {
if cell.ID == cellID {
idx = i
break
}
}
if idx == -1 {
return &influxdb.Error{
Code: influxdb.ENotFound,
Msg: influxdb.ErrCellNotFound,
}
}
if err := upd.Apply(d.Cells[idx]); err != nil {
return err
}
cell = d.Cells[idx]
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardCellUpdatedEvent); err != nil {
return err
}
return s.putDashboardWithMeta(ctx, tx, d)
})
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return cell, nil
}
// PutDashboard will put a dashboard without setting an ID.
func (s *Service) PutDashboard(ctx context.Context, d *influxdb.Dashboard) error {
return s.kv.Update(ctx, func(tx Tx) error {
for _, cell := range d.Cells {
if err := s.createCellView(ctx, tx, d.ID, cell.ID, nil); err != nil {
return err
}
}
if err := s.putOrganizationDashboardIndex(ctx, tx, d); err != nil {
return err
}
return s.putDashboard(ctx, tx, d)
})
}
func encodeOrgDashboardIndex(orgID influxdb.ID, dashID influxdb.ID) ([]byte, error) {
oid, err := orgID.Encode()
if err != nil {
return nil, err
}
did, err := dashID.Encode()
if err != nil {
return nil, err
}
key := make([]byte, 0, len(oid)+len(did))
key = append(key, oid...)
key = append(key, did...)
return key, nil
}
func (s *Service) putOrganizationDashboardIndex(ctx context.Context, tx Tx, d *influxdb.Dashboard) error {
k, err := encodeOrgDashboardIndex(d.OrganizationID, d.ID)
if err != nil {
return err
}
idx, err := tx.Bucket(orgDashboardIndex)
if err != nil {
return err
}
if err := idx.Put(k, nil); err != nil {
return err
}
return nil
}
func (s *Service) removeOrganizationDashboardIndex(ctx context.Context, tx Tx, d *influxdb.Dashboard) error {
k, err := encodeOrgDashboardIndex(d.OrganizationID, d.ID)
if err != nil {
return err
}
idx, err := tx.Bucket(orgDashboardIndex)
if err != nil {
return err
}
if err := idx.Delete(k); err != nil {
return err
}
return nil
}
func (s *Service) putDashboard(ctx context.Context, tx Tx, d *influxdb.Dashboard) error {
v, err := json.Marshal(d)
if err != nil {
return err
}
encodedID, err := d.ID.Encode()
if err != nil {
return err
}
b, err := tx.Bucket(dashboardBucket)
if err != nil {
return err
}
if err := b.Put(encodedID, v); err != nil {
return err
}
return nil
}
func (s *Service) putDashboardWithMeta(ctx context.Context, tx Tx, d *influxdb.Dashboard) error {
// TODO(desa): don't populate this here. use the first/last methods of the oplog to get meta fields.
d.Meta.UpdatedAt = s.Now()
return s.putDashboard(ctx, tx, d)
}
// forEachDashboard will iterate through all dashboards while fn returns true.
func (s *Service) forEachDashboard(ctx context.Context, tx Tx, descending bool, fn func(*influxdb.Dashboard) bool) error {
b, err := tx.Bucket(dashboardBucket)
if err != nil {
return err
}
cur, err := b.Cursor()
if err != nil {
return err
}
var k, v []byte
if descending {
k, v = cur.Last()
} else {
k, v = cur.First()
}
for k != nil {
d := &influxdb.Dashboard{}
if err := json.Unmarshal(v, d); err != nil {
return err
}
if !fn(d) {
break
}
if descending {
k, v = cur.Prev()
} else {
k, v = cur.Next()
}
}
return nil
}
// UpdateDashboard updates a dashboard according the parameters set on upd.
func (s *Service) UpdateDashboard(ctx context.Context, id influxdb.ID, upd influxdb.DashboardUpdate) (*influxdb.Dashboard, error) {
if err := upd.Valid(); err != nil {
return nil, err
}
var d *influxdb.Dashboard
err := s.kv.Update(ctx, func(tx Tx) error {
dash, err := s.updateDashboard(ctx, tx, id, upd)
if err != nil {
return err
}
d = dash
return nil
})
if err != nil {
return nil, &influxdb.Error{
Err: err,
}
}
return d, err
}
func (s *Service) updateDashboard(ctx context.Context, tx Tx, id influxdb.ID, upd influxdb.DashboardUpdate) (*influxdb.Dashboard, error) {
d, err := s.findDashboardByID(ctx, tx, id)
if err != nil {
return nil, err
}
if err := upd.Apply(d); err != nil {
return nil, err
}
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardUpdatedEvent); err != nil {
return nil, err
}
if err := s.putDashboardWithMeta(ctx, tx, d); err != nil {
return nil, err
}
return d, nil
}
// DeleteDashboard deletes a dashboard and prunes it from the index.
func (s *Service) DeleteDashboard(ctx context.Context, id influxdb.ID) error {
return s.kv.Update(ctx, func(tx Tx) error {
if pe := s.deleteDashboard(ctx, tx, id); pe != nil {
return &influxdb.Error{
Err: pe,
}
}
return nil
})
}
func (s *Service) deleteDashboard(ctx context.Context, tx Tx, id influxdb.ID) error {
d, pe := s.findDashboardByID(ctx, tx, id)
if pe != nil {
return pe
}
for _, cell := range d.Cells {
if err := s.deleteDashboardCellView(ctx, tx, d.ID, cell.ID); err != nil {
return &influxdb.Error{
Err: err,
}
}
}
encodedID, err := id.Encode()
if err != nil {
return &influxdb.Error{
Err: err,
}
}
if err := s.removeOrganizationDashboardIndex(ctx, tx, d); err != nil {
return influxdb.NewError(influxdb.WithErrorErr(err))
}
b, err := tx.Bucket(dashboardBucket)
if err != nil {
return err
}
if err := b.Delete(encodedID); err != nil {
return &influxdb.Error{
Err: err,
}
}
err = s.deleteUserResourceMappings(ctx, tx, influxdb.UserResourceMappingFilter{
ResourceID: id,
ResourceType: influxdb.DashboardsResourceType,
})
if err != nil {
return &influxdb.Error{
Err: err,
}
}
if err := s.appendDashboardEventToLog(ctx, tx, d.ID, dashboardRemovedEvent); err != nil {
return &influxdb.Error{
Err: err,
}
}
return nil
}
const dashboardOperationLogKeyPrefix = "dashboard"
func encodeDashboardOperationLogKey(id influxdb.ID) ([]byte, error) {
buf, err := id.Encode()
if err != nil {
return nil, err
}
return append([]byte(dashboardOperationLogKeyPrefix), buf...), nil
}
// GetDashboardOperationLog retrieves a dashboards operation log.
func (s *Service) GetDashboardOperationLog(ctx context.Context, id influxdb.ID, opts influxdb.FindOptions) ([]*influxdb.OperationLogEntry, int, error) {
// TODO(desa): might be worthwhile to allocate a slice of size opts.Limit
log := []*influxdb.OperationLogEntry{}
err := s.kv.View(ctx, func(tx Tx) error {
key, err := encodeDashboardOperationLogKey(id)
if err != nil {
return err
}
return s.forEachLogEntry(ctx, tx, key, opts, func(v []byte, t time.Time) error {
e := &influxdb.OperationLogEntry{}
if err := json.Unmarshal(v, e); err != nil {
return err
}
e.Time = t
log = append(log, e)
return nil
})
})
if err != nil && err != errKeyValueLogBoundsNotFound {
return nil, 0, err
}
return log, len(log), nil
}
func (s *Service) appendDashboardEventToLog(ctx context.Context, tx Tx, id influxdb.ID, st string) error {
e := &influxdb.OperationLogEntry{
Description: st,
}
// TODO(desa): this is fragile and non explicit since it requires an authorizer to be on context. It should be
// replaced with a higher level transaction so that adding to the log can take place in the http handler
// where the userID will exist explicitly.
a, err := icontext.GetAuthorizer(ctx)
if err == nil {
// Add the user to the log if you can, but don't error if its not there.
e.UserID = a.GetUserID()
}
v, err := json.Marshal(e)
if err != nil {
return err
}
k, err := encodeDashboardOperationLogKey(id)
if err != nil {
return err
}
return s.addLogEntry(ctx, tx, k, v, s.Now())
}