2018-12-03 16:07:08 +00:00
|
|
|
package inmem
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"path"
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
"github.com/influxdata/influxdb"
|
2018-12-03 16:07:08 +00:00
|
|
|
)
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) loadLabel(ctx context.Context, id influxdb.ID) (*influxdb.Label, error) {
|
2019-01-18 19:03:36 +00:00
|
|
|
i, ok := s.labelKV.Load(id.String())
|
2018-12-03 16:07:08 +00:00
|
|
|
if !ok {
|
2019-01-22 12:57:43 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.ENotFound,
|
|
|
|
Err: influxdb.ErrLabelNotFound,
|
2019-01-18 19:03:36 +00:00
|
|
|
}
|
2018-12-03 16:07:08 +00:00
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
l, ok := i.(influxdb.Label)
|
2018-12-03 16:07:08 +00:00
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("type %T is not a label", i)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &l, nil
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) forEachLabel(ctx context.Context, fn func(m *influxdb.Label) bool) error {
|
2018-12-03 16:07:08 +00:00
|
|
|
var err error
|
|
|
|
s.labelKV.Range(func(k, v interface{}) bool {
|
2019-01-22 12:57:43 +00:00
|
|
|
l, ok := v.(influxdb.Label)
|
2018-12-03 16:07:08 +00:00
|
|
|
if !ok {
|
|
|
|
err = fmt.Errorf("type %T is not a label", v)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return fn(&l)
|
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) forEachLabelMapping(ctx context.Context, fn func(m *influxdb.LabelMapping) bool) error {
|
2019-01-18 19:03:36 +00:00
|
|
|
var err error
|
|
|
|
s.labelMappingKV.Range(func(k, v interface{}) bool {
|
2019-01-22 12:57:43 +00:00
|
|
|
m, ok := v.(influxdb.LabelMapping)
|
2019-01-18 19:03:36 +00:00
|
|
|
if !ok {
|
|
|
|
err = fmt.Errorf("type %T is not a label mapping", v)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return fn(&m)
|
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) filterLabels(ctx context.Context, fn func(m *influxdb.Label) bool) ([]*influxdb.Label, error) {
|
|
|
|
labels := []*influxdb.Label{}
|
|
|
|
err := s.forEachLabel(ctx, func(l *influxdb.Label) bool {
|
2018-12-03 16:07:08 +00:00
|
|
|
if fn(l) {
|
|
|
|
labels = append(labels, l)
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return labels, nil
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) filterLabelMappings(ctx context.Context, fn func(m *influxdb.LabelMapping) bool) ([]*influxdb.LabelMapping, error) {
|
|
|
|
mappings := []*influxdb.LabelMapping{}
|
|
|
|
err := s.forEachLabelMapping(ctx, func(m *influxdb.LabelMapping) bool {
|
2019-01-18 19:03:36 +00:00
|
|
|
if fn(m) {
|
|
|
|
mappings = append(mappings, m)
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return mappings, nil
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
func encodeLabelMappingKey(m *influxdb.LabelMapping) string {
|
2019-01-18 19:03:36 +00:00
|
|
|
return path.Join(m.ResourceID.String(), m.LabelID.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindLabelByID returns a single user by ID.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) FindLabelByID(ctx context.Context, id influxdb.ID) (*influxdb.Label, error) {
|
2019-01-18 19:03:36 +00:00
|
|
|
return s.loadLabel(ctx, id)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindLabels will retrieve a list of labels from storage.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) FindLabels(ctx context.Context, filter influxdb.LabelFilter, opt ...influxdb.FindOptions) ([]*influxdb.Label, error) {
|
|
|
|
filterFunc := func(label *influxdb.Label) bool {
|
2019-01-18 19:03:36 +00:00
|
|
|
return (filter.Name == "" || (filter.Name == label.Name))
|
2018-12-03 16:07:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
labels, err := s.filterLabels(ctx, filterFunc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return labels, nil
|
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
// FindResourceLabels returns a list of labels that are mapped to a resource.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) FindResourceLabels(ctx context.Context, filter influxdb.LabelMappingFilter) ([]*influxdb.Label, error) {
|
|
|
|
filterFunc := func(mapping *influxdb.LabelMapping) bool {
|
2019-01-18 19:03:36 +00:00
|
|
|
return (filter.ResourceID.String() == mapping.ResourceID.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
mappings, err := s.filterLabelMappings(ctx, filterFunc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-01-22 12:57:43 +00:00
|
|
|
ls := []*influxdb.Label{}
|
2019-01-18 19:03:36 +00:00
|
|
|
for _, m := range mappings {
|
2019-01-22 17:55:04 +00:00
|
|
|
l, err := s.FindLabelByID(ctx, m.LabelID)
|
2019-01-18 19:03:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ls = append(ls, l)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ls, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateLabel creates a new label.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) CreateLabel(ctx context.Context, l *influxdb.Label) error {
|
2019-01-18 19:03:36 +00:00
|
|
|
l.ID = s.IDGenerator.ID()
|
|
|
|
s.labelKV.Store(l.ID, *l)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateLabelMapping creates a mapping that associates a label to a resource.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) CreateLabelMapping(ctx context.Context, m *influxdb.LabelMapping) error {
|
2019-01-22 17:55:04 +00:00
|
|
|
_, err := s.FindLabelByID(ctx, m.LabelID)
|
2019-01-18 19:03:36 +00:00
|
|
|
if err != nil {
|
2019-01-22 12:57:43 +00:00
|
|
|
return &influxdb.Error{
|
2019-01-18 19:03:36 +00:00
|
|
|
Err: err,
|
2019-01-22 12:57:43 +00:00
|
|
|
Op: influxdb.OpCreateLabel,
|
2018-12-18 07:59:04 +00:00
|
|
|
}
|
2018-12-03 16:07:08 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
s.labelMappingKV.Store(encodeLabelMappingKey(m), *m)
|
2018-12-03 16:07:08 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
// UpdateLabel updates a label.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) UpdateLabel(ctx context.Context, id influxdb.ID, upd influxdb.LabelUpdate) (*influxdb.Label, error) {
|
2019-01-18 19:03:36 +00:00
|
|
|
label, err := s.FindLabelByID(ctx, id)
|
2018-12-18 07:59:04 +00:00
|
|
|
if err != nil {
|
2019-01-22 12:57:43 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.ENotFound,
|
|
|
|
Op: OpPrefix + influxdb.OpUpdateLabel,
|
|
|
|
Err: influxdb.ErrLabelNotFound,
|
2018-12-18 07:59:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-13 00:28:15 +00:00
|
|
|
if len(upd.Properties) > 0 && label.Properties == nil {
|
2018-12-20 21:41:20 +00:00
|
|
|
label.Properties = make(map[string]string)
|
2018-12-20 20:52:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range upd.Properties {
|
|
|
|
if v == "" {
|
2018-12-20 21:41:20 +00:00
|
|
|
delete(label.Properties, k)
|
2018-12-20 20:52:48 +00:00
|
|
|
} else {
|
2018-12-20 21:41:20 +00:00
|
|
|
label.Properties[k] = v
|
2018-12-20 20:52:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-13 00:28:15 +00:00
|
|
|
if upd.Name != "" {
|
|
|
|
label.Name = upd.Name
|
|
|
|
}
|
|
|
|
|
2018-12-18 07:59:04 +00:00
|
|
|
if err := label.Validate(); err != nil {
|
2019-01-22 12:57:43 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
|
|
|
Op: OpPrefix + influxdb.OpUpdateLabel,
|
2018-12-18 17:22:48 +00:00
|
|
|
Err: err,
|
2018-12-18 07:59:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
s.labelKV.Store(label.ID.String(), *label)
|
2018-12-18 07:59:04 +00:00
|
|
|
|
|
|
|
return label, nil
|
2018-12-18 06:30:41 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
// PutLabel writes a label directly to the database without generating IDs
|
|
|
|
// or making checks.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) PutLabel(ctx context.Context, l *influxdb.Label) error {
|
2019-01-18 19:03:36 +00:00
|
|
|
s.labelKV.Store(l.ID.String(), *l)
|
2018-12-03 16:07:08 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
// DeleteLabel deletes a label.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) DeleteLabel(ctx context.Context, id influxdb.ID) error {
|
2019-01-18 19:03:36 +00:00
|
|
|
label, err := s.FindLabelByID(ctx, id)
|
2018-12-03 16:07:08 +00:00
|
|
|
if label == nil && err != nil {
|
2019-01-22 12:57:43 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.ENotFound,
|
|
|
|
Op: OpPrefix + influxdb.OpDeleteLabel,
|
|
|
|
Err: influxdb.ErrLabelNotFound,
|
2018-12-18 07:59:04 +00:00
|
|
|
}
|
2018-12-03 16:07:08 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
s.labelKV.Delete(id.String())
|
2018-12-03 16:07:08 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-01-18 19:03:36 +00:00
|
|
|
// DeleteLabelMapping deletes a label mapping.
|
2019-01-22 12:57:43 +00:00
|
|
|
func (s *Service) DeleteLabelMapping(ctx context.Context, m *influxdb.LabelMapping) error {
|
2019-01-18 19:03:36 +00:00
|
|
|
s.labelMappingKV.Delete(encodeLabelMappingKey(m))
|
2018-12-03 16:07:08 +00:00
|
|
|
return nil
|
|
|
|
}
|