2018-05-15 21:19:44 +00:00
|
|
|
package bolt
|
|
|
|
|
|
|
|
import (
|
2018-08-01 18:54:32 +00:00
|
|
|
"bytes"
|
2018-05-15 21:19:44 +00:00
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/coreos/bbolt"
|
|
|
|
"github.com/influxdata/platform"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
bucketBucket = []byte("bucketsv1")
|
|
|
|
bucketIndex = []byte("bucketindexv1")
|
|
|
|
)
|
|
|
|
|
|
|
|
func (c *Client) initializeBuckets(ctx context.Context, tx *bolt.Tx) error {
|
|
|
|
if _, err := tx.CreateBucketIfNotExists([]byte(bucketBucket)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if _, err := tx.CreateBucketIfNotExists([]byte(bucketIndex)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-05-16 18:59:35 +00:00
|
|
|
func (c *Client) setOrganizationOnBucket(ctx context.Context, tx *bolt.Tx, b *platform.Bucket) error {
|
2018-07-30 14:29:52 +00:00
|
|
|
o, err := c.findOrganizationByID(ctx, tx, b.OrganizationID)
|
2018-05-16 18:59:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.Organization = o.Name
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
// FindBucketByID retrieves a bucket by id.
|
|
|
|
func (c *Client) FindBucketByID(ctx context.Context, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
var b *platform.Bucket
|
|
|
|
|
|
|
|
err := c.db.View(func(tx *bolt.Tx) error {
|
|
|
|
bkt, err := c.findBucketByID(ctx, tx, id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b = bkt
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) findBucketByID(ctx context.Context, tx *bolt.Tx, id platform.ID) (*platform.Bucket, error) {
|
|
|
|
var b platform.Bucket
|
|
|
|
|
2018-08-01 18:54:32 +00:00
|
|
|
v := tx.Bucket(bucketBucket).Get(id)
|
2018-05-15 21:19:44 +00:00
|
|
|
|
|
|
|
if len(v) == 0 {
|
|
|
|
// TODO: Make standard error
|
|
|
|
return nil, fmt.Errorf("bucket not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(v, &b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-16 18:59:35 +00:00
|
|
|
if err := c.setOrganizationOnBucket(ctx, tx, &b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
return &b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindBucketByName returns a bucket by name for a particular organization.
|
|
|
|
// TODO: have method for finding bucket using organization name and bucket name.
|
|
|
|
func (c *Client) FindBucketByName(ctx context.Context, orgID platform.ID, n string) (*platform.Bucket, error) {
|
|
|
|
var b *platform.Bucket
|
|
|
|
|
|
|
|
err := c.db.View(func(tx *bolt.Tx) error {
|
|
|
|
bkt, err := c.findBucketByName(ctx, tx, orgID, n)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b = bkt
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return b, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) findBucketByName(ctx context.Context, tx *bolt.Tx, orgID platform.ID, n string) (*platform.Bucket, error) {
|
|
|
|
b := &platform.Bucket{
|
2018-07-30 14:29:52 +00:00
|
|
|
OrganizationID: orgID,
|
2018-05-15 21:19:44 +00:00
|
|
|
Name: n,
|
|
|
|
}
|
2018-08-01 18:54:32 +00:00
|
|
|
id := tx.Bucket(bucketIndex).Get(bucketIndexKey(b))
|
|
|
|
return c.findBucketByID(ctx, tx, platform.ID(id))
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// FindBucket retrives a bucket using an arbitrary bucket filter.
|
|
|
|
// Filters using ID, or OrganizationID and bucket Name should be efficient.
|
|
|
|
// Other filters will do a linear scan across buckets until it finds a match.
|
|
|
|
func (c *Client) FindBucket(ctx context.Context, filter platform.BucketFilter) (*platform.Bucket, error) {
|
|
|
|
if filter.ID != nil {
|
|
|
|
return c.FindBucketByID(ctx, *filter.ID)
|
|
|
|
}
|
|
|
|
|
|
|
|
if filter.Name != nil && filter.OrganizationID != nil {
|
|
|
|
return c.FindBucketByName(ctx, *filter.OrganizationID, *filter.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
var b *platform.Bucket
|
|
|
|
err := c.db.View(func(tx *bolt.Tx) error {
|
2018-05-16 18:59:35 +00:00
|
|
|
if filter.Organization != nil {
|
|
|
|
o, err := c.findOrganizationByName(ctx, tx, *filter.Organization)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-07-30 14:29:52 +00:00
|
|
|
filter.OrganizationID = &o.ID
|
2018-05-16 18:59:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
filterFn := filterBucketsFn(filter)
|
|
|
|
return c.forEachBucket(ctx, tx, func(bkt *platform.Bucket) bool {
|
2018-05-15 21:19:44 +00:00
|
|
|
if filterFn(bkt) {
|
|
|
|
b = bkt
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if b == nil {
|
|
|
|
return nil, fmt.Errorf("bucket not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func filterBucketsFn(filter platform.BucketFilter) func(b *platform.Bucket) bool {
|
|
|
|
if filter.ID != nil {
|
|
|
|
return func(b *platform.Bucket) bool {
|
2018-08-01 18:54:32 +00:00
|
|
|
return bytes.Equal(b.ID, *filter.ID)
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if filter.Name != nil && filter.OrganizationID != nil {
|
|
|
|
return func(b *platform.Bucket) bool {
|
2018-08-01 18:54:32 +00:00
|
|
|
return bytes.Equal(b.OrganizationID, *filter.OrganizationID) && b.Name == *filter.Name
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if filter.Name != nil {
|
|
|
|
return func(b *platform.Bucket) bool {
|
|
|
|
return b.Name == *filter.Name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if filter.OrganizationID != nil {
|
|
|
|
return func(b *platform.Bucket) bool {
|
2018-08-01 18:54:32 +00:00
|
|
|
return bytes.Equal(b.OrganizationID, *filter.OrganizationID)
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-06 16:19:58 +00:00
|
|
|
if filter.Type != 0 {
|
|
|
|
return func(b *platform.Bucket) bool {
|
|
|
|
return b.Type == filter.Type
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
return func(b *platform.Bucket) bool { return true }
|
|
|
|
}
|
|
|
|
|
|
|
|
// FindBuckets retrives all buckets that match an arbitrary bucket filter.
|
|
|
|
// Filters using ID, or OrganizationID and bucket Name should be efficient.
|
|
|
|
// Other filters will do a linear scan across all buckets searching for a match.
|
|
|
|
func (c *Client) FindBuckets(ctx context.Context, filter platform.BucketFilter, opt ...platform.FindOptions) ([]*platform.Bucket, int, error) {
|
|
|
|
if filter.ID != nil {
|
|
|
|
b, err := c.FindBucketByID(ctx, *filter.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return []*platform.Bucket{b}, 1, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if filter.Name != nil && filter.OrganizationID != nil {
|
|
|
|
b, err := c.FindBucketByName(ctx, *filter.OrganizationID, *filter.Name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return []*platform.Bucket{b}, 1, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
bs := []*platform.Bucket{}
|
|
|
|
err := c.db.View(func(tx *bolt.Tx) error {
|
2018-05-16 18:59:35 +00:00
|
|
|
bkts, err := c.findBuckets(ctx, tx, filter)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
bs = bkts
|
|
|
|
return nil
|
2018-05-15 21:19:44 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return bs, len(bs), nil
|
|
|
|
}
|
|
|
|
|
2018-05-16 18:59:35 +00:00
|
|
|
func (c *Client) findBuckets(ctx context.Context, tx *bolt.Tx, filter platform.BucketFilter) ([]*platform.Bucket, error) {
|
|
|
|
bs := []*platform.Bucket{}
|
|
|
|
if filter.Organization != nil {
|
|
|
|
o, err := c.findOrganizationByName(ctx, tx, *filter.Organization)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-07-30 14:29:52 +00:00
|
|
|
filter.OrganizationID = &o.ID
|
2018-05-16 18:59:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
filterFn := filterBucketsFn(filter)
|
|
|
|
err := c.forEachBucket(ctx, tx, func(b *platform.Bucket) bool {
|
|
|
|
if filterFn(b) {
|
|
|
|
bs = append(bs, b)
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return bs, nil
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
// CreateBucket creates a platform bucket and sets b.ID.
|
|
|
|
func (c *Client) CreateBucket(ctx context.Context, b *platform.Bucket) error {
|
2018-09-06 16:19:58 +00:00
|
|
|
if b.Type == 0 {
|
|
|
|
b.Type = platform.BucketTypeUser
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
return c.db.Update(func(tx *bolt.Tx) error {
|
2018-08-01 18:54:32 +00:00
|
|
|
if len(b.OrganizationID) == 0 {
|
2018-05-16 18:59:35 +00:00
|
|
|
o, err := c.findOrganizationByName(ctx, tx, b.Organization)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.OrganizationID = o.ID
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
unique := c.uniqueBucketName(ctx, tx, b)
|
|
|
|
|
|
|
|
if !unique {
|
|
|
|
// TODO: make standard error
|
|
|
|
return fmt.Errorf("bucket with name %s already exists", b.Name)
|
|
|
|
}
|
|
|
|
|
2018-09-06 16:19:58 +00:00
|
|
|
if b.Type == platform.BucketTypeUser {
|
|
|
|
b.ID = c.IDGenerator.ID()
|
|
|
|
} else {
|
|
|
|
idstr := fmt.Sprintf("%s%d", b.OrganizationID, b.Type)
|
|
|
|
id, err := platform.IDFromString(idstr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b.ID = *id
|
|
|
|
}
|
2018-05-15 21:19:44 +00:00
|
|
|
|
|
|
|
return c.putBucket(ctx, tx, b)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// PutBucket will put a bucket without setting an ID.
|
|
|
|
func (c *Client) PutBucket(ctx context.Context, b *platform.Bucket) error {
|
|
|
|
return c.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return c.putBucket(ctx, tx, b)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) putBucket(ctx context.Context, tx *bolt.Tx, b *platform.Bucket) error {
|
2018-05-16 18:59:35 +00:00
|
|
|
b.Organization = ""
|
2018-05-15 21:19:44 +00:00
|
|
|
v, err := json.Marshal(b)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2018-08-01 18:54:32 +00:00
|
|
|
if err := tx.Bucket(bucketIndex).Put(bucketIndexKey(b), b.ID); err != nil {
|
2018-07-30 14:29:52 +00:00
|
|
|
return err
|
|
|
|
}
|
2018-08-01 18:54:32 +00:00
|
|
|
if err := tx.Bucket(bucketBucket).Put(b.ID, v); err != nil {
|
2018-05-16 18:59:35 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return c.setOrganizationOnBucket(ctx, tx, b)
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
|
2018-08-01 18:54:32 +00:00
|
|
|
func bucketIndexKey(b *platform.Bucket) []byte {
|
|
|
|
k := make([]byte, len(b.OrganizationID)+len(b.Name))
|
|
|
|
copy(k, b.OrganizationID)
|
|
|
|
copy(k[len(b.OrganizationID):], []byte(b.Name))
|
|
|
|
return k
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// forEachBucket will iterate through all buckets while fn returns true.
|
2018-05-16 18:59:35 +00:00
|
|
|
func (c *Client) forEachBucket(ctx context.Context, tx *bolt.Tx, fn func(*platform.Bucket) bool) error {
|
2018-05-15 21:19:44 +00:00
|
|
|
cur := tx.Bucket(bucketBucket).Cursor()
|
|
|
|
for k, v := cur.First(); k != nil; k, v = cur.Next() {
|
|
|
|
b := &platform.Bucket{}
|
|
|
|
if err := json.Unmarshal(v, b); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-05-16 18:59:35 +00:00
|
|
|
if err := c.setOrganizationOnBucket(ctx, tx, b); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-05-15 21:19:44 +00:00
|
|
|
if !fn(b) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) uniqueBucketName(ctx context.Context, tx *bolt.Tx, b *platform.Bucket) bool {
|
2018-08-01 18:54:32 +00:00
|
|
|
v := tx.Bucket(bucketIndex).Get(bucketIndexKey(b))
|
2018-05-15 21:19:44 +00:00
|
|
|
return len(v) == 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateBucket updates a bucket according the parameters set on upd.
|
|
|
|
func (c *Client) UpdateBucket(ctx context.Context, id platform.ID, upd platform.BucketUpdate) (*platform.Bucket, error) {
|
|
|
|
var b *platform.Bucket
|
|
|
|
err := c.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
bkt, err := c.updateBucket(ctx, tx, id, upd)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b = bkt
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return b, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) updateBucket(ctx context.Context, tx *bolt.Tx, id platform.ID, upd platform.BucketUpdate) (*platform.Bucket, error) {
|
|
|
|
b, err := c.findBucketByID(ctx, tx, id)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if upd.RetentionPeriod != nil {
|
|
|
|
b.RetentionPeriod = *upd.RetentionPeriod
|
|
|
|
}
|
|
|
|
|
|
|
|
if upd.Name != nil {
|
2018-08-01 18:54:32 +00:00
|
|
|
// Buckets are indexed by name and so the bucket index must be pruned when name
|
|
|
|
// is modified.
|
|
|
|
if err := tx.Bucket(bucketIndex).Delete(bucketIndexKey(b)); err != nil {
|
2018-05-15 21:19:44 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
b.Name = *upd.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.putBucket(ctx, tx, b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-16 18:59:35 +00:00
|
|
|
if err := c.setOrganizationOnBucket(ctx, tx, b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-05-15 21:19:44 +00:00
|
|
|
return b, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteBucket deletes a bucket and prunes it from the index.
|
|
|
|
func (c *Client) DeleteBucket(ctx context.Context, id platform.ID) error {
|
|
|
|
return c.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return c.deleteBucket(ctx, tx, id)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) deleteBucket(ctx context.Context, tx *bolt.Tx, id platform.ID) error {
|
|
|
|
b, err := c.findBucketByID(ctx, tx, id)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// make lowercase deleteBucket with tx
|
2018-08-01 18:54:32 +00:00
|
|
|
if err := tx.Bucket(bucketIndex).Delete(bucketIndexKey(b)); err != nil {
|
2018-05-15 21:19:44 +00:00
|
|
|
return err
|
|
|
|
}
|
2018-08-01 18:54:32 +00:00
|
|
|
return tx.Bucket(bucketBucket).Delete(id)
|
2018-05-15 21:19:44 +00:00
|
|
|
}
|