influxdb/kv/kvlog.go

398 lines
8.6 KiB
Go
Raw Normal View History

feat(kv): implemented key/value store with end-to-end integration tests * feat(kv:inmem:bolt): implement user service in a kv * refactor(kv): use consistent func receiver name * feat(kv): add initial basic auth service * refactor(passwords): move auth interface into own file * refactor(passwords): rename basic auth files to passwords * refactor(passwords): rename from BasicAuth to Passwords * refactor(kv): copy bolt user test into kv Co-authored-by: Michael Desa <mjdesa@gmail.com> * feat(kv): add inmem testing to kv store * fix(kv): remove extra user index initialization * feat(kv): attempt at making errors nice * fix(http): return not found error if filter is invalid * fix(http): s/platform/influxdb/ for user service * fix(http): s/platform/influxdb/ for user service * feat(kv): initial port of telegraf configs to kv * feat(kv): first pass at migrating bolt org service to kv * feat(kv): first pass at bucket service * feat(kv): first pass at migrating kvlog to kv package * feat(kv): add resource op logs * feat(kv): first pass at user resource mapping migration * feat(kv): add urm usage to bucket and org services * feat(kv): first pass at kv authz service * feat(kv): add cascading auth delete for users * feat(kv): first pass d authorizer.OrganizationService in kv * feat(cmd/influxd/launcher): user kv services where appropriate * fix(kv): initialize authorizations * fix(influxdb): use same buckets while slowly migrating stuff * fix(kv): make staticcheck pass * feat(kv): add dashboards to kv review: make suggestions from pr review fix: use common bucket names for bolt/kv stores * test(kv): add complete password test coverage * chore(kv): fixes for staticcheck * feat(kv): implement labels generically on kv * feat(kv): implement macro service * feat(kv): add source service * feat(kv): add session service * feat(kv): add kv secret service * refactor(kv): update telegraf and urm with error messages * feat(kv): add lookup service * feat(kv): add kv onboarding service * refactor(kv): update telegraf to avoid repetition * feat(cmd/influxd): use kv lookup service * feat(kv): add telegraf to lookup service * feat(cmd/influxd): use kv telegraf service * feat(kv): initial port of scrapers in bolt to kv * feat(kv): update scraper error messaging * feat(cmd/influxd): add kv scraper * feat(kv): add inmem backend tests * refactor(kv): copy paste errors * refactor(kv): add code to password errors * fix(testing): update error messages for incorrect passwords * feat(kv:inmem:bolt): implement user service in a kv * refactor(kv): use consistent func receiver name * refactor(kv): copy bolt user test into kv Co-authored-by: Michael Desa <mjdesa@gmail.com> * feat(kv): add inmem testing to kv store * fix(kv): remove extra user index initialization * feat(kv): attempt at making errors nice * fix(http): return not found error if filter is invalid * fix(http): s/platform/influxdb/ for user service * feat(kv): first pass at migrating bolt org service to kv * feat(kv): first pass at bucket service * feat(kv): first pass at migrating kvlog to kv package * feat(kv): add resource op logs * feat(kv): first pass at user resource mapping migration * feat(kv): add urm usage to bucket and org services * feat(kv): first pass at kv authz service * feat(kv): add cascading auth delete for users * feat(kv): first pass d authorizer.OrganizationService in kv * feat(cmd/influxd/launcher): user kv services where appropriate * feat(kv): add initial basic auth service * refactor(passwords): move auth interface into own file * refactor(passwords): rename basic auth files to passwords * fix(http): s/platform/influxdb/ for user service * fix(kv): initialize authorizations * fix(influxdb): use same buckets while slowly migrating stuff * fix(kv): make staticcheck pass * feat(kv): add dashboards to kv review: make suggestions from pr review fix: use common bucket names for bolt/kv stores * feat(kv): implement labels generically on kv * refactor(passwords): rename from BasicAuth to Passwords * test(kv): add complete password test coverage * chore(kv): fixes for staticcheck * feat(kv): implement macro service * feat(kv): add source service * feat(kv): add session service * feat(kv): initial port of telegraf configs to kv * feat(kv): initial port of scrapers in bolt to kv * feat(kv): add kv secret service * refactor(kv): update telegraf and urm with error messages * feat(kv): add lookup service * feat(kv): add kv onboarding service * refactor(kv): update telegraf to avoid repetition * feat(cmd/influxd): use kv lookup service * feat(kv): add telegraf to lookup service * feat(cmd/influxd): use kv telegraf service * feat(kv): update scraper error messaging * feat(cmd/influxd): add kv scraper * feat(kv): add inmem backend tests * refactor(kv): copy paste errors * refactor(kv): add code to password errors * fix(testing): update error messages for incorrect passwords * feat(http): initial support for flushing all key/values from kv store * feat(kv): rename macro to variable * feat(cmd/influxd/launcher): user kv services where appropriate * refactor(passwords): rename from BasicAuth to Passwords * feat(kv): implement macro service * test(ui): introduce cypress * test(ui): introduce first typescript test * test(ui/e2e): add ci job * chore: update gitignore to ignore test outputs * feat(inmem): in memory influxdb * test(e2e): adding pinger that checks if influxdb is alive * hackathon * hack * hack * hack * hack * Revert "feat(inmem): in memory influxdb" This reverts commit 30ddf032003e704643b07ce80df61c3299ea7295. * hack * hack * hack * hack * hack * hack * hack * hack * hack * hack * hack * hack * hack * chore: lint ignore node_modules * hack * hack * hack * add user and flush * hack * remove unused vars * hack * hack * ci(circle): prefix e2e artifacts * change test to testid * update cypress * moar testid * fix npm warnings * remove absolte path * chore(ci): remove /home/circleci proto mkdir hack * wip: crud resources e2e * fix(inmem): use inmem kv store services * test(dashboard): add first dashboard crud tests * hack * undo hack * fix: use response from setup for orgID * chore: wip * add convenience getByTitle function * test(e2e): ui can create orgs * test(e2e): add test for org deletion and update * test(e2e): introduce task creation test * test(e2e): create and update of buckets on org view * chore: move types to declaration file * chore: use route fixture in dashboard tests * chore(ci): hack back * test(ui): update snapshots * chore: package-lock * chore: remove macros * fix: launcher rebase issues * fix: compile errors * fix: compile errors * feat(cmd/influxdb): add explicit testing, asset-path, and store flags Co-authored-by: Andrew Watkins <watts@influxdb.com> * fix(cmd/influxd): set default HTTP handler and flags Co-authored-by: Andrew Watkins <watts@influxdb.com> * build(Makefile): add run-e2e and PHONY * feat(kv:inmem:bolt): implement user service in a kv * refactor(kv): use consistent func receiver name * feat(kv): add initial basic auth service * refactor(passwords): move auth interface into own file * refactor(passwords): rename basic auth files to passwords * refactor(passwords): rename from BasicAuth to Passwords * refactor(kv): copy bolt user test into kv Co-authored-by: Michael Desa <mjdesa@gmail.com> * feat(kv): add inmem testing to kv store * fix(kv): remove extra user index initialization * feat(kv): attempt at making errors nice * fix(http): return not found error if filter is invalid * fix(http): s/platform/influxdb/ for user service * fix(http): s/platform/influxdb/ for user service * feat(kv): initial port of telegraf configs to kv * feat(kv): initial port of scrapers in bolt to kv * feat(kv): first pass at migrating bolt org service to kv * feat(kv): first pass at bucket service * feat(kv): first pass at migrating kvlog to kv package * feat(kv): add resource op logs * feat(kv): first pass at user resource mapping migration * feat(kv): add urm usage to bucket and org services * feat(kv): first pass at kv authz service * feat(kv): add cascading auth delete for users * feat(kv): first pass d authorizer.OrganizationService in kv * feat(cmd/influxd/launcher): user kv services where appropriate * fix(kv): initialize authorizations * fix(influxdb): use same buckets while slowly migrating stuff * fix(kv): make staticcheck pass * feat(kv): add dashboards to kv review: make suggestions from pr review fix: use common bucket names for bolt/kv stores * test(kv): add complete password test coverage * chore(kv): fixes for staticcheck * feat(kv): implement labels generically on kv * feat(kv): implement macro service * feat(kv): add source service * feat(kv): add session service * feat(kv): add kv secret service * refactor(kv): update telegraf and urm with error messages * feat(kv): add lookup service * feat(kv): add kv onboarding service * refactor(kv): update telegraf to avoid repetition * feat(cmd/influxd): use kv lookup service * feat(kv): add telegraf to lookup service * feat(cmd/influxd): use kv telegraf service * feat(kv): update scraper error messaging * feat(cmd/influxd): add kv scraper * feat(kv): add inmem backend tests * refactor(kv): copy paste errors * refactor(kv): add code to password errors * fix(testing): update error messages for incorrect passwords * feat(kv): rename macro to variable * refactor(kv): auth/bucket/org/user unique checks return errors now * feat(inmem): add way to get all bucket names from store * feat(inmem): Buckets to return slice of bytes rather than strings * feat(inmem): add locks around Buckets to avoid races * feat(cmd/influx): check for unauthorized error in wrapCheckSetup * chore(e2e): add video and screenshot artifcats to gitignore * docs(ci): add build instructions for e2e tests * feat(kv): add id lookup for authorized resources
2019-02-19 23:47:19 +00:00
package kv
import (
"bytes"
"context"
"crypto/sha1"
"encoding/binary"
"encoding/json"
"fmt"
"time"
platform "github.com/influxdata/influxdb"
)
var (
kvlogBucket = []byte("keyvaluelogv1")
kvlogIndex = []byte("keyvaluelogindexv1")
)
var _ platform.KeyValueLog = (*Service)(nil)
type keyValueLogBounds struct {
Start int64 `json:"start"`
Stop int64 `json:"stop"`
}
func newKeyValueLogBounds(now time.Time) *keyValueLogBounds {
return &keyValueLogBounds{
Start: now.UTC().UnixNano(),
Stop: now.UTC().UnixNano(),
}
}
func (b *keyValueLogBounds) update(t time.Time) {
now := t.UTC().UnixNano()
if now < b.Start {
b.Start = now
} else if b.Stop < now {
b.Stop = now
}
}
// StartTime retrieves the start value of a bounds as a time.Time
func (b *keyValueLogBounds) StartTime() time.Time {
return time.Unix(0, b.Start)
}
// StopTime retrieves the stop value of a bounds as a time.Time
func (b *keyValueLogBounds) StopTime() time.Time {
return time.Unix(0, b.Stop)
}
// Bounds returns the key boundaries for the keyvaluelog for a resourceType/resourceID pair.
func (b *keyValueLogBounds) Bounds(k []byte) ([]byte, []byte, error) {
start, err := encodeLogEntryKey(k, b.Start)
if err != nil {
return nil, nil, err
}
stop, err := encodeLogEntryKey(k, b.Stop)
if err != nil {
return nil, nil, err
}
return start, stop, nil
}
func encodeLogEntryKey(key []byte, v int64) ([]byte, error) {
prefix := encodeKeyValueIndexKey(key)
k := make([]byte, len(prefix)+8)
buf := bytes.NewBuffer(k)
_, err := buf.Write(prefix)
if err != nil {
return nil, err
}
// This needs to be big-endian so that the iteration order is preserved when scanning keys
if err := binary.Write(buf, binary.BigEndian, v); err != nil {
return nil, err
}
return buf.Bytes(), err
}
func decodeLogEntryKey(key []byte) ([]byte, time.Time, error) {
buf := bytes.NewReader(key[len(key)-8:])
var ts int64
// This needs to be big-endian so that the iteration order is preserved when scanning keys
err := binary.Read(buf, binary.BigEndian, &ts)
if err != nil {
return nil, time.Unix(0, 0), err
}
return key[:len(key)-8], time.Unix(0, ts), nil
}
func encodeKeyValueIndexKey(k []byte) []byte {
// keys produced must be fixed length to ensure that we can iterate through the keyspace without any error.
h := sha1.New()
h.Write([]byte(k))
return h.Sum(nil)
}
func (s *Service) initializeKVLog(ctx context.Context, tx Tx) error {
if _, err := tx.Bucket(kvlogBucket); err != nil {
return err
}
if _, err := tx.Bucket(kvlogIndex); err != nil {
return err
}
return nil
}
var errKeyValueLogBoundsNotFound = fmt.Errorf("oplog not found")
func (s *Service) getKeyValueLogBounds(ctx context.Context, tx Tx, key []byte) (*keyValueLogBounds, error) {
k := encodeKeyValueIndexKey(key)
b, err := tx.Bucket(kvlogIndex)
if err != nil {
return nil, err
}
v, err := b.Get(k)
if IsNotFound(err) {
return nil, errKeyValueLogBoundsNotFound
}
if err != nil {
return nil, err
}
bounds := &keyValueLogBounds{}
if err := json.Unmarshal(v, bounds); err != nil {
return nil, err
}
return bounds, nil
}
func (s *Service) putKeyValueLogBounds(ctx context.Context, tx Tx, key []byte, bounds *keyValueLogBounds) error {
k := encodeKeyValueIndexKey(key)
v, err := json.Marshal(bounds)
if err != nil {
return err
}
b, err := tx.Bucket(kvlogIndex)
if err != nil {
return err
}
if err := b.Put(k, v); err != nil {
return err
}
return nil
}
func (s *Service) updateKeyValueLogBounds(ctx context.Context, tx Tx, k []byte, t time.Time) error {
// retrieve the keyValue log boundaries
bounds, err := s.getKeyValueLogBounds(ctx, tx, k)
if err != nil && err != errKeyValueLogBoundsNotFound {
return err
}
if err == errKeyValueLogBoundsNotFound {
// if the bounds don't exist yet, create them
bounds = newKeyValueLogBounds(t)
}
// update the bounds to if needed
bounds.update(t)
if err := s.putKeyValueLogBounds(ctx, tx, k, bounds); err != nil {
return err
}
return nil
}
// ForEachLogEntry retrieves the keyValue log for a resource type ID combination. KeyValues may be returned in ascending and descending order.
func (s *Service) ForEachLogEntry(ctx context.Context, k []byte, opts platform.FindOptions, fn func([]byte, time.Time) error) error {
return s.kv.View(func(tx Tx) error {
return s.forEachLogEntry(ctx, tx, k, opts, fn)
})
}
func (s *Service) forEachLogEntry(ctx context.Context, tx Tx, k []byte, opts platform.FindOptions, fn func([]byte, time.Time) error) error {
b, err := s.getKeyValueLogBounds(ctx, tx, k)
if err != nil {
return err
}
bkt, err := tx.Bucket(kvlogBucket)
if err != nil {
return err
}
cur, err := bkt.Cursor()
if err != nil {
return err
}
next := cur.Next
startKey, stopKey, err := b.Bounds(k)
if err != nil {
return err
}
if opts.Descending {
next = cur.Prev
startKey, stopKey = stopKey, startKey
}
k, v := cur.Seek(startKey)
if !bytes.Equal(k, startKey) {
return fmt.Errorf("the first key not the key found in the log bounds. This should be impossible. Please report this error")
}
count := 0
if opts.Offset == 0 {
// Seek returns the kv at the position that was seeked to which should be the first element
// in the sequence of keyValues. If this condition is reached we need to start of iteration
// at 1 instead of 0.
_, ts, err := decodeLogEntryKey(k)
if err != nil {
return err
}
if err := fn(v, ts); err != nil {
return err
}
count++
if bytes.Equal(startKey, stopKey) {
// If the start and stop are the same, then there is only a single entry in the log
return nil
}
} else {
// Skip offset many items
for i := 0; i < opts.Offset-1; i++ {
k, _ := next()
if bytes.Equal(k, stopKey) {
return nil
}
}
}
for {
if count >= opts.Limit && opts.Limit != 0 {
break
}
k, v := next()
_, ts, err := decodeLogEntryKey(k)
if err != nil {
return err
}
if err := fn(v, ts); err != nil {
return err
}
if bytes.Equal(k, stopKey) {
// if we've reached the stop key, there are no keys log entries left
// in the keyspace.
break
}
count++
}
return nil
}
// AddLogEntry logs an keyValue for a particular resource type ID pairing.
func (s *Service) AddLogEntry(ctx context.Context, k, v []byte, t time.Time) error {
return s.kv.Update(func(tx Tx) error {
return s.addLogEntry(ctx, tx, k, v, t)
})
}
func (s *Service) addLogEntry(ctx context.Context, tx Tx, k, v []byte, t time.Time) error {
if err := s.updateKeyValueLogBounds(ctx, tx, k, t); err != nil {
return err
}
if err := s.putLogEntry(ctx, tx, k, v, t); err != nil {
return err
}
return nil
}
func (s *Service) putLogEntry(ctx context.Context, tx Tx, k, v []byte, t time.Time) error {
key, err := encodeLogEntryKey(k, t.UTC().UnixNano())
if err != nil {
return err
}
b, err := tx.Bucket(kvlogBucket)
if err != nil {
return err
}
if err := b.Put(key, v); err != nil {
return err
}
return nil
}
func (s *Service) getLogEntry(ctx context.Context, tx Tx, k []byte, t time.Time) ([]byte, time.Time, error) {
key, err := encodeLogEntryKey(k, t.UTC().UnixNano())
if err != nil {
return nil, t, err
}
b, err := tx.Bucket(kvlogBucket)
if err != nil {
return nil, t, err
}
v, err := b.Get(key)
if IsNotFound(err) {
return nil, t, fmt.Errorf("log entry not found")
}
if err != nil {
return nil, t, err
}
return v, t, nil
}
// FirstLogEntry retrieves the first log entry for a key value log.
func (s *Service) FirstLogEntry(ctx context.Context, k []byte) ([]byte, time.Time, error) {
var v []byte
var t time.Time
err := s.kv.View(func(tx Tx) error {
val, ts, err := s.firstLogEntry(ctx, tx, k)
if err != nil {
return err
}
v, t = val, ts
return nil
})
if err != nil {
return nil, t, err
}
return v, t, nil
}
// LastLogEntry retrieves the first log entry for a key value log.
func (s *Service) LastLogEntry(ctx context.Context, k []byte) ([]byte, time.Time, error) {
var v []byte
var t time.Time
err := s.kv.View(func(tx Tx) error {
val, ts, err := s.lastLogEntry(ctx, tx, k)
if err != nil {
return err
}
v, t = val, ts
return nil
})
if err != nil {
return nil, t, err
}
return v, t, nil
}
func (s *Service) firstLogEntry(ctx context.Context, tx Tx, k []byte) ([]byte, time.Time, error) {
bounds, err := s.getKeyValueLogBounds(ctx, tx, k)
if err != nil {
return nil, bounds.StartTime(), err
}
return s.getLogEntry(ctx, tx, k, bounds.StartTime())
}
func (s *Service) lastLogEntry(ctx context.Context, tx Tx, k []byte) ([]byte, time.Time, error) {
bounds, err := s.getKeyValueLogBounds(ctx, tx, k)
if err != nil {
return nil, bounds.StopTime(), err
}
return s.getLogEntry(ctx, tx, k, bounds.StopTime())
}