2015-05-08 04:47:52 +00:00
|
|
|
package meta
|
|
|
|
|
2015-05-08 17:51:50 +00:00
|
|
|
import (
|
2015-06-05 20:40:18 +00:00
|
|
|
"encoding/binary"
|
2015-05-20 20:57:04 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2015-06-01 17:20:57 +00:00
|
|
|
"io/ioutil"
|
2015-05-20 20:57:04 +00:00
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2015-06-01 17:20:57 +00:00
|
|
|
"strconv"
|
2015-06-06 03:58:54 +00:00
|
|
|
"strings"
|
2015-05-20 20:57:04 +00:00
|
|
|
"sync"
|
2015-05-08 17:51:50 +00:00
|
|
|
"time"
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
|
|
|
"github.com/hashicorp/raft"
|
|
|
|
"github.com/hashicorp/raft-boltdb"
|
2015-05-27 19:41:58 +00:00
|
|
|
"github.com/influxdb/influxdb/influxql"
|
2015-05-20 20:57:04 +00:00
|
|
|
"github.com/influxdb/influxdb/meta/internal"
|
2015-05-21 20:06:01 +00:00
|
|
|
"golang.org/x/crypto/bcrypt"
|
2015-05-08 17:51:50 +00:00
|
|
|
)
|
|
|
|
|
2015-06-05 22:08:07 +00:00
|
|
|
// tcp.Mux header bytes.
|
|
|
|
const (
|
|
|
|
MuxRaftHeader = 0
|
|
|
|
MuxExecHeader = 1
|
|
|
|
)
|
|
|
|
|
|
|
|
// ExecMagic is the first 4 bytes sent to a remote exec connection to verify
|
|
|
|
// that it is coming from a remote exec client connection.
|
|
|
|
const ExecMagic = "EXEC"
|
|
|
|
|
2015-06-06 16:37:43 +00:00
|
|
|
// Retention policy auto-create settings.
|
|
|
|
const (
|
2015-06-07 00:03:52 +00:00
|
|
|
AutoCreateRetentionPolicyName = "default"
|
|
|
|
AutoCreateRetentionPolicyReplicaN = 1
|
2015-06-09 02:47:37 +00:00
|
|
|
AutoCreateRetentionPolicyPeriod = 0
|
2015-06-06 16:37:43 +00:00
|
|
|
)
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
// Raft configuration.
|
|
|
|
const (
|
|
|
|
raftLogCacheSize = 512
|
|
|
|
raftSnapshotsRetained = 2
|
|
|
|
raftTransportMaxPool = 3
|
|
|
|
raftTransportTimeout = 10 * time.Second
|
|
|
|
)
|
|
|
|
|
|
|
|
// Store represents a raft-backed metastore.
|
|
|
|
type Store struct {
|
2015-05-27 19:41:58 +00:00
|
|
|
mu sync.RWMutex
|
|
|
|
path string
|
|
|
|
opened bool
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
id uint64 // local node id
|
2015-06-01 17:20:57 +00:00
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// All peers in cluster. Used during bootstrapping.
|
|
|
|
peers []string
|
|
|
|
|
|
|
|
data *Data
|
|
|
|
|
|
|
|
remoteAddr net.Addr
|
|
|
|
raft *raft.Raft
|
|
|
|
raftLayer *raftLayer
|
|
|
|
peerStore raft.PeerStore
|
|
|
|
transport *raft.NetworkTransport
|
|
|
|
store *raftboltdb.BoltStore
|
|
|
|
|
|
|
|
ready chan struct{}
|
|
|
|
err chan error
|
|
|
|
closing chan struct{}
|
|
|
|
wg sync.WaitGroup
|
|
|
|
|
2015-06-07 00:03:52 +00:00
|
|
|
retentionAutoCreate bool
|
2015-06-06 16:24:42 +00:00
|
|
|
|
2015-06-05 22:08:07 +00:00
|
|
|
// The listeners to accept raft and remote exec connections from.
|
|
|
|
RaftListener net.Listener
|
|
|
|
ExecListener net.Listener
|
2015-06-05 20:40:18 +00:00
|
|
|
|
2015-06-05 22:08:07 +00:00
|
|
|
// The advertised hostname of the store.
|
|
|
|
Addr net.Addr
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
// The amount of time before a follower starts a new election.
|
|
|
|
HeartbeatTimeout time.Duration
|
|
|
|
|
|
|
|
// The amount of time before a candidate starts a new election.
|
|
|
|
ElectionTimeout time.Duration
|
|
|
|
|
|
|
|
// The amount of time without communication to the cluster before a
|
|
|
|
// leader steps down to a follower state.
|
|
|
|
LeaderLeaseTimeout time.Duration
|
|
|
|
|
|
|
|
// The amount of time without an apply before sending a heartbeat.
|
|
|
|
CommitTimeout time.Duration
|
|
|
|
|
|
|
|
Logger *log.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStore returns a new instance of Store.
|
2015-06-01 22:00:13 +00:00
|
|
|
func NewStore(c Config) *Store {
|
2015-05-20 20:57:04 +00:00
|
|
|
return &Store{
|
2015-06-05 20:40:18 +00:00
|
|
|
path: c.Dir,
|
|
|
|
peers: c.Peers,
|
|
|
|
data: &Data{},
|
|
|
|
|
|
|
|
ready: make(chan struct{}),
|
|
|
|
err: make(chan error),
|
|
|
|
closing: make(chan struct{}),
|
|
|
|
|
2015-06-07 00:03:52 +00:00
|
|
|
retentionAutoCreate: c.RetentionAutoCreate,
|
2015-06-06 16:29:20 +00:00
|
|
|
|
2015-06-01 22:00:13 +00:00
|
|
|
HeartbeatTimeout: time.Duration(c.HeartbeatTimeout),
|
|
|
|
ElectionTimeout: time.Duration(c.ElectionTimeout),
|
|
|
|
LeaderLeaseTimeout: time.Duration(c.LeaderLeaseTimeout),
|
|
|
|
CommitTimeout: time.Duration(c.CommitTimeout),
|
2015-05-20 20:57:04 +00:00
|
|
|
Logger: log.New(os.Stderr, "", log.LstdFlags),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Path returns the root path when open.
|
|
|
|
// Returns an empty string when the store is closed.
|
2015-05-27 19:41:58 +00:00
|
|
|
func (s *Store) Path() string { return s.path }
|
2015-05-08 17:51:50 +00:00
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// IDPath returns the path to the local node ID file.
|
|
|
|
func (s *Store) IDPath() string { return filepath.Join(s.path, "id") }
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
// Open opens and initializes the raft store.
|
2015-05-27 19:41:58 +00:00
|
|
|
func (s *Store) Open() error {
|
2015-06-05 22:08:07 +00:00
|
|
|
// Verify listeners are set.
|
|
|
|
if s.RaftListener == nil {
|
|
|
|
panic("Store.RaftListener not set")
|
|
|
|
} else if s.ExecListener == nil {
|
|
|
|
panic("Store.ExecListener not set")
|
|
|
|
}
|
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
if err := func() error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// Check if store has already been opened.
|
|
|
|
if s.opened {
|
|
|
|
return ErrStoreOpen
|
|
|
|
}
|
|
|
|
s.opened = true
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
// Create the root directory if it doesn't already exist.
|
2015-05-27 19:41:58 +00:00
|
|
|
if err := os.MkdirAll(s.path, 0777); err != nil {
|
2015-05-20 20:57:04 +00:00
|
|
|
return fmt.Errorf("mkdir all: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Open the raft store.
|
|
|
|
if err := s.openRaft(); err != nil {
|
|
|
|
return fmt.Errorf("raft: %s", err)
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Initialize the store, if necessary.
|
|
|
|
if err := s.initialize(); err != nil {
|
|
|
|
return fmt.Errorf("initialize raft: %s", err)
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Load existing ID, if exists.
|
2015-06-01 17:20:57 +00:00
|
|
|
if err := s.readID(); err != nil {
|
|
|
|
return fmt.Errorf("read id: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
return nil
|
|
|
|
}(); err != nil {
|
|
|
|
s.close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Begin serving listener.
|
|
|
|
s.wg.Add(1)
|
2015-06-05 22:08:07 +00:00
|
|
|
go s.serveExecListener()
|
2015-06-05 20:40:18 +00:00
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// If the ID doesn't exist then create a new node.
|
|
|
|
if s.id == 0 {
|
2015-06-05 20:40:18 +00:00
|
|
|
go s.init()
|
|
|
|
} else {
|
|
|
|
close(s.ready)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// openRaft initializes the raft store.
|
|
|
|
func (s *Store) openRaft() error {
|
|
|
|
// Setup raft configuration.
|
|
|
|
config := raft.DefaultConfig()
|
|
|
|
config.Logger = s.Logger
|
|
|
|
config.HeartbeatTimeout = s.HeartbeatTimeout
|
|
|
|
config.ElectionTimeout = s.ElectionTimeout
|
|
|
|
config.LeaderLeaseTimeout = s.LeaderLeaseTimeout
|
|
|
|
config.CommitTimeout = s.CommitTimeout
|
|
|
|
|
|
|
|
// If no peers are set in the config then start as a single server.
|
|
|
|
config.EnableSingleNode = (len(s.peers) == 0)
|
|
|
|
|
|
|
|
// Build raft layer to multiplex listener.
|
2015-06-05 22:08:07 +00:00
|
|
|
s.raftLayer = newRaftLayer(s.RaftListener, s.Addr)
|
2015-06-05 20:40:18 +00:00
|
|
|
|
|
|
|
// Create a transport layer
|
|
|
|
s.transport = raft.NewNetworkTransport(s.raftLayer, 3, 10*time.Second, os.Stderr)
|
|
|
|
|
|
|
|
// Create peer storage.
|
|
|
|
s.peerStore = raft.NewJSONPeers(s.path, s.transport)
|
|
|
|
|
|
|
|
// Create the log store and stable store.
|
|
|
|
store, err := raftboltdb.NewBoltStore(filepath.Join(s.path, "raft.db"))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("new bolt store: %s", err)
|
|
|
|
}
|
|
|
|
s.store = store
|
|
|
|
|
|
|
|
// Create the snapshot store.
|
|
|
|
snapshots, err := raft.NewFileSnapshotStore(s.path, raftSnapshotsRetained, os.Stderr)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("file snapshot store: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create raft log.
|
|
|
|
r, err := raft.NewRaft(config, (*storeFSM)(s), store, store, snapshots, s.peerStore, s.transport)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("new raft: %s", err)
|
|
|
|
}
|
|
|
|
s.raft = r
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// initialize attempts to bootstrap the raft store if there are no committed entries.
|
|
|
|
func (s *Store) initialize() error {
|
|
|
|
// If we have committed entries then the store is already in the cluster.
|
|
|
|
/*
|
|
|
|
if index, err := s.store.LastIndex(); err != nil {
|
|
|
|
return fmt.Errorf("last index: %s", err)
|
|
|
|
} else if index > 0 {
|
|
|
|
return nil
|
2015-06-01 17:20:57 +00:00
|
|
|
}
|
2015-06-05 20:40:18 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
// Force set peers.
|
|
|
|
if err := s.SetPeers(s.peers); err != nil {
|
|
|
|
return fmt.Errorf("set raft peers: %s", err)
|
2015-06-01 17:20:57 +00:00
|
|
|
}
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
return nil
|
2015-05-08 17:51:50 +00:00
|
|
|
}
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
// Close closes the store and shuts down the node in the cluster.
|
|
|
|
func (s *Store) Close() error {
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
return s.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Store) close() error {
|
|
|
|
// Check if store has already been closed.
|
2015-05-27 19:41:58 +00:00
|
|
|
if !s.opened {
|
2015-05-21 20:06:01 +00:00
|
|
|
return ErrStoreClosed
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
2015-05-27 19:41:58 +00:00
|
|
|
s.opened = false
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Notify goroutines of close.
|
|
|
|
close(s.closing)
|
|
|
|
// FIXME(benbjohnson): s.wg.Wait()
|
|
|
|
|
|
|
|
// Shutdown raft.
|
2015-05-20 20:57:04 +00:00
|
|
|
if s.raft != nil {
|
|
|
|
s.raft.Shutdown()
|
|
|
|
s.raft = nil
|
|
|
|
}
|
|
|
|
if s.transport != nil {
|
|
|
|
s.transport.Close()
|
|
|
|
s.transport = nil
|
|
|
|
}
|
|
|
|
if s.store != nil {
|
|
|
|
s.store.Close()
|
|
|
|
s.store = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// readID reads the local node ID from the ID file.
|
|
|
|
func (s *Store) readID() error {
|
|
|
|
b, err := ioutil.ReadFile(s.IDPath())
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
s.id = 0
|
|
|
|
return nil
|
|
|
|
} else if err != nil {
|
|
|
|
return fmt.Errorf("read file: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
id, err := strconv.ParseUint(string(b), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("parse id: %s", err)
|
|
|
|
}
|
|
|
|
s.id = id
|
|
|
|
|
|
|
|
s.Logger.Printf("read local node id: %d", s.id)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// init initializes the store in a separate goroutine.
|
|
|
|
// This occurs when the store first creates or joins a cluster.
|
|
|
|
// The ready channel is closed once the store is initialized.
|
|
|
|
func (s *Store) init() {
|
|
|
|
// Create a node for this store.
|
|
|
|
if err := s.createLocalNode(); err != nil {
|
|
|
|
s.err <- fmt.Errorf("create local node: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notify the ready channel.
|
|
|
|
close(s.ready)
|
|
|
|
}
|
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// createLocalNode creates the node for this local instance.
|
|
|
|
// Writes the id of the node to file on success.
|
|
|
|
func (s *Store) createLocalNode() error {
|
|
|
|
// Wait for leader.
|
2015-06-05 20:40:18 +00:00
|
|
|
if err := s.waitForLeader(5 * time.Second); err != nil {
|
|
|
|
return fmt.Errorf("wait for leader: %s", err)
|
|
|
|
}
|
2015-06-01 17:20:57 +00:00
|
|
|
|
|
|
|
// Create new node.
|
2015-06-05 22:08:07 +00:00
|
|
|
ni, err := s.CreateNode(s.Addr.String())
|
2015-06-01 17:20:57 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("create node: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write node id to file.
|
|
|
|
if err := ioutil.WriteFile(s.IDPath(), []byte(strconv.FormatUint(ni.ID, 10)), 0666); err != nil {
|
|
|
|
return fmt.Errorf("write file: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set ID locally.
|
|
|
|
s.id = ni.ID
|
|
|
|
|
2015-06-05 22:08:07 +00:00
|
|
|
s.Logger.Printf("created local node: id=%d, host=%s", s.id, s.Addr.String())
|
2015-06-01 17:20:57 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// waitForLeader sleeps until a leader is found or a timeout occurs.
|
|
|
|
func (s *Store) waitForLeader(timeout time.Duration) error {
|
|
|
|
// Begin timeout timer.
|
|
|
|
timer := time.NewTimer(timeout)
|
|
|
|
defer timer.Stop()
|
|
|
|
|
|
|
|
// Continually check for leader until timeout.
|
|
|
|
ticker := time.NewTicker(100 * time.Millisecond)
|
|
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-s.closing:
|
|
|
|
return errors.New("closing")
|
|
|
|
case <-timer.C:
|
|
|
|
return errors.New("timeout")
|
|
|
|
case <-ticker.C:
|
|
|
|
if s.raft.Leader() != "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ready returns a channel that is closed once the store is initialized.
|
|
|
|
func (s *Store) Ready() <-chan struct{} { return s.ready }
|
|
|
|
|
|
|
|
// Err returns a channel for all out-of-band errors.
|
|
|
|
func (s *Store) Err() <-chan error { return s.err }
|
|
|
|
|
|
|
|
// IsLeader returns true if the store is currently the leader.
|
2015-06-06 00:49:12 +00:00
|
|
|
func (s *Store) IsLeader() bool {
|
|
|
|
s.mu.RLock()
|
|
|
|
defer s.mu.RUnlock()
|
2015-06-06 20:24:41 +00:00
|
|
|
if s.raft == nil {
|
|
|
|
return false
|
|
|
|
}
|
2015-06-06 00:49:12 +00:00
|
|
|
return s.raft.State() == raft.Leader
|
|
|
|
}
|
2015-06-05 20:40:18 +00:00
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
// LeaderCh returns a channel that notifies on leadership change.
|
|
|
|
// Panics when the store has not been opened yet.
|
|
|
|
func (s *Store) LeaderCh() <-chan bool {
|
|
|
|
s.mu.RLock()
|
|
|
|
defer s.mu.RUnlock()
|
2015-05-21 20:06:01 +00:00
|
|
|
assert(s.raft != nil, "cannot retrieve leadership channel when closed")
|
|
|
|
return s.raft.LeaderCh()
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// SetPeers sets a list of peers in the cluster.
|
|
|
|
func (s *Store) SetPeers(addrs []string) error {
|
|
|
|
a := make([]string, len(addrs))
|
|
|
|
for i, s := range addrs {
|
|
|
|
addr, err := net.ResolveTCPAddr("tcp", s)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("cannot resolve addr: %s, err=%s", s, err)
|
|
|
|
}
|
|
|
|
a[i] = addr.String()
|
|
|
|
}
|
|
|
|
return s.raft.SetPeers(a).Error()
|
|
|
|
}
|
|
|
|
|
2015-06-05 22:08:07 +00:00
|
|
|
// serveExecListener processes remote exec connections.
|
2015-06-05 20:40:18 +00:00
|
|
|
// This function runs in a separate goroutine.
|
2015-06-05 22:08:07 +00:00
|
|
|
func (s *Store) serveExecListener() {
|
2015-06-05 20:40:18 +00:00
|
|
|
defer s.wg.Done()
|
|
|
|
|
|
|
|
for {
|
|
|
|
// Accept next TCP connection.
|
2015-06-05 22:08:07 +00:00
|
|
|
conn, err := s.ExecListener.Accept()
|
2015-06-05 20:40:18 +00:00
|
|
|
if err != nil {
|
2015-06-06 03:58:54 +00:00
|
|
|
if strings.Contains(err.Error(), "connection closed") {
|
|
|
|
return
|
|
|
|
} else {
|
|
|
|
s.Logger.Printf("temporary accept error: %s", err)
|
|
|
|
continue
|
|
|
|
}
|
2015-06-05 20:40:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Handle connection in a separate goroutine.
|
|
|
|
s.wg.Add(1)
|
2015-06-05 22:08:07 +00:00
|
|
|
go s.handleExecConn(conn)
|
2015-06-05 20:40:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleExecConn reads a command from the connection and executes it.
|
|
|
|
func (s *Store) handleExecConn(conn net.Conn) {
|
2015-06-08 18:10:02 +00:00
|
|
|
defer s.wg.Done()
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Read and execute command.
|
|
|
|
err := func() error {
|
|
|
|
// Read marker message.
|
|
|
|
b := make([]byte, 4)
|
|
|
|
if _, err := io.ReadFull(conn, b); err != nil {
|
2015-06-05 22:08:07 +00:00
|
|
|
return fmt.Errorf("read magic: %s", err)
|
|
|
|
} else if string(b) != ExecMagic {
|
|
|
|
return fmt.Errorf("invalid exec magic: %q", string(b))
|
2015-06-05 20:40:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read command size.
|
|
|
|
var sz uint64
|
|
|
|
if err := binary.Read(conn, binary.BigEndian, &sz); err != nil {
|
|
|
|
return fmt.Errorf("read size: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read command.
|
|
|
|
buf := make([]byte, sz)
|
|
|
|
if _, err := io.ReadFull(conn, buf); err != nil {
|
|
|
|
return fmt.Errorf("read command: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure command can be deserialized before applying.
|
|
|
|
if err := proto.Unmarshal(buf, &internal.Command{}); err != nil {
|
|
|
|
return fmt.Errorf("unable to unmarshal command: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply against the raft log.
|
|
|
|
if err := s.apply(buf); err != nil {
|
|
|
|
return fmt.Errorf("apply: %s", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Build response message.
|
|
|
|
var resp internal.Response
|
|
|
|
resp.OK = proto.Bool(err == nil)
|
|
|
|
resp.Index = proto.Uint64(s.raft.LastIndex())
|
|
|
|
if err != nil {
|
|
|
|
resp.Error = proto.String(err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Encode response back to connection.
|
|
|
|
if b, err := proto.Marshal(&resp); err != nil {
|
|
|
|
panic(err)
|
|
|
|
} else if err = binary.Write(conn, binary.BigEndian, uint64(len(b))); err != nil {
|
|
|
|
s.Logger.Printf("unable to write exec response size: %s", err)
|
|
|
|
} else if _, err = conn.Write(b); err != nil {
|
|
|
|
s.Logger.Printf("unable to write exec response: %s", err)
|
|
|
|
}
|
|
|
|
conn.Close()
|
|
|
|
}
|
|
|
|
|
2015-06-01 17:20:57 +00:00
|
|
|
// NodeID returns the identifier for the local node.
|
|
|
|
// Panics if the node has not joined the cluster.
|
|
|
|
func (s *Store) NodeID() uint64 { return s.id }
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Node returns a node by id.
|
|
|
|
func (s *Store) Node(id uint64) (ni *NodeInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
ni = data.Node(id)
|
|
|
|
if ni == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// NodeByHost returns a node by hostname.
|
|
|
|
func (s *Store) NodeByHost(host string) (ni *NodeInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
ni = data.NodeByHost(host)
|
|
|
|
if ni == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// Nodes returns a list of all nodes.
|
|
|
|
func (s *Store) Nodes() (a []NodeInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
a = data.Nodes
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-20 20:57:04 +00:00
|
|
|
// CreateNode creates a new node in the store.
|
|
|
|
func (s *Store) CreateNode(host string) (*NodeInfo, error) {
|
2015-05-21 20:06:01 +00:00
|
|
|
if err := s.exec(internal.Command_CreateNodeCommand, internal.E_CreateNodeCommand_Command,
|
2015-05-20 20:57:04 +00:00
|
|
|
&internal.CreateNodeCommand{
|
|
|
|
Host: proto.String(host),
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return s.NodeByHost(host)
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// DeleteNode removes a node from the metastore by id.
|
|
|
|
func (s *Store) DeleteNode(id uint64) error {
|
|
|
|
return s.exec(internal.Command_DeleteNodeCommand, internal.E_DeleteNodeCommand_Command,
|
|
|
|
&internal.DeleteNodeCommand{
|
|
|
|
ID: proto.Uint64(id),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Database returns a database by name.
|
|
|
|
func (s *Store) Database(name string) (di *DatabaseInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
di = data.Database(name)
|
|
|
|
if di == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// Databases returns a list of all databases.
|
|
|
|
func (s *Store) Databases() (dis []DatabaseInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
dis = data.Databases
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// CreateDatabase creates a new database in the store.
|
|
|
|
func (s *Store) CreateDatabase(name string) (*DatabaseInfo, error) {
|
|
|
|
if err := s.exec(internal.Command_CreateDatabaseCommand, internal.E_CreateDatabaseCommand_Command,
|
|
|
|
&internal.CreateDatabaseCommand{
|
|
|
|
Name: proto.String(name),
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-06-06 16:24:42 +00:00
|
|
|
|
2015-06-07 00:03:52 +00:00
|
|
|
if s.retentionAutoCreate {
|
|
|
|
rpi := NewRetentionPolicyInfo(AutoCreateRetentionPolicyName)
|
|
|
|
rpi.ReplicaN = AutoCreateRetentionPolicyReplicaN
|
2015-06-09 02:47:37 +00:00
|
|
|
rpi.Duration = AutoCreateRetentionPolicyPeriod
|
2015-06-06 16:41:27 +00:00
|
|
|
if _, err := s.CreateRetentionPolicy(name, rpi); err != nil {
|
2015-06-06 16:24:42 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2015-06-07 00:03:52 +00:00
|
|
|
if err := s.SetDefaultRetentionPolicy(name, AutoCreateRetentionPolicyName); err != nil {
|
2015-06-06 16:24:42 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
return s.Database(name)
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// CreateDatabaseIfNotExists creates a new database in the store if it doesn't already exist.
|
|
|
|
func (s *Store) CreateDatabaseIfNotExists(name string) (*DatabaseInfo, error) {
|
|
|
|
// Try to find database locally first.
|
|
|
|
if di, err := s.Database(name); err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if di != nil {
|
|
|
|
return di, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to create database.
|
|
|
|
if di, err := s.CreateDatabase(name); err == ErrDatabaseExists {
|
|
|
|
return s.Database(name)
|
|
|
|
} else {
|
|
|
|
return di, err
|
|
|
|
}
|
|
|
|
}
|
2015-05-21 20:06:01 +00:00
|
|
|
|
|
|
|
// DropDatabase removes a database from the metastore by name.
|
|
|
|
func (s *Store) DropDatabase(name string) error {
|
|
|
|
return s.exec(internal.Command_DropDatabaseCommand, internal.E_DropDatabaseCommand_Command,
|
|
|
|
&internal.DropDatabaseCommand{
|
|
|
|
Name: proto.String(name),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RetentionPolicy returns a retention policy for a database by name.
|
|
|
|
func (s *Store) RetentionPolicy(database, name string) (rpi *RetentionPolicyInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
rpi, err = data.RetentionPolicy(database, name)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else if rpi == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// DefaultRetentionPolicy returns the default retention policy for a database.
|
|
|
|
func (s *Store) DefaultRetentionPolicy(database string) (rpi *RetentionPolicyInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
di := data.Database(database)
|
|
|
|
if di == nil {
|
|
|
|
return ErrDatabaseNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := range di.RetentionPolicies {
|
|
|
|
if di.RetentionPolicies[i].Name == di.DefaultRetentionPolicy {
|
|
|
|
rpi = &di.RetentionPolicies[i]
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return errInvalidate
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// RetentionPolicies returns a list of all retention policies for a database.
|
|
|
|
func (s *Store) RetentionPolicies(database string) (a []RetentionPolicyInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
di := data.Database(database)
|
|
|
|
if di != nil {
|
|
|
|
return ErrDatabaseNotFound
|
|
|
|
}
|
|
|
|
a = di.RetentionPolicies
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// CreateRetentionPolicy creates a new retention policy for a database.
|
|
|
|
func (s *Store) CreateRetentionPolicy(database string, rpi *RetentionPolicyInfo) (*RetentionPolicyInfo, error) {
|
|
|
|
if err := s.exec(internal.Command_CreateRetentionPolicyCommand, internal.E_CreateRetentionPolicyCommand_Command,
|
|
|
|
&internal.CreateRetentionPolicyCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
RetentionPolicy: rpi.protobuf(),
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.RetentionPolicy(database, rpi.Name)
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// CreateRetentionPolicyIfNotExists creates a new policy in the store if it doesn't already exist.
|
|
|
|
func (s *Store) CreateRetentionPolicyIfNotExists(database string, rpi *RetentionPolicyInfo) (*RetentionPolicyInfo, error) {
|
|
|
|
// Try to find policy locally first.
|
|
|
|
if rpi, err := s.RetentionPolicy(database, rpi.Name); err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if rpi != nil {
|
|
|
|
return rpi, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to create policy.
|
|
|
|
if other, err := s.CreateRetentionPolicy(database, rpi); err == ErrRetentionPolicyExists {
|
|
|
|
return s.RetentionPolicy(database, rpi.Name)
|
|
|
|
} else {
|
|
|
|
return other, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// SetDefaultRetentionPolicy sets the default retention policy for a database.
|
|
|
|
func (s *Store) SetDefaultRetentionPolicy(database, name string) error {
|
|
|
|
return s.exec(internal.Command_SetDefaultRetentionPolicyCommand, internal.E_SetDefaultRetentionPolicyCommand_Command,
|
|
|
|
&internal.SetDefaultRetentionPolicyCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
Name: proto.String(name),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateRetentionPolicy updates an existing retention policy.
|
|
|
|
func (s *Store) UpdateRetentionPolicy(database, name string, rpu *RetentionPolicyUpdate) error {
|
|
|
|
var newName *string
|
|
|
|
if rpu.Name != nil {
|
|
|
|
newName = rpu.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
var duration *int64
|
|
|
|
if rpu.Duration != nil {
|
|
|
|
value := int64(*rpu.Duration)
|
|
|
|
duration = &value
|
|
|
|
}
|
|
|
|
|
|
|
|
var replicaN *uint32
|
2015-06-09 18:37:49 +00:00
|
|
|
if rpu.ReplicaN != nil {
|
2015-05-21 20:06:01 +00:00
|
|
|
value := uint32(*rpu.ReplicaN)
|
|
|
|
replicaN = &value
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.exec(internal.Command_UpdateRetentionPolicyCommand, internal.E_UpdateRetentionPolicyCommand_Command,
|
|
|
|
&internal.UpdateRetentionPolicyCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
Name: proto.String(name),
|
|
|
|
NewName: newName,
|
|
|
|
Duration: duration,
|
|
|
|
ReplicaN: replicaN,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DropRetentionPolicy removes a policy from a database by name.
|
|
|
|
func (s *Store) DropRetentionPolicy(database, name string) error {
|
|
|
|
return s.exec(internal.Command_DropRetentionPolicyCommand, internal.E_DropRetentionPolicyCommand_Command,
|
|
|
|
&internal.DropRetentionPolicyCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
Name: proto.String(name),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIX: CreateRetentionPolicyIfNotExists(database string, rp *RetentionPolicyInfo) (*RetentionPolicyInfo, error)
|
|
|
|
|
|
|
|
// CreateShardGroup creates a new shard group in a retention policy for a given time.
|
|
|
|
func (s *Store) CreateShardGroup(database, policy string, timestamp time.Time) (*ShardGroupInfo, error) {
|
|
|
|
if err := s.exec(internal.Command_CreateShardGroupCommand, internal.E_CreateShardGroupCommand_Command,
|
|
|
|
&internal.CreateShardGroupCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
Policy: proto.String(policy),
|
|
|
|
Timestamp: proto.Int64(timestamp.UnixNano()),
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.ShardGroupByTimestamp(database, policy, timestamp)
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// CreateShardGroupIfNotExists creates a new shard group if one doesn't already exist.
|
|
|
|
func (s *Store) CreateShardGroupIfNotExists(database, policy string, timestamp time.Time) (*ShardGroupInfo, error) {
|
|
|
|
// Try to find shard group locally first.
|
|
|
|
if sgi, err := s.ShardGroupByTimestamp(database, policy, timestamp); err != nil {
|
|
|
|
return nil, err
|
2015-06-04 22:50:33 +00:00
|
|
|
} else if sgi != nil && !sgi.Deleted() {
|
2015-05-27 19:41:58 +00:00
|
|
|
return sgi, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to create database.
|
|
|
|
if sgi, err := s.CreateShardGroup(database, policy, timestamp); err == ErrShardGroupExists {
|
|
|
|
return s.ShardGroupByTimestamp(database, policy, timestamp)
|
|
|
|
} else {
|
|
|
|
return sgi, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// DeleteShardGroup removes an existing shard group from a policy by ID.
|
|
|
|
func (s *Store) DeleteShardGroup(database, policy string, id uint64) error {
|
|
|
|
return s.exec(internal.Command_DeleteShardGroupCommand, internal.E_DeleteShardGroupCommand_Command,
|
|
|
|
&internal.DeleteShardGroupCommand{
|
|
|
|
Database: proto.String(database),
|
|
|
|
Policy: proto.String(policy),
|
|
|
|
ShardGroupID: proto.Uint64(id),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// ShardGroups returns a list of all shard groups for a policy by timestamp.
|
|
|
|
func (s *Store) ShardGroups(database, policy string) (a []ShardGroupInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
a, err = data.ShardGroups(database, policy)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-06-05 05:18:52 +00:00
|
|
|
// VisitRetentionPolicies calls the given function with full retention policy details.
|
|
|
|
func (s *Store) VisitRetentionPolicies(f func(d DatabaseInfo, r RetentionPolicyInfo)) {
|
|
|
|
s.read(func(data *Data) error {
|
|
|
|
for _, di := range data.Databases {
|
|
|
|
for _, rp := range di.RetentionPolicies {
|
|
|
|
f(di, rp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// ShardGroupByTimestamp returns a shard group for a policy by timestamp.
|
|
|
|
func (s *Store) ShardGroupByTimestamp(database, policy string, timestamp time.Time) (sgi *ShardGroupInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
sgi, err = data.ShardGroupByTimestamp(database, policy, timestamp)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else if sgi == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-06-05 21:32:15 +00:00
|
|
|
func (s *Store) ShardOwner(shardID uint64) (database, policy string, sgi *ShardGroupInfo) {
|
2015-06-06 05:23:10 +00:00
|
|
|
s.read(func(data *Data) error {
|
|
|
|
for _, dbi := range data.Databases {
|
|
|
|
for _, rpi := range dbi.RetentionPolicies {
|
|
|
|
for _, g := range rpi.ShardGroups {
|
|
|
|
if g.Deleted() {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, sh := range g.Shards {
|
|
|
|
if sh.ID == shardID {
|
|
|
|
database = dbi.Name
|
|
|
|
policy = rpi.Name
|
|
|
|
sgi = &g
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-06-05 21:32:15 +00:00
|
|
|
}
|
|
|
|
}
|
2015-06-06 05:23:10 +00:00
|
|
|
return errInvalidate
|
2015-06-05 21:32:15 +00:00
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// CreateContinuousQuery creates a new continuous query on the store.
|
2015-05-28 04:06:09 +00:00
|
|
|
func (s *Store) CreateContinuousQuery(database, name, query string) error {
|
2015-05-21 20:06:01 +00:00
|
|
|
return s.exec(internal.Command_CreateContinuousQueryCommand, internal.E_CreateContinuousQueryCommand_Command,
|
|
|
|
&internal.CreateContinuousQueryCommand{
|
2015-05-28 04:06:09 +00:00
|
|
|
Database: proto.String(database),
|
|
|
|
Name: proto.String(name),
|
|
|
|
Query: proto.String(query),
|
2015-05-21 20:06:01 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DropContinuousQuery removes a continuous query from the store.
|
2015-05-28 04:06:09 +00:00
|
|
|
func (s *Store) DropContinuousQuery(database, name string) error {
|
2015-05-21 20:06:01 +00:00
|
|
|
return s.exec(internal.Command_DropContinuousQueryCommand, internal.E_DropContinuousQueryCommand_Command,
|
|
|
|
&internal.DropContinuousQueryCommand{
|
2015-05-28 04:06:09 +00:00
|
|
|
Database: proto.String(database),
|
|
|
|
Name: proto.String(name),
|
2015-05-21 20:06:01 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// User returns a user by name.
|
|
|
|
func (s *Store) User(name string) (ui *UserInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
ui = data.User(name)
|
|
|
|
if ui == nil {
|
|
|
|
return errInvalidate
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Users returns a list of all users.
|
|
|
|
func (s *Store) Users() (a []UserInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
a = data.Users
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// AdminUserExists returns true if an admin user exists on the system.
|
|
|
|
func (s *Store) AdminUserExists() (exists bool, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
for i := range data.Users {
|
|
|
|
if data.Users[i].Admin {
|
|
|
|
exists = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Authenticate retrieves a user with a matching username and password.
|
|
|
|
func (s *Store) Authenticate(username, password string) (ui *UserInfo, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
// Find user.
|
|
|
|
u := data.User(username)
|
|
|
|
if u == nil {
|
|
|
|
return ErrUserNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compare password with user hash.
|
|
|
|
if err := bcrypt.CompareHashAndPassword([]byte(u.Hash), []byte(password)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
ui = u
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// CreateUser creates a new user in the store.
|
|
|
|
func (s *Store) CreateUser(name, password string, admin bool) (*UserInfo, error) {
|
|
|
|
// Hash the password before serializing it.
|
|
|
|
hash, err := HashPassword(password)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Serialize command and send it to the leader.
|
|
|
|
if err := s.exec(internal.Command_CreateUserCommand, internal.E_CreateUserCommand_Command,
|
|
|
|
&internal.CreateUserCommand{
|
|
|
|
Name: proto.String(name),
|
|
|
|
Hash: proto.String(string(hash)),
|
|
|
|
Admin: proto.Bool(admin),
|
|
|
|
},
|
|
|
|
); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return s.User(name)
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// DropUser removes a user from the metastore by name.
|
|
|
|
func (s *Store) DropUser(name string) error {
|
|
|
|
return s.exec(internal.Command_DropUserCommand, internal.E_DropUserCommand_Command,
|
|
|
|
&internal.DropUserCommand{
|
|
|
|
Name: proto.String(name),
|
|
|
|
},
|
|
|
|
)
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// UpdateUser updates an existing user in the store.
|
|
|
|
func (s *Store) UpdateUser(name, password string) error {
|
|
|
|
// Hash the password before serializing it.
|
|
|
|
hash, err := HashPassword(password)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Serialize command and send it to the leader.
|
|
|
|
return s.exec(internal.Command_UpdateUserCommand, internal.E_UpdateUserCommand_Command,
|
|
|
|
&internal.UpdateUserCommand{
|
|
|
|
Name: proto.String(name),
|
|
|
|
Hash: proto.String(string(hash)),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-27 19:41:58 +00:00
|
|
|
// SetPrivilege sets a privilege for a user on a database.
|
|
|
|
func (s *Store) SetPrivilege(username, database string, p influxql.Privilege) error {
|
|
|
|
return s.exec(internal.Command_SetPrivilegeCommand, internal.E_SetPrivilegeCommand_Command,
|
|
|
|
&internal.SetPrivilegeCommand{
|
|
|
|
Username: proto.String(username),
|
|
|
|
Database: proto.String(database),
|
|
|
|
Privilege: proto.Int32(int32(p)),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2015-05-30 18:29:16 +00:00
|
|
|
// UserCount returns the number of users defined in the cluster.
|
|
|
|
func (s *Store) UserCount() (count int, err error) {
|
|
|
|
err = s.read(func(data *Data) error {
|
|
|
|
count = len(data.Users)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// read executes a function with the current metadata.
|
|
|
|
// If an error is returned then the cache is invalidated and retried.
|
|
|
|
//
|
|
|
|
// The error returned by the retry is passed through to the original caller
|
|
|
|
// unless the error is errInvalidate. A nil error is passed through when
|
|
|
|
// errInvalidate is returned.
|
|
|
|
func (s *Store) read(fn func(*Data) error) error {
|
|
|
|
// First use the cached metadata.
|
|
|
|
s.mu.RLock()
|
|
|
|
data := s.data
|
|
|
|
s.mu.RUnlock()
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Execute fn against cached data.
|
|
|
|
// Return immediately if there was no error.
|
|
|
|
if err := fn(data); err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// If an error occurred then invalidate cache and retry.
|
|
|
|
if err := s.invalidate(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Re-read the metadata.
|
|
|
|
s.mu.RLock()
|
|
|
|
data = s.data
|
|
|
|
s.mu.RUnlock()
|
|
|
|
|
|
|
|
// Passthrough error unless it is a cache invalidation.
|
|
|
|
if err := fn(data); err != nil && err != errInvalidate {
|
|
|
|
return err
|
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// errInvalidate is returned to read() when the cache should be invalidated
|
|
|
|
// but an error should not be passed through to the caller.
|
|
|
|
var errInvalidate = errors.New("invalidate cache")
|
|
|
|
|
|
|
|
func (s *Store) invalidate() error {
|
2015-06-05 20:40:18 +00:00
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
return nil // FIXME(benbjohnson): Reload cache from the leader.
|
2015-05-21 20:06:01 +00:00
|
|
|
}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
func (s *Store) exec(typ internal.Command_Type, desc *proto.ExtensionDesc, value interface{}) error {
|
|
|
|
// Create command.
|
|
|
|
cmd := &internal.Command{Type: &typ}
|
2015-05-21 20:06:01 +00:00
|
|
|
err := proto.SetExtension(cmd, desc, value)
|
|
|
|
assert(err == nil, "proto.SetExtension: %s", err)
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
// Marshal to a byte slice.
|
|
|
|
b, err := proto.Marshal(cmd)
|
2015-05-21 20:06:01 +00:00
|
|
|
assert(err == nil, "proto.Marshal: %s", err)
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// Apply the command if this is the leader.
|
|
|
|
// Otherwise remotely execute the command against the current leader.
|
|
|
|
if s.raft.State() == raft.Leader {
|
|
|
|
return s.apply(b)
|
|
|
|
} else {
|
|
|
|
return s.remoteExec(b)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// apply applies a serialized command to the raft log.
|
|
|
|
func (s *Store) apply(b []byte) error {
|
2015-05-20 20:57:04 +00:00
|
|
|
// Apply to raft log.
|
|
|
|
f := s.raft.Apply(b, 0)
|
|
|
|
if err := f.Error(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Return response if it's an error.
|
|
|
|
// No other non-nil objects should be returned.
|
|
|
|
resp := f.Response()
|
|
|
|
if err, ok := resp.(error); ok {
|
|
|
|
return lookupError(err)
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
2015-05-21 20:06:01 +00:00
|
|
|
assert(resp == nil, "unexpected response: %#v", resp)
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// remoteExec sends an encoded command to the remote leader.
|
|
|
|
func (s *Store) remoteExec(b []byte) error {
|
|
|
|
// Retrieve the current known leader.
|
|
|
|
leader := s.raft.Leader()
|
|
|
|
if leader == "" {
|
|
|
|
return errors.New("no leader")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a connection to the leader.
|
|
|
|
conn, err := net.DialTimeout("tcp", leader, 10*time.Second)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer conn.Close()
|
|
|
|
|
|
|
|
// Write a marker byte for exec messages.
|
2015-06-05 22:08:07 +00:00
|
|
|
_, err = conn.Write([]byte{MuxExecHeader})
|
2015-06-05 20:40:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write a marker message.
|
2015-06-05 22:08:07 +00:00
|
|
|
_, err = conn.Write([]byte(ExecMagic))
|
2015-06-05 20:40:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write command size & bytes.
|
|
|
|
if err := binary.Write(conn, binary.BigEndian, uint64(len(b))); err != nil {
|
|
|
|
return fmt.Errorf("write command size: %s", err)
|
|
|
|
} else if _, err := conn.Write(b); err != nil {
|
|
|
|
return fmt.Errorf("write command: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read response bytes.
|
|
|
|
var sz uint64
|
|
|
|
if err := binary.Read(conn, binary.BigEndian, &sz); err != nil {
|
|
|
|
return fmt.Errorf("read response size: %s", err)
|
|
|
|
}
|
|
|
|
buf := make([]byte, sz)
|
|
|
|
if _, err := io.ReadFull(conn, buf); err != nil {
|
|
|
|
return fmt.Errorf("read response: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal response.
|
|
|
|
var resp internal.Response
|
|
|
|
if err := proto.Unmarshal(buf, &resp); err != nil {
|
|
|
|
return fmt.Errorf("unmarshal response: %s", err)
|
|
|
|
} else if !resp.GetOK() {
|
|
|
|
return fmt.Errorf("exec failed: %s", resp.GetError())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for local FSM to sync to index.
|
|
|
|
if err := s.sync(resp.GetIndex(), 5*time.Second); err != nil {
|
|
|
|
return fmt.Errorf("sync: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// sync polls the state machine until it reaches a given index.
|
|
|
|
func (s *Store) sync(index uint64, timeout time.Duration) error {
|
|
|
|
ticker := time.NewTicker(100 * time.Millisecond)
|
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
timer := time.NewTimer(timeout)
|
|
|
|
defer timer.Stop()
|
|
|
|
|
|
|
|
for {
|
|
|
|
// Wait for next tick or timeout.
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
|
|
|
case <-timer.C:
|
|
|
|
return errors.New("timeout")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compare index against current metadata.
|
|
|
|
s.mu.Lock()
|
|
|
|
ok := (s.data.Index >= index)
|
|
|
|
s.mu.Unlock()
|
|
|
|
|
|
|
|
// Exit if we are at least at the given index.
|
|
|
|
if ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// storeFSM represents the finite state machine used by Store to interact with Raft.
|
|
|
|
type storeFSM Store
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) Apply(l *raft.Log) interface{} {
|
2015-05-20 20:57:04 +00:00
|
|
|
var cmd internal.Command
|
|
|
|
if err := proto.Unmarshal(l.Data, &cmd); err != nil {
|
|
|
|
panic(fmt.Errorf("cannot marshal command: %x", l.Data))
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
// Lock the store.
|
|
|
|
s := (*Store)(fsm)
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
err := func() interface{} {
|
|
|
|
switch cmd.GetType() {
|
|
|
|
case internal.Command_CreateNodeCommand:
|
|
|
|
return fsm.applyCreateNodeCommand(&cmd)
|
|
|
|
case internal.Command_DeleteNodeCommand:
|
|
|
|
return fsm.applyDeleteNodeCommand(&cmd)
|
|
|
|
case internal.Command_CreateDatabaseCommand:
|
|
|
|
return fsm.applyCreateDatabaseCommand(&cmd)
|
|
|
|
case internal.Command_DropDatabaseCommand:
|
|
|
|
return fsm.applyDropDatabaseCommand(&cmd)
|
|
|
|
case internal.Command_CreateRetentionPolicyCommand:
|
|
|
|
return fsm.applyCreateRetentionPolicyCommand(&cmd)
|
|
|
|
case internal.Command_DropRetentionPolicyCommand:
|
|
|
|
return fsm.applyDropRetentionPolicyCommand(&cmd)
|
|
|
|
case internal.Command_SetDefaultRetentionPolicyCommand:
|
|
|
|
return fsm.applySetDefaultRetentionPolicyCommand(&cmd)
|
|
|
|
case internal.Command_UpdateRetentionPolicyCommand:
|
|
|
|
return fsm.applyUpdateRetentionPolicyCommand(&cmd)
|
|
|
|
case internal.Command_CreateShardGroupCommand:
|
|
|
|
return fsm.applyCreateShardGroupCommand(&cmd)
|
|
|
|
case internal.Command_DeleteShardGroupCommand:
|
|
|
|
return fsm.applyDeleteShardGroupCommand(&cmd)
|
|
|
|
case internal.Command_CreateContinuousQueryCommand:
|
|
|
|
return fsm.applyCreateContinuousQueryCommand(&cmd)
|
|
|
|
case internal.Command_DropContinuousQueryCommand:
|
|
|
|
return fsm.applyDropContinuousQueryCommand(&cmd)
|
|
|
|
case internal.Command_CreateUserCommand:
|
|
|
|
return fsm.applyCreateUserCommand(&cmd)
|
|
|
|
case internal.Command_DropUserCommand:
|
|
|
|
return fsm.applyDropUserCommand(&cmd)
|
|
|
|
case internal.Command_UpdateUserCommand:
|
|
|
|
return fsm.applyUpdateUserCommand(&cmd)
|
|
|
|
case internal.Command_SetPrivilegeCommand:
|
|
|
|
return fsm.applySetPrivilegeCommand(&cmd)
|
|
|
|
default:
|
|
|
|
panic(fmt.Errorf("cannot apply command: %x", l.Data))
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Copy term and index to new metadata.
|
|
|
|
fsm.data.Term = l.Term
|
|
|
|
fsm.data.Index = l.Index
|
|
|
|
|
|
|
|
return err
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateNodeCommand(cmd *internal.Command) interface{} {
|
2015-05-20 20:57:04 +00:00
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateNodeCommand_Command)
|
|
|
|
v := ext.(*internal.CreateNodeCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
2015-05-21 20:06:01 +00:00
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.CreateNode(v.GetHost()); err != nil {
|
2015-05-20 20:57:04 +00:00
|
|
|
return err
|
|
|
|
}
|
2015-05-21 20:06:01 +00:00
|
|
|
fsm.data = other
|
2015-05-20 20:57:04 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDeleteNodeCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DeleteNodeCommand_Command)
|
|
|
|
v := ext.(*internal.DeleteNodeCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.DeleteNode(v.GetID()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateDatabaseCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateDatabaseCommand_Command)
|
|
|
|
v := ext.(*internal.CreateDatabaseCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.CreateDatabase(v.GetName()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDropDatabaseCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DropDatabaseCommand_Command)
|
|
|
|
v := ext.(*internal.DropDatabaseCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.DropDatabase(v.GetName()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateRetentionPolicyCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateRetentionPolicyCommand_Command)
|
|
|
|
v := ext.(*internal.CreateRetentionPolicyCommand)
|
|
|
|
pb := v.GetRetentionPolicy()
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.CreateRetentionPolicy(v.GetDatabase(),
|
|
|
|
&RetentionPolicyInfo{
|
|
|
|
Name: pb.GetName(),
|
|
|
|
ReplicaN: int(pb.GetReplicaN()),
|
|
|
|
Duration: time.Duration(pb.GetDuration()),
|
|
|
|
ShardGroupDuration: time.Duration(pb.GetShardGroupDuration()),
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDropRetentionPolicyCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DropRetentionPolicyCommand_Command)
|
|
|
|
v := ext.(*internal.DropRetentionPolicyCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.DropRetentionPolicy(v.GetDatabase(), v.GetName()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applySetDefaultRetentionPolicyCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_SetDefaultRetentionPolicyCommand_Command)
|
|
|
|
v := ext.(*internal.SetDefaultRetentionPolicyCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.SetDefaultRetentionPolicy(v.GetDatabase(), v.GetName()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyUpdateRetentionPolicyCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_UpdateRetentionPolicyCommand_Command)
|
|
|
|
v := ext.(*internal.UpdateRetentionPolicyCommand)
|
|
|
|
|
|
|
|
// Create update object.
|
|
|
|
rpu := RetentionPolicyUpdate{Name: v.NewName}
|
|
|
|
if v.Duration != nil {
|
|
|
|
value := time.Duration(v.GetDuration())
|
|
|
|
rpu.Duration = &value
|
|
|
|
}
|
|
|
|
if v.ReplicaN != nil {
|
|
|
|
value := int(v.GetReplicaN())
|
|
|
|
rpu.ReplicaN = &value
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.UpdateRetentionPolicy(v.GetDatabase(), v.GetName(), &rpu); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateShardGroupCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateShardGroupCommand_Command)
|
|
|
|
v := ext.(*internal.CreateShardGroupCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.CreateShardGroup(v.GetDatabase(), v.GetPolicy(), time.Unix(0, v.GetTimestamp())); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDeleteShardGroupCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DeleteShardGroupCommand_Command)
|
|
|
|
v := ext.(*internal.DeleteShardGroupCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.DeleteShardGroup(v.GetDatabase(), v.GetPolicy(), v.GetShardGroupID()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateContinuousQueryCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateContinuousQueryCommand_Command)
|
|
|
|
v := ext.(*internal.CreateContinuousQueryCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
2015-05-28 04:06:09 +00:00
|
|
|
if err := other.CreateContinuousQuery(v.GetDatabase(), v.GetName(), v.GetQuery()); err != nil {
|
2015-05-21 20:06:01 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDropContinuousQueryCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DropContinuousQueryCommand_Command)
|
|
|
|
v := ext.(*internal.DropContinuousQueryCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
2015-05-28 04:06:09 +00:00
|
|
|
if err := other.DropContinuousQuery(v.GetDatabase(), v.GetName()); err != nil {
|
2015-05-21 20:06:01 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyCreateUserCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_CreateUserCommand_Command)
|
|
|
|
v := ext.(*internal.CreateUserCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.CreateUser(v.GetName(), v.GetHash(), v.GetAdmin()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyDropUserCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_DropUserCommand_Command)
|
|
|
|
v := ext.(*internal.DropUserCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.DropUser(v.GetName()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applyUpdateUserCommand(cmd *internal.Command) interface{} {
|
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_UpdateUserCommand_Command)
|
|
|
|
v := ext.(*internal.UpdateUserCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.UpdateUser(v.GetName(), v.GetHash()); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) applySetPrivilegeCommand(cmd *internal.Command) interface{} {
|
2015-05-27 19:41:58 +00:00
|
|
|
ext, _ := proto.GetExtension(cmd, internal.E_SetPrivilegeCommand_Command)
|
|
|
|
v := ext.(*internal.SetPrivilegeCommand)
|
|
|
|
|
|
|
|
// Copy data and update.
|
|
|
|
other := fsm.data.Clone()
|
|
|
|
if err := other.SetPrivilege(v.GetUsername(), v.GetDatabase(), influxql.Privilege(v.GetPrivilege())); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fsm.data = other
|
|
|
|
return nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) Snapshot() (raft.FSMSnapshot, error) {
|
2015-05-20 20:57:04 +00:00
|
|
|
s := (*Store)(fsm)
|
|
|
|
s.mu.Lock()
|
|
|
|
defer s.mu.Unlock()
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
return &storeFSMSnapshot{Data: (*Store)(fsm).data}, nil
|
2015-05-20 20:57:04 +00:00
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (fsm *storeFSM) Restore(r io.ReadCloser) error {
|
2015-05-20 20:57:04 +00:00
|
|
|
// Read all bytes.
|
|
|
|
// b, err := ioutil.ReadAll(r)
|
|
|
|
// if err != nil {
|
|
|
|
// return err
|
|
|
|
// }
|
|
|
|
|
|
|
|
// TODO: Decode metadata.
|
|
|
|
// TODO: Set metadata on store.
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
type storeFSMSnapshot struct {
|
2015-05-20 20:57:04 +00:00
|
|
|
Data *Data
|
|
|
|
}
|
|
|
|
|
2015-05-21 20:06:01 +00:00
|
|
|
func (s *storeFSMSnapshot) Persist(sink raft.SnapshotSink) error {
|
2015-05-20 20:57:04 +00:00
|
|
|
// TODO: Encode data.
|
|
|
|
// TODO: sink.Write(p)
|
|
|
|
// TODO: sink.Close()
|
|
|
|
panic("not implemented yet")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release is invoked when we are finished with the snapshot
|
2015-05-21 20:06:01 +00:00
|
|
|
func (s *storeFSMSnapshot) Release() {}
|
2015-05-20 20:57:04 +00:00
|
|
|
|
2015-06-05 20:40:18 +00:00
|
|
|
// raftLayer wraps the connection so it can be re-used for forwarding.
|
|
|
|
type raftLayer struct {
|
2015-06-05 22:08:07 +00:00
|
|
|
ln net.Listener
|
2015-06-05 20:40:18 +00:00
|
|
|
addr net.Addr
|
|
|
|
conn chan net.Conn
|
|
|
|
closed chan struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// newRaftLayer returns a new instance of raftLayer.
|
2015-06-05 22:08:07 +00:00
|
|
|
func newRaftLayer(ln net.Listener, addr net.Addr) *raftLayer {
|
2015-06-05 20:40:18 +00:00
|
|
|
return &raftLayer{
|
2015-06-05 22:08:07 +00:00
|
|
|
ln: ln,
|
2015-06-05 20:40:18 +00:00
|
|
|
addr: addr,
|
|
|
|
conn: make(chan net.Conn),
|
|
|
|
closed: make(chan struct{}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Addr returns the local address for the layer.
|
|
|
|
func (l *raftLayer) Addr() net.Addr { return l.addr }
|
|
|
|
|
|
|
|
// Dial creates a new network connection.
|
|
|
|
func (l *raftLayer) Dial(addr string, timeout time.Duration) (net.Conn, error) {
|
|
|
|
conn, err := net.DialTimeout("tcp", addr, timeout)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write a marker byte for raft messages.
|
2015-06-05 22:08:07 +00:00
|
|
|
_, err = conn.Write([]byte{MuxRaftHeader})
|
2015-06-05 20:40:18 +00:00
|
|
|
if err != nil {
|
|
|
|
conn.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return conn, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Accept waits for the next connection.
|
2015-06-05 22:08:07 +00:00
|
|
|
func (l *raftLayer) Accept() (net.Conn, error) { return l.ln.Accept() }
|
2015-06-05 20:40:18 +00:00
|
|
|
|
|
|
|
// Close closes the layer.
|
2015-06-05 22:08:07 +00:00
|
|
|
func (l *raftLayer) Close() error { return l.ln.Close() }
|
2015-06-05 20:40:18 +00:00
|
|
|
|
2015-05-08 17:51:50 +00:00
|
|
|
// RetentionPolicyUpdate represents retention policy fields to be updated.
|
|
|
|
type RetentionPolicyUpdate struct {
|
|
|
|
Name *string
|
|
|
|
Duration *time.Duration
|
2015-05-21 20:06:01 +00:00
|
|
|
ReplicaN *int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (rpu *RetentionPolicyUpdate) SetName(v string) { rpu.Name = &v }
|
|
|
|
func (rpu *RetentionPolicyUpdate) SetDuration(v time.Duration) { rpu.Duration = &v }
|
|
|
|
func (rpu *RetentionPolicyUpdate) SetReplicaN(v int) { rpu.ReplicaN = &v }
|
|
|
|
|
|
|
|
// BcryptCost is the cost associated with generating password with Bcrypt.
|
|
|
|
// This setting is lowered during testing to improve test suite performance.
|
|
|
|
var BcryptCost = 10
|
|
|
|
|
|
|
|
// HashPassword generates a cryptographically secure hash for password.
|
|
|
|
// Returns an error if the password is invalid or a hash cannot be generated.
|
2015-06-05 20:40:18 +00:00
|
|
|
var HashPassword = func(password string) ([]byte, error) {
|
2015-05-21 20:06:01 +00:00
|
|
|
// The second arg is the cost of the hashing, higher is slower but makes
|
|
|
|
// it harder to brute force, since it will be really slow and impractical
|
|
|
|
return bcrypt.GenerateFromPassword([]byte(password), BcryptCost)
|
|
|
|
}
|
|
|
|
|
|
|
|
// assert will panic with a given formatted message if the given condition is false.
|
|
|
|
func assert(condition bool, msg string, v ...interface{}) {
|
|
|
|
if !condition {
|
|
|
|
panic(fmt.Sprintf("assert failed: "+msg, v...))
|
|
|
|
}
|
2015-05-08 17:51:50 +00:00
|
|
|
}
|