303 lines
7.9 KiB
Go
303 lines
7.9 KiB
Go
package kv
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"strconv"
|
|
|
|
"github.com/influxdata/chronograf"
|
|
"github.com/influxdata/chronograf/kv/internal"
|
|
"github.com/influxdata/chronograf/organizations"
|
|
)
|
|
|
|
// Ensure organizationsStore implements chronograf.OrganizationsStore.
|
|
var _ chronograf.OrganizationsStore = &organizationsStore{}
|
|
|
|
var (
|
|
// DefaultOrganizationID is the ID of the default organization.
|
|
DefaultOrganizationID = []byte("default")
|
|
)
|
|
|
|
const (
|
|
// DefaultOrganizationName is the Name of the default organization
|
|
DefaultOrganizationName string = "Default"
|
|
// DefaultOrganizationRole is the DefaultRole for the Default organization
|
|
DefaultOrganizationRole string = "member"
|
|
)
|
|
|
|
// organizationsStore uses a kv to store and retrieve Organizations
|
|
type organizationsStore struct {
|
|
client *Service
|
|
}
|
|
|
|
// CreateDefault does a findOrCreate on the default organization
|
|
func (s *organizationsStore) CreateDefault(ctx context.Context) error {
|
|
o := chronograf.Organization{
|
|
ID: string(DefaultOrganizationID),
|
|
Name: DefaultOrganizationName,
|
|
DefaultRole: DefaultOrganizationRole,
|
|
}
|
|
|
|
return s.client.kv.Update(ctx, func(tx Tx) error {
|
|
b := tx.Bucket(organizationsBucket)
|
|
v, _ := b.Get(DefaultOrganizationID)
|
|
if v != nil {
|
|
return nil
|
|
}
|
|
if v, err := internal.MarshalOrganization(&o); err != nil {
|
|
return err
|
|
} else if err := b.Put(DefaultOrganizationID, v); err != nil {
|
|
return err
|
|
}
|
|
|
|
b = tx.Bucket(mappingsBucket)
|
|
v, _ = b.Get(DefaultOrganizationID)
|
|
if v != nil {
|
|
return nil
|
|
}
|
|
if v, err := internal.MarshalMapping(&chronograf.Mapping{
|
|
ID: string(DefaultOrganizationID),
|
|
Organization: string(DefaultOrganizationID),
|
|
Provider: chronograf.MappingWildcard,
|
|
Scheme: chronograf.MappingWildcard,
|
|
ProviderOrganization: chronograf.MappingWildcard,
|
|
}); err != nil {
|
|
return err
|
|
} else if err := b.Put(DefaultOrganizationID, v); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func (s *organizationsStore) nameIsUnique(ctx context.Context, name string) bool {
|
|
_, err := s.Get(ctx, chronograf.OrganizationQuery{Name: &name})
|
|
switch err {
|
|
case chronograf.ErrOrganizationNotFound:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// DefaultOrganizationID returns the ID of the default organization
|
|
func (s *organizationsStore) DefaultOrganization(ctx context.Context) (*chronograf.Organization, error) {
|
|
var org chronograf.Organization
|
|
if err := s.client.kv.View(ctx, func(tx Tx) error {
|
|
v, err := tx.Bucket(organizationsBucket).Get(DefaultOrganizationID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return internal.UnmarshalOrganization(v, &org)
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &org, nil
|
|
}
|
|
|
|
// Add creates a new Organization in the organizationsStore
|
|
func (s *organizationsStore) Add(ctx context.Context, o *chronograf.Organization) (*chronograf.Organization, error) {
|
|
if !s.nameIsUnique(ctx, o.Name) {
|
|
return nil, chronograf.ErrOrganizationAlreadyExists
|
|
}
|
|
err := s.client.kv.Update(ctx, func(tx Tx) error {
|
|
b := tx.Bucket(organizationsBucket)
|
|
seq, err := b.NextSequence()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
o.ID = strconv.FormatUint(seq, 10)
|
|
|
|
v, err := internal.MarshalOrganization(o)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return b.Put([]byte(o.ID), v)
|
|
})
|
|
|
|
return o, err
|
|
}
|
|
|
|
// All returns all known organizations
|
|
func (s *organizationsStore) All(ctx context.Context) ([]chronograf.Organization, error) {
|
|
var orgs []chronograf.Organization
|
|
err := s.each(ctx, func(o *chronograf.Organization) {
|
|
orgs = append(orgs, *o)
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return orgs, nil
|
|
}
|
|
|
|
// Delete the organization from organizationsStore
|
|
func (s *organizationsStore) Delete(ctx context.Context, o *chronograf.Organization) error {
|
|
if o.ID == string(DefaultOrganizationID) {
|
|
return chronograf.ErrCannotDeleteDefaultOrganization
|
|
}
|
|
_, err := s.get(ctx, o.ID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := s.client.kv.Update(ctx, func(tx Tx) error {
|
|
return tx.Bucket(organizationsBucket).Delete([]byte(o.ID))
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Dependent Delete of all resources
|
|
|
|
// Each of the associated organization stores expects organization to be
|
|
// set on the context.
|
|
if ctx == nil {
|
|
ctx = context.Background() // context could be possible nil before go 1.15, see https://github.com/golang/go/issues/40737
|
|
}
|
|
ctx = context.WithValue(ctx, organizations.ContextKey, o.ID)
|
|
|
|
sourcesStore := organizations.NewSourcesStore(s.client.SourcesStore(), o.ID)
|
|
sources, err := sourcesStore.All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, source := range sources {
|
|
if err := sourcesStore.Delete(ctx, source); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
serversStore := organizations.NewServersStore(s.client.ServersStore(), o.ID)
|
|
servers, err := serversStore.All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, server := range servers {
|
|
if err := serversStore.Delete(ctx, server); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
dashboardsStore := organizations.NewDashboardsStore(s.client.DashboardsStore(), o.ID)
|
|
dashboards, err := dashboardsStore.All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, dashboard := range dashboards {
|
|
if err := dashboardsStore.Delete(ctx, dashboard); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
usersStore := organizations.NewUsersStore(s.client.UsersStore(), o.ID)
|
|
users, err := usersStore.All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, user := range users {
|
|
if err := usersStore.Delete(ctx, &user); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
mappings, err := s.client.MappingsStore().All(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, mapping := range mappings {
|
|
if mapping.Organization == o.ID {
|
|
if err := s.client.MappingsStore().Delete(ctx, &mapping); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *organizationsStore) get(ctx context.Context, id string) (*chronograf.Organization, error) {
|
|
var o chronograf.Organization
|
|
err := s.client.kv.View(ctx, func(tx Tx) error {
|
|
v, err := tx.Bucket(organizationsBucket).Get([]byte(id))
|
|
if v == nil || err != nil {
|
|
return chronograf.ErrOrganizationNotFound
|
|
}
|
|
return internal.UnmarshalOrganization(v, &o)
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &o, nil
|
|
}
|
|
|
|
func (s *organizationsStore) each(ctx context.Context, fn func(*chronograf.Organization)) error {
|
|
return s.client.kv.View(ctx, func(tx Tx) error {
|
|
return tx.Bucket(organizationsBucket).ForEach(func(k, v []byte) error {
|
|
var org chronograf.Organization
|
|
if err := internal.UnmarshalOrganization(v, &org); err != nil {
|
|
return err
|
|
}
|
|
fn(&org)
|
|
return nil
|
|
})
|
|
})
|
|
}
|
|
|
|
// Get returns a Organization if the id exists.
|
|
// If an ID is provided in the query, the lookup time for an organization will be O(1).
|
|
// If Name is provided, the lookup time will be O(n).
|
|
// Get expects that only one of ID or Name will be specified, but will prefer ID over Name if both are specified.
|
|
func (s *organizationsStore) Get(ctx context.Context, q chronograf.OrganizationQuery) (*chronograf.Organization, error) {
|
|
if q.ID != nil {
|
|
return s.get(ctx, *q.ID)
|
|
}
|
|
|
|
if q.Name != nil {
|
|
var org *chronograf.Organization
|
|
err := s.each(ctx, func(o *chronograf.Organization) {
|
|
if org != nil {
|
|
return
|
|
}
|
|
|
|
if o.Name == *q.Name {
|
|
org = o
|
|
}
|
|
})
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if org == nil {
|
|
return nil, chronograf.ErrOrganizationNotFound
|
|
}
|
|
|
|
return org, nil
|
|
}
|
|
return nil, errors.New("must specify either ID, or Name in OrganizationQuery")
|
|
}
|
|
|
|
// Update the organization in organizationsStore
|
|
func (s *organizationsStore) Update(ctx context.Context, o *chronograf.Organization) error {
|
|
org, err := s.get(ctx, o.ID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if o.Name != org.Name && !s.nameIsUnique(ctx, o.Name) {
|
|
return chronograf.ErrOrganizationAlreadyExists
|
|
}
|
|
return s.client.kv.Update(ctx, func(tx Tx) error {
|
|
if v, err := internal.MarshalOrganization(o); err != nil {
|
|
return err
|
|
} else if err := tx.Bucket(organizationsBucket).Put([]byte(o.ID), v); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
}
|