462 lines
9.6 KiB
Go
462 lines
9.6 KiB
Go
package tenant
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
|
|
"github.com/influxdata/influxdb/v2"
|
|
"github.com/influxdata/influxdb/v2/kit/platform"
|
|
"github.com/influxdata/influxdb/v2/kit/platform/errors"
|
|
"github.com/influxdata/influxdb/v2/kv"
|
|
)
|
|
|
|
var (
|
|
bucketBucket = []byte("bucketsv1")
|
|
bucketIndex = []byte("bucketindexv1")
|
|
)
|
|
|
|
func bucketIndexKey(o platform.ID, name string) ([]byte, error) {
|
|
orgID, err := o.Encode()
|
|
|
|
if err != nil {
|
|
return nil, &errors.Error{
|
|
Code: errors.EInvalid,
|
|
Err: err,
|
|
}
|
|
}
|
|
k := make([]byte, platform.IDLength+len(name))
|
|
copy(k, orgID)
|
|
copy(k[platform.IDLength:], name)
|
|
return k, nil
|
|
}
|
|
|
|
// uniqueBucketName ensures this bucket is unique for this organization
|
|
func (s *Store) uniqueBucketName(ctx context.Context, tx kv.Tx, oid platform.ID, uname string) error {
|
|
key, err := bucketIndexKey(oid, uname)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(key) == 0 {
|
|
return ErrNameisEmpty
|
|
}
|
|
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = idx.Get(key)
|
|
// if not found then this is _unique_.
|
|
if kv.IsNotFound(err) {
|
|
return nil
|
|
}
|
|
|
|
// no error means this is not unique
|
|
if err == nil {
|
|
return BucketAlreadyExistsError(uname)
|
|
}
|
|
|
|
// any other error is some sort of internal server error
|
|
return ErrInternalServiceError(err)
|
|
}
|
|
|
|
func unmarshalBucket(v []byte) (*influxdb.Bucket, error) {
|
|
u := &influxdb.Bucket{}
|
|
if err := json.Unmarshal(v, u); err != nil {
|
|
return nil, ErrCorruptBucket(err)
|
|
}
|
|
|
|
return u, nil
|
|
}
|
|
|
|
func marshalBucket(u *influxdb.Bucket) ([]byte, error) {
|
|
v, err := json.Marshal(u)
|
|
if err != nil {
|
|
return nil, ErrUnprocessableBucket(err)
|
|
}
|
|
|
|
return v, nil
|
|
}
|
|
|
|
func (s *Store) GetBucket(ctx context.Context, tx kv.Tx, id platform.ID) (*influxdb.Bucket, error) {
|
|
encodedID, err := id.Encode()
|
|
if err != nil {
|
|
return nil, InvalidOrgIDError(err)
|
|
}
|
|
|
|
b, err := tx.Bucket(bucketBucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
v, err := b.Get(encodedID)
|
|
if kv.IsNotFound(err) {
|
|
return nil, ErrBucketNotFound
|
|
}
|
|
|
|
if err != nil {
|
|
return nil, ErrInternalServiceError(err)
|
|
}
|
|
|
|
return unmarshalBucket(v)
|
|
}
|
|
|
|
func (s *Store) GetBucketByName(ctx context.Context, tx kv.Tx, orgID platform.ID, n string) (*influxdb.Bucket, error) {
|
|
key, err := bucketIndexKey(orgID, n)
|
|
if err != nil {
|
|
return nil, &errors.Error{
|
|
Code: errors.EInvalid,
|
|
Err: err,
|
|
}
|
|
}
|
|
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
buf, err := idx.Get(key)
|
|
|
|
// allow for hard coded bucket names that dont exist in the system
|
|
if kv.IsNotFound(err) {
|
|
return nil, ErrBucketNotFoundByName(n)
|
|
}
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var id platform.ID
|
|
if err := id.Decode(buf); err != nil {
|
|
return nil, &errors.Error{
|
|
Err: err,
|
|
}
|
|
}
|
|
return s.GetBucket(ctx, tx, id)
|
|
}
|
|
|
|
type BucketFilter struct {
|
|
Name *string
|
|
OrganizationID *platform.ID
|
|
}
|
|
|
|
func (s *Store) ListBuckets(ctx context.Context, tx kv.Tx, filter BucketFilter, opt ...influxdb.FindOptions) ([]*influxdb.Bucket, error) {
|
|
// this isn't a list action its a `GetBucketByName`
|
|
if (filter.OrganizationID != nil && filter.OrganizationID.Valid()) && filter.Name != nil {
|
|
return nil, invalidBucketListRequest
|
|
}
|
|
|
|
// if we dont have any options it would be irresponsible to just give back all orgs in the system
|
|
if len(opt) == 0 {
|
|
opt = append(opt, influxdb.FindOptions{
|
|
Limit: influxdb.DefaultPageSize,
|
|
})
|
|
}
|
|
o := opt[0]
|
|
if o.Limit > influxdb.MaxPageSize || o.Limit == 0 {
|
|
o.Limit = influxdb.MaxPageSize
|
|
}
|
|
|
|
// if an organization is passed we need to use the index
|
|
if filter.OrganizationID != nil {
|
|
return s.listBucketsByOrg(ctx, tx, *filter.OrganizationID, o)
|
|
}
|
|
|
|
b, err := tx.Bucket(bucketBucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var opts []kv.CursorOption
|
|
if o.Descending {
|
|
opts = append(opts, kv.WithCursorDirection(kv.CursorDescending))
|
|
}
|
|
|
|
var seek []byte
|
|
if o.After != nil {
|
|
after := (*o.After) + 1
|
|
seek, err = after.Encode()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
cursor, err := b.ForwardCursor(seek, opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer cursor.Close()
|
|
|
|
count := 0
|
|
bs := []*influxdb.Bucket{}
|
|
for k, v := cursor.Next(); k != nil; k, v = cursor.Next() {
|
|
if o.Offset != 0 && count < o.Offset {
|
|
count++
|
|
continue
|
|
}
|
|
b, err := unmarshalBucket(v)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// check to see if it matches the filter
|
|
if filter.Name == nil || (*filter.Name == b.Name) {
|
|
bs = append(bs, b)
|
|
}
|
|
|
|
if len(bs) >= o.Limit {
|
|
break
|
|
}
|
|
}
|
|
|
|
return bs, cursor.Err()
|
|
}
|
|
|
|
func (s *Store) listBucketsByOrg(ctx context.Context, tx kv.Tx, orgID platform.ID, o influxdb.FindOptions) ([]*influxdb.Bucket, error) {
|
|
// get the prefix key (org id with an empty name)
|
|
key, err := bucketIndexKey(orgID, "")
|
|
if err != nil {
|
|
return nil, &errors.Error{
|
|
Code: errors.EInvalid,
|
|
Err: err,
|
|
}
|
|
}
|
|
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
start := key
|
|
opts := []kv.CursorOption{kv.WithCursorPrefix(key)}
|
|
if o.Descending {
|
|
// To list in descending order, we have to find the last entry prefixed
|
|
// by the org ID. AFAICT the only way to do this given our current indexing
|
|
// scheme is to iterate through all entries in the org once, remembering the
|
|
// last-seen key.
|
|
start, err = func() ([]byte, error) {
|
|
cursor, err := idx.ForwardCursor(start, opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer cursor.Close()
|
|
|
|
lastKey := start
|
|
for k, _ := cursor.Next(); k != nil; k, _ = cursor.Next() {
|
|
lastKey = k
|
|
}
|
|
return lastKey, nil
|
|
}()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// Once we've found the end, walk backwards from it on the next iteration.
|
|
opts = append(opts, kv.WithCursorDirection(kv.CursorDescending))
|
|
}
|
|
cursor, err := idx.ForwardCursor(start, opts...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer cursor.Close()
|
|
|
|
count := 0
|
|
bs := []*influxdb.Bucket{}
|
|
searchingForAfter := o.After != nil
|
|
for k, v := cursor.Next(); k != nil; k, v = cursor.Next() {
|
|
if o.Offset != 0 && count < o.Offset {
|
|
count++
|
|
continue
|
|
}
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var id platform.ID
|
|
if err := id.Decode(v); err != nil {
|
|
return nil, &errors.Error{
|
|
Err: err,
|
|
}
|
|
}
|
|
if searchingForAfter {
|
|
searchingForAfter = id != *o.After
|
|
continue
|
|
}
|
|
b, err := s.GetBucket(ctx, tx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bs = append(bs, b)
|
|
|
|
if len(bs) >= o.Limit {
|
|
break
|
|
}
|
|
}
|
|
|
|
return bs, cursor.Err()
|
|
}
|
|
|
|
func (s *Store) CreateBucket(ctx context.Context, tx kv.Tx, bucket *influxdb.Bucket) (err error) {
|
|
// generate new bucket ID
|
|
bucket.ID, err = s.generateSafeID(ctx, tx, bucketBucket, s.BucketIDGen)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
encodedID, err := bucket.ID.Encode()
|
|
if err != nil {
|
|
return InvalidOrgIDError(err)
|
|
}
|
|
|
|
if err := s.uniqueBucketName(ctx, tx, bucket.OrgID, bucket.Name); err != nil {
|
|
return err
|
|
}
|
|
|
|
bucket.SetCreatedAt(s.now())
|
|
bucket.SetUpdatedAt(s.now())
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
b, err := tx.Bucket(bucketBucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
v, err := marshalBucket(bucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ikey, err := bucketIndexKey(bucket.OrgID, bucket.Name)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := idx.Put(ikey, encodedID); err != nil {
|
|
return ErrInternalServiceError(err)
|
|
}
|
|
|
|
if err := b.Put(encodedID, v); err != nil {
|
|
return ErrInternalServiceError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *Store) UpdateBucket(ctx context.Context, tx kv.Tx, id platform.ID, upd influxdb.BucketUpdate) (*influxdb.Bucket, error) {
|
|
encodedID, err := id.Encode()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bucket, err := s.GetBucket(ctx, tx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
bucket.SetUpdatedAt(s.now())
|
|
if upd.Name != nil && bucket.Name != *upd.Name {
|
|
// validation
|
|
if bucket.Type == influxdb.BucketTypeSystem {
|
|
return nil, errRenameSystemBucket
|
|
}
|
|
|
|
if err := validBucketName(*upd.Name, bucket.Type); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := s.uniqueBucketName(ctx, tx, bucket.OrgID, *upd.Name); err != nil {
|
|
return nil, ErrBucketNameNotUnique
|
|
}
|
|
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
oldIkey, err := bucketIndexKey(bucket.OrgID, bucket.Name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := idx.Delete(oldIkey); err != nil {
|
|
return nil, ErrInternalServiceError(err)
|
|
}
|
|
|
|
bucket.Name = *upd.Name
|
|
newIkey, err := bucketIndexKey(bucket.OrgID, bucket.Name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := idx.Put(newIkey, encodedID); err != nil {
|
|
return nil, ErrInternalServiceError(err)
|
|
}
|
|
}
|
|
|
|
if upd.Description != nil {
|
|
bucket.Description = *upd.Description
|
|
}
|
|
|
|
if upd.RetentionPeriod != nil {
|
|
bucket.RetentionPeriod = *upd.RetentionPeriod
|
|
}
|
|
if upd.ShardGroupDuration != nil {
|
|
bucket.ShardGroupDuration = *upd.ShardGroupDuration
|
|
}
|
|
|
|
v, err := marshalBucket(bucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
b, err := tx.Bucket(bucketBucket)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err := b.Put(encodedID, v); err != nil {
|
|
return nil, ErrInternalServiceError(err)
|
|
}
|
|
|
|
return bucket, nil
|
|
}
|
|
|
|
func (s *Store) DeleteBucket(ctx context.Context, tx kv.Tx, id platform.ID) error {
|
|
bucket, err := s.GetBucket(ctx, tx, id)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
encodedID, err := id.Encode()
|
|
if err != nil {
|
|
return InvalidOrgIDError(err)
|
|
}
|
|
|
|
idx, err := tx.Bucket(bucketIndex)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ikey, err := bucketIndexKey(bucket.OrgID, bucket.Name)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := idx.Delete(ikey); err != nil {
|
|
return ErrInternalServiceError(err)
|
|
}
|
|
|
|
b, err := tx.Bucket(bucketBucket)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := b.Delete(encodedID); err != nil {
|
|
return ErrInternalServiceError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|