2017-02-07 22:33:23 +01:00
|
|
|
package store
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"errors"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Backend represents a KV Store Backend
|
|
|
|
type Backend string
|
|
|
|
|
|
|
|
const (
|
|
|
|
// CONSUL backend
|
|
|
|
CONSUL Backend = "consul"
|
2017-11-17 17:22:03 +01:00
|
|
|
// ETCD backend with v2 client (backward compatibility)
|
2017-02-07 22:33:23 +01:00
|
|
|
ETCD Backend = "etcd"
|
2017-11-17 17:22:03 +01:00
|
|
|
// ETCDV3 backend with v3 client
|
|
|
|
ETCDV3 Backend = "etcdv3"
|
2017-02-07 22:33:23 +01:00
|
|
|
// ZK backend
|
|
|
|
ZK Backend = "zk"
|
|
|
|
// BOLTDB backend
|
|
|
|
BOLTDB Backend = "boltdb"
|
2017-11-17 17:22:03 +01:00
|
|
|
// REDIS backend
|
|
|
|
REDIS Backend = "redis"
|
2017-02-07 22:33:23 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrBackendNotSupported is thrown when the backend k/v store is not supported by libkv
|
|
|
|
ErrBackendNotSupported = errors.New("Backend storage not supported yet, please choose one of")
|
|
|
|
// ErrCallNotSupported is thrown when a method is not implemented/supported by the current backend
|
|
|
|
ErrCallNotSupported = errors.New("The current call is not supported with this backend")
|
|
|
|
// ErrNotReachable is thrown when the API cannot be reached for issuing common store operations
|
|
|
|
ErrNotReachable = errors.New("Api not reachable")
|
|
|
|
// ErrCannotLock is thrown when there is an error acquiring a lock on a key
|
|
|
|
ErrCannotLock = errors.New("Error acquiring the lock")
|
|
|
|
// ErrKeyModified is thrown during an atomic operation if the index does not match the one in the store
|
|
|
|
ErrKeyModified = errors.New("Unable to complete atomic operation, key modified")
|
|
|
|
// ErrKeyNotFound is thrown when the key is not found in the store during a Get operation
|
|
|
|
ErrKeyNotFound = errors.New("Key not found in store")
|
|
|
|
// ErrPreviousNotSpecified is thrown when the previous value is not specified for an atomic operation
|
|
|
|
ErrPreviousNotSpecified = errors.New("Previous K/V pair should be provided for the Atomic operation")
|
|
|
|
// ErrKeyExists is thrown when the previous value exists in the case of an AtomicPut
|
|
|
|
ErrKeyExists = errors.New("Previous K/V pair exists, cannot complete Atomic operation")
|
|
|
|
)
|
|
|
|
|
|
|
|
// Config contains the options for a storage client
|
|
|
|
type Config struct {
|
|
|
|
ClientTLS *ClientTLSConfig
|
|
|
|
TLS *tls.Config
|
|
|
|
ConnectionTimeout time.Duration
|
2017-11-17 17:22:03 +01:00
|
|
|
SyncPeriod time.Duration
|
2017-02-07 22:33:23 +01:00
|
|
|
Bucket string
|
|
|
|
PersistConnection bool
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
}
|
|
|
|
|
|
|
|
// ClientTLSConfig contains data for a Client TLS configuration in the form
|
|
|
|
// the etcd client wants it. Eventually we'll adapt it for ZK and Consul.
|
|
|
|
type ClientTLSConfig struct {
|
|
|
|
CertFile string
|
|
|
|
KeyFile string
|
|
|
|
CACertFile string
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store represents the backend K/V storage
|
|
|
|
// Each store should support every call listed
|
|
|
|
// here. Or it couldn't be implemented as a K/V
|
|
|
|
// backend for libkv
|
|
|
|
type Store interface {
|
|
|
|
// Put a value at the specified key
|
|
|
|
Put(key string, value []byte, options *WriteOptions) error
|
|
|
|
|
|
|
|
// Get a value given its key
|
2017-11-17 17:22:03 +01:00
|
|
|
Get(key string, options *ReadOptions) (*KVPair, error)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
// Delete the value at the specified key
|
|
|
|
Delete(key string) error
|
|
|
|
|
|
|
|
// Verify if a Key exists in the store
|
2017-11-17 17:22:03 +01:00
|
|
|
Exists(key string, options *ReadOptions) (bool, error)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
// Watch for changes on a key
|
2017-11-17 17:22:03 +01:00
|
|
|
Watch(key string, stopCh <-chan struct{}, options *ReadOptions) (<-chan *KVPair, error)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
// WatchTree watches for changes on child nodes under
|
|
|
|
// a given directory
|
2017-11-17 17:22:03 +01:00
|
|
|
WatchTree(directory string, stopCh <-chan struct{}, options *ReadOptions) (<-chan []*KVPair, error)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
// NewLock creates a lock for a given key.
|
|
|
|
// The returned Locker is not held and must be acquired
|
|
|
|
// with `.Lock`. The Value is optional.
|
|
|
|
NewLock(key string, options *LockOptions) (Locker, error)
|
|
|
|
|
|
|
|
// List the content of a given prefix
|
2017-11-17 17:22:03 +01:00
|
|
|
List(directory string, options *ReadOptions) ([]*KVPair, error)
|
2017-02-07 22:33:23 +01:00
|
|
|
|
|
|
|
// DeleteTree deletes a range of keys under a given directory
|
|
|
|
DeleteTree(directory string) error
|
|
|
|
|
|
|
|
// Atomic CAS operation on a single value.
|
|
|
|
// Pass previous = nil to create a new key.
|
|
|
|
AtomicPut(key string, value []byte, previous *KVPair, options *WriteOptions) (bool, *KVPair, error)
|
|
|
|
|
|
|
|
// Atomic delete of a single value
|
|
|
|
AtomicDelete(key string, previous *KVPair) (bool, error)
|
|
|
|
|
|
|
|
// Close the store connection
|
|
|
|
Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// KVPair represents {Key, Value, Lastindex} tuple
|
|
|
|
type KVPair struct {
|
|
|
|
Key string
|
|
|
|
Value []byte
|
|
|
|
LastIndex uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteOptions contains optional request parameters
|
|
|
|
type WriteOptions struct {
|
|
|
|
IsDir bool
|
|
|
|
TTL time.Duration
|
|
|
|
}
|
|
|
|
|
2017-11-17 17:22:03 +01:00
|
|
|
// ReadOptions contains optional request parameters
|
|
|
|
type ReadOptions struct {
|
|
|
|
// Consistent defines if the behavior of a Get operation is
|
|
|
|
// linearizable or not. Linearizability allows us to 'see'
|
|
|
|
// objects based on a real-time total order as opposed to
|
|
|
|
// an arbitrary order or with stale values ('inconsistent'
|
|
|
|
// scenario).
|
|
|
|
Consistent bool
|
|
|
|
}
|
|
|
|
|
2017-02-07 22:33:23 +01:00
|
|
|
// LockOptions contains optional request parameters
|
|
|
|
type LockOptions struct {
|
|
|
|
Value []byte // Optional, value to associate with the lock
|
|
|
|
TTL time.Duration // Optional, expiration ttl associated with the lock
|
|
|
|
RenewLock chan struct{} // Optional, chan used to control and stop the session ttl renewal for the lock
|
|
|
|
}
|
|
|
|
|
|
|
|
// Locker provides locking mechanism on top of the store.
|
|
|
|
// Similar to `sync.Lock` except it may return errors.
|
|
|
|
type Locker interface {
|
|
|
|
Lock(stopChan chan struct{}) (<-chan struct{}, error)
|
|
|
|
Unlock() error
|
|
|
|
}
|