2018-09-07 15:45:28 +00:00
|
|
|
package bolt
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
|
|
|
|
bolt "github.com/coreos/bbolt"
|
2019-04-12 16:45:48 +00:00
|
|
|
influxdb "github.com/influxdata/influxdb"
|
2018-09-07 15:45:28 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
scraperBucket = []byte("scraperv2")
|
|
|
|
)
|
|
|
|
|
2019-04-12 16:45:48 +00:00
|
|
|
var _ influxdb.ScraperTargetStoreService = (*Client)(nil)
|
2018-09-07 15:45:28 +00:00
|
|
|
|
|
|
|
func (c *Client) initializeScraperTargets(ctx context.Context, tx *bolt.Tx) error {
|
|
|
|
if _, err := tx.CreateBucketIfNotExists([]byte(scraperBucket)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListTargets will list all scrape targets.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) ListTargets(ctx context.Context, filter influxdb.ScraperTargetFilter) (list []influxdb.ScraperTarget, err error) {
|
|
|
|
list = make([]influxdb.ScraperTarget, 0)
|
2018-09-07 15:45:28 +00:00
|
|
|
err = c.db.View(func(tx *bolt.Tx) (err error) {
|
|
|
|
cur := tx.Bucket(scraperBucket).Cursor()
|
|
|
|
for k, v := cur.First(); k != nil; k, v = cur.Next() {
|
2019-04-12 16:45:48 +00:00
|
|
|
target := new(influxdb.ScraperTarget)
|
2018-09-07 15:45:28 +00:00
|
|
|
if err = json.Unmarshal(v, target); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-04-12 16:45:48 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if filter.IDs != nil {
|
|
|
|
if _, ok := filter.IDs[target.ID]; !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if filter.Name != nil && target.Name != *filter.Name {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if filter.Org != nil {
|
|
|
|
o, err := c.findOrganizationByName(ctx, tx, *filter.Org)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if target.OrgID != o.ID {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if filter.OrgID != nil {
|
|
|
|
o, err := c.findOrganizationByID(ctx, tx, *filter.OrgID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if target.OrgID != o.ID {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
list = append(list, *target)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
})
|
2018-12-17 14:07:38 +00:00
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Op: getOp(influxdb.OpListTargets),
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
return list, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddTarget add a new scraper target into storage.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) AddTarget(ctx context.Context, target *influxdb.ScraperTarget, userID influxdb.ID) (err error) {
|
2019-01-10 17:39:37 +00:00
|
|
|
if !target.OrgID.Valid() {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided organization ID has invalid format",
|
2019-04-12 16:45:48 +00:00
|
|
|
Op: OpPrefix + influxdb.OpAddTarget,
|
2019-01-10 17:39:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if !target.BucketID.Valid() {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided bucket ID has invalid format",
|
2019-04-12 16:45:48 +00:00
|
|
|
Op: OpPrefix + influxdb.OpAddTarget,
|
2019-01-10 17:39:37 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-17 14:07:38 +00:00
|
|
|
err = c.db.Update(func(tx *bolt.Tx) error {
|
2018-09-07 15:45:28 +00:00
|
|
|
target.ID = c.IDGenerator.ID()
|
2019-01-18 20:46:37 +00:00
|
|
|
if err := c.putTarget(ctx, tx, target); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-04-12 16:45:48 +00:00
|
|
|
urm := &influxdb.UserResourceMapping{
|
2019-01-18 20:46:37 +00:00
|
|
|
ResourceID: target.ID,
|
|
|
|
UserID: userID,
|
2019-04-12 16:45:48 +00:00
|
|
|
UserType: influxdb.Owner,
|
|
|
|
ResourceType: influxdb.ScraperResourceType,
|
2019-01-18 20:46:37 +00:00
|
|
|
}
|
|
|
|
return c.createUserResourceMapping(ctx, tx, urm)
|
2018-09-07 15:45:28 +00:00
|
|
|
})
|
2018-12-17 14:07:38 +00:00
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
2019-04-12 16:45:48 +00:00
|
|
|
Op: OpPrefix + influxdb.OpAddTarget,
|
2018-12-17 14:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RemoveTarget removes a scraper target from the bucket.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) RemoveTarget(ctx context.Context, id influxdb.ID) error {
|
2018-12-17 14:07:38 +00:00
|
|
|
err := c.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
_, pe := c.findTargetByID(ctx, tx, id)
|
|
|
|
if pe != nil {
|
|
|
|
return pe
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
2018-09-24 13:39:28 +00:00
|
|
|
encID, err := id.Encode()
|
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
|
|
|
}
|
2018-09-24 13:39:28 +00:00
|
|
|
}
|
2019-01-18 20:46:37 +00:00
|
|
|
if err = tx.Bucket(scraperBucket).Delete(encID); err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
2019-04-12 16:45:48 +00:00
|
|
|
return c.deleteUserResourceMappings(ctx, tx, influxdb.UserResourceMappingFilter{
|
2019-01-18 20:46:37 +00:00
|
|
|
ResourceID: id,
|
2019-04-12 16:45:48 +00:00
|
|
|
ResourceType: influxdb.ScraperResourceType,
|
2019-01-18 20:46:37 +00:00
|
|
|
})
|
2018-09-07 15:45:28 +00:00
|
|
|
})
|
2018-12-17 14:07:38 +00:00
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
2019-04-12 16:45:48 +00:00
|
|
|
Op: OpPrefix + influxdb.OpRemoveTarget,
|
2018-12-17 14:07:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateTarget updates a scraper target.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) UpdateTarget(ctx context.Context, update *influxdb.ScraperTarget, userID influxdb.ID) (target *influxdb.ScraperTarget, err error) {
|
|
|
|
op := getOp(influxdb.OpUpdateTarget)
|
|
|
|
var pe *influxdb.Error
|
2018-09-24 13:39:28 +00:00
|
|
|
if !update.ID.Valid() {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.EInvalid,
|
2018-12-17 14:07:38 +00:00
|
|
|
Op: op,
|
2019-02-19 23:47:19 +00:00
|
|
|
Msg: "provided scraper target ID has invalid format",
|
2018-12-17 14:07:38 +00:00
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
err = c.db.Update(func(tx *bolt.Tx) error {
|
2018-12-17 14:07:38 +00:00
|
|
|
target, pe = c.findTargetByID(ctx, tx, update.ID)
|
|
|
|
if pe != nil {
|
|
|
|
return pe
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
2019-01-10 17:39:37 +00:00
|
|
|
if !update.BucketID.Valid() {
|
|
|
|
update.BucketID = target.BucketID
|
|
|
|
}
|
|
|
|
if !update.OrgID.Valid() {
|
|
|
|
update.OrgID = target.OrgID
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
target = update
|
|
|
|
return c.putTarget(ctx, tx, target)
|
|
|
|
})
|
|
|
|
|
2018-12-17 14:07:38 +00:00
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
2018-12-17 14:07:38 +00:00
|
|
|
Op: op,
|
|
|
|
Err: err,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return target, nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetTargetByID retrieves a scraper target by id.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) GetTargetByID(ctx context.Context, id influxdb.ID) (target *influxdb.ScraperTarget, err error) {
|
|
|
|
var pe *influxdb.Error
|
2018-09-07 15:45:28 +00:00
|
|
|
err = c.db.View(func(tx *bolt.Tx) error {
|
2018-12-17 14:07:38 +00:00
|
|
|
target, pe = c.findTargetByID(ctx, tx, id)
|
|
|
|
if pe != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return &influxdb.Error{
|
|
|
|
Op: getOp(influxdb.OpGetTargetByID),
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: pe,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2018-09-07 15:45:28 +00:00
|
|
|
})
|
2018-12-17 14:07:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return target, nil
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) findTargetByID(ctx context.Context, tx *bolt.Tx, id influxdb.ID) (target *influxdb.ScraperTarget, pe *influxdb.Error) {
|
|
|
|
target = new(influxdb.ScraperTarget)
|
2018-09-24 13:39:28 +00:00
|
|
|
encID, err := id.Encode()
|
|
|
|
if err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
|
|
|
}
|
2018-09-24 13:39:28 +00:00
|
|
|
}
|
|
|
|
v := tx.Bucket(scraperBucket).Get(encID)
|
2018-09-07 15:45:28 +00:00
|
|
|
if len(v) == 0 {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
|
|
|
Code: influxdb.ENotFound,
|
2018-12-17 14:07:38 +00:00
|
|
|
Msg: "scraper target is not found",
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if err := json.Unmarshal(v, target); err != nil {
|
2019-04-12 16:45:48 +00:00
|
|
|
return nil, &influxdb.Error{
|
2018-12-17 14:07:38 +00:00
|
|
|
Err: err,
|
|
|
|
}
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
return target, nil
|
|
|
|
}
|
|
|
|
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) putTarget(ctx context.Context, tx *bolt.Tx, target *influxdb.ScraperTarget) (err error) {
|
2018-09-07 15:45:28 +00:00
|
|
|
v, err := json.Marshal(target)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-09-24 13:39:28 +00:00
|
|
|
encID, err := target.ID.Encode()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return tx.Bucket(scraperBucket).Put(encID, v)
|
2018-09-07 15:45:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// PutTarget will put a scraper target without setting an ID.
|
2019-04-12 16:45:48 +00:00
|
|
|
func (c *Client) PutTarget(ctx context.Context, target *influxdb.ScraperTarget) error {
|
2018-09-07 15:45:28 +00:00
|
|
|
return c.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return c.putTarget(ctx, tx, target)
|
|
|
|
})
|
|
|
|
}
|