mirror of
https://github.com/status-im/consul.git
synced 2025-01-09 21:35:52 +00:00
Starting port of state store to MDB
This commit is contained in:
parent
3b47cce733
commit
85251683cc
@ -1,33 +1,18 @@
|
|||||||
package consul
|
package consul
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"database/sql"
|
"bytes"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"github.com/armon/gomdb"
|
||||||
"github.com/hashicorp/consul/rpc"
|
"github.com/hashicorp/consul/rpc"
|
||||||
_ "github.com/mattn/go-sqlite3"
|
"io/ioutil"
|
||||||
"log"
|
"os"
|
||||||
"sync/atomic"
|
|
||||||
"time"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// nextDBIndex is used to generate a new ID
|
|
||||||
// using sync/atomic to ensure it is safe
|
|
||||||
var nextDBIndex uint32 = 0
|
|
||||||
|
|
||||||
type namedQuery uint8
|
|
||||||
|
|
||||||
const (
|
const (
|
||||||
queryEnsureNode namedQuery = iota
|
dbNodes = "nodes" // Maps node -> addr
|
||||||
queryNode
|
dbServices = "services" // Maps node||serv -> rpc.NodeService
|
||||||
queryNodes
|
dbServiceIndex = "serviceIndex" // Maps serv||tag||node -> rpc.ServiceNode
|
||||||
queryEnsureService
|
|
||||||
queryNodeServices
|
|
||||||
queryDeleteNodeService
|
|
||||||
queryDeleteNode
|
|
||||||
queryServices
|
|
||||||
queryServiceNodes
|
|
||||||
queryServiceTagNodes
|
|
||||||
queryAllServices
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// The StateStore is responsible for maintaining all the Consul
|
// The StateStore is responsible for maintaining all the Consul
|
||||||
@ -35,34 +20,36 @@ const (
|
|||||||
// through the use of Raft. The goals of the StateStore are to provide
|
// through the use of Raft. The goals of the StateStore are to provide
|
||||||
// high concurrency for read operations without blocking writes, and
|
// high concurrency for read operations without blocking writes, and
|
||||||
// to provide write availability in the face of reads. The current
|
// to provide write availability in the face of reads. The current
|
||||||
// implementation uses an in-memory SQLite database. This reduced the
|
// implementation uses the Lightning Memory-Mapped Database (MDB).
|
||||||
// GC pressure on Go, and also gives us Multi-Version Concurrency Control
|
// This gives us Multi-Version Concurrency Control for "free"
|
||||||
// for "free".
|
|
||||||
type StateStore struct {
|
type StateStore struct {
|
||||||
db *sql.DB
|
path string
|
||||||
prepared map[namedQuery]*sql.Stmt
|
env *mdb.Env
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewStateStore is used to create a new state store
|
// NewStateStore is used to create a new state store
|
||||||
func NewStateStore() (*StateStore, error) {
|
func NewStateStore() (*StateStore, error) {
|
||||||
// Get the DB ID
|
// Create a new temp dir
|
||||||
id := atomic.AddUint32(&nextDBIndex, 1)
|
path, err := ioutil.TempDir("", "consul")
|
||||||
path := fmt.Sprintf("file:StateStore-%d?mode=memory&cache=shared", id)
|
|
||||||
|
|
||||||
// Open the db
|
|
||||||
db, err := sql.Open("sqlite3", path)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to open db: %v", err)
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Open the env
|
||||||
|
env, err := mdb.NewEnv()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
s := &StateStore{
|
s := &StateStore{
|
||||||
db: db,
|
path: path,
|
||||||
prepared: make(map[namedQuery]*sql.Stmt),
|
env: env,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure we can initialize
|
// Ensure we can initialize
|
||||||
if err := s.initialize(); err != nil {
|
if err := s.initialize(); err != nil {
|
||||||
db.Close()
|
env.Close()
|
||||||
|
os.RemoveAll(path)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return s, nil
|
return s, nil
|
||||||
@ -70,155 +57,249 @@ func NewStateStore() (*StateStore, error) {
|
|||||||
|
|
||||||
// Close is used to safely shutdown the state store
|
// Close is used to safely shutdown the state store
|
||||||
func (s *StateStore) Close() error {
|
func (s *StateStore) Close() error {
|
||||||
return s.db.Close()
|
s.env.Close()
|
||||||
|
os.RemoveAll(s.path)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// initialize is used to setup the sqlite store for use
|
// initialize is used to setup the store for use
|
||||||
func (s *StateStore) initialize() error {
|
func (s *StateStore) initialize() error {
|
||||||
// Set the pragma first
|
// Setup the Env first
|
||||||
pragmas := []string{
|
if err := s.env.SetMaxDBs(mdb.DBI(16)); err != nil {
|
||||||
"pragma journal_mode=memory;",
|
return err
|
||||||
"pragma foreign_keys=ON;",
|
|
||||||
"pragma read_uncommitted=true;",
|
|
||||||
}
|
|
||||||
for _, p := range pragmas {
|
|
||||||
if _, err := s.db.Exec(p); err != nil {
|
|
||||||
return fmt.Errorf("Failed to set '%s': %v", p, err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create the tables
|
// Optimize our flags for speed over safety, since the Raft log + snapshots
|
||||||
tables := []string{
|
// are durable. We treat this as an ephemeral in-memory DB, since we nuke
|
||||||
`CREATE TABLE nodes (name text unique, address text);`,
|
// the data anyways.
|
||||||
`CREATE TABLE services (node text REFERENCES nodes(name) ON DELETE CASCADE, service text, tag text, port integer);`,
|
var flags uint = mdb.NOMETASYNC | mdb.NOSYNC | mdb.NOTLS
|
||||||
`CREATE INDEX servName ON services(service, tag);`,
|
if err := s.env.Open(s.path, flags, 0755); err != nil {
|
||||||
`CREATE INDEX nodeName ON services(node);`,
|
return err
|
||||||
}
|
|
||||||
for _, t := range tables {
|
|
||||||
if _, err := s.db.Exec(t); err != nil {
|
|
||||||
return fmt.Errorf("Failed to call '%s': %v", t, err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prepare the queries
|
// Create all the tables
|
||||||
queries := map[namedQuery]string{
|
tx, _, err := s.startTxn(false, dbNodes, dbServices, dbServiceIndex)
|
||||||
queryEnsureNode: "INSERT OR REPLACE INTO nodes (name, address) VALUES (?, ?)",
|
if err != nil {
|
||||||
queryNode: "SELECT address FROM nodes where name=?",
|
tx.Abort()
|
||||||
queryNodes: "SELECT * FROM nodes",
|
return err
|
||||||
queryEnsureService: "INSERT OR REPLACE INTO services (node, service, tag, port) VALUES (?, ?, ?, ?)",
|
|
||||||
queryNodeServices: "SELECT service, tag, port from services where node=?",
|
|
||||||
queryDeleteNodeService: "DELETE FROM services WHERE node=? AND service=?",
|
|
||||||
queryDeleteNode: "DELETE FROM nodes WHERE name=?",
|
|
||||||
queryServices: "SELECT DISTINCT service, tag FROM services",
|
|
||||||
queryServiceNodes: "SELECT n.name, n.address, s.tag, s.port from nodes n, services s WHERE s.service=? AND s.node=n.name",
|
|
||||||
queryServiceTagNodes: "SELECT n.name, n.address, s.tag, s.port from nodes n, services s WHERE s.service=? AND s.tag=? AND s.node=n.name",
|
|
||||||
queryAllServices: "SELECT * FROM services",
|
|
||||||
}
|
}
|
||||||
for name, query := range queries {
|
return tx.Commit()
|
||||||
stmt, err := s.db.Prepare(query)
|
}
|
||||||
|
|
||||||
|
// startTxn is used to start a transaction and open all the associated sub-databases
|
||||||
|
func (s *StateStore) startTxn(readonly bool, open ...string) (*mdb.Txn, []mdb.DBI, error) {
|
||||||
|
var txFlags uint = 0
|
||||||
|
var dbFlags uint = 0
|
||||||
|
if readonly {
|
||||||
|
txFlags |= mdb.RDONLY
|
||||||
|
} else {
|
||||||
|
dbFlags |= mdb.CREATE
|
||||||
|
}
|
||||||
|
|
||||||
|
tx, err := s.env.BeginTxn(nil, txFlags)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var dbs []mdb.DBI
|
||||||
|
for _, name := range open {
|
||||||
|
dbi, err := tx.DBIOpen(name, dbFlags)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("Failed to prepare '%s': %v", query, err)
|
tx.Abort()
|
||||||
|
return nil, nil, err
|
||||||
}
|
}
|
||||||
s.prepared[name] = stmt
|
dbs = append(dbs, dbi)
|
||||||
}
|
}
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *StateStore) checkSet(res sql.Result, err error) error {
|
return tx, dbs, nil
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
n, err := res.RowsAffected()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if n != 1 {
|
|
||||||
return fmt.Errorf("Failed to set row")
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *StateStore) checkDelete(res sql.Result, err error) error {
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
_, err = res.RowsAffected()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnsureNode is used to ensure a given node exists, with the provided address
|
// EnsureNode is used to ensure a given node exists, with the provided address
|
||||||
func (s *StateStore) EnsureNode(name string, address string) error {
|
func (s *StateStore) EnsureNode(name string, address string) error {
|
||||||
stmt := s.prepared[queryEnsureNode]
|
tx, dbis, err := s.startTxn(false, dbNodes)
|
||||||
return s.checkSet(stmt.Exec(name, address))
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := tx.Put(dbis[0], []byte(name), []byte(address), 0); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return tx.Commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetNode returns all the address of the known and if it was found
|
// GetNode returns all the address of the known and if it was found
|
||||||
func (s *StateStore) GetNode(name string) (bool, string) {
|
func (s *StateStore) GetNode(name string) (bool, string) {
|
||||||
stmt := s.prepared[queryNode]
|
tx, dbis, err := s.startTxn(true, dbNodes)
|
||||||
row := stmt.QueryRow(name)
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node: %v", err))
|
||||||
var addr string
|
|
||||||
if err := row.Scan(&addr); err != nil {
|
|
||||||
if err == sql.ErrNoRows {
|
|
||||||
return false, addr
|
|
||||||
} else {
|
|
||||||
panic(fmt.Errorf("Failed to get node: %v", err))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return true, addr
|
defer tx.Abort()
|
||||||
|
|
||||||
|
val, err := tx.Get(dbis[0], []byte(name))
|
||||||
|
if err == mdb.NotFound {
|
||||||
|
return false, ""
|
||||||
|
} else if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
return true, string(val)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetNodes returns all the known nodes, the slice alternates between
|
// GetNodes returns all the known nodes, the slice alternates between
|
||||||
// the node name and address
|
// the node name and address
|
||||||
func (s *StateStore) Nodes() []string {
|
func (s *StateStore) Nodes() []string {
|
||||||
stmt := s.prepared[queryNodes]
|
tx, dbis, err := s.startTxn(true, dbNodes)
|
||||||
return parseNodes(stmt.Query())
|
|
||||||
}
|
|
||||||
|
|
||||||
// parseNodes parses the result of a queryNodes statement
|
|
||||||
func parseNodes(rows *sql.Rows, err error) []string {
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
||||||
}
|
}
|
||||||
data := make([]string, 0, 32)
|
defer tx.Abort()
|
||||||
var name, address string
|
|
||||||
for rows.Next() {
|
cursor, err := tx.CursorOpen(dbis[0])
|
||||||
if err := rows.Scan(&name, &address); err != nil {
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
var nodes []string
|
||||||
|
for {
|
||||||
|
key, val, err := cursor.Get(nil, mdb.NEXT)
|
||||||
|
if err == mdb.NotFound {
|
||||||
|
break
|
||||||
|
} else if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
||||||
}
|
}
|
||||||
data = append(data, name, address)
|
nodes = append(nodes, string(key), string(val))
|
||||||
}
|
}
|
||||||
return data
|
return nodes
|
||||||
}
|
}
|
||||||
|
|
||||||
// EnsureService is used to ensure a given node exposes a service
|
// EnsureService is used to ensure a given node exposes a service
|
||||||
func (s *StateStore) EnsureService(name, service, tag string, port int) error {
|
func (s *StateStore) EnsureService(name, service, tag string, port int) error {
|
||||||
stmt := s.prepared[queryEnsureService]
|
// Start a txn
|
||||||
return s.checkSet(stmt.Exec(name, service, tag, port))
|
tx, dbis, err := s.startTxn(false, dbNodes, dbServices, dbServiceIndex)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
nodes := dbis[0]
|
||||||
|
services := dbis[1]
|
||||||
|
index := dbis[2]
|
||||||
|
|
||||||
|
// Get the existing services
|
||||||
|
existing := filterNodeServices(tx, services, name)
|
||||||
|
|
||||||
|
// Get the node
|
||||||
|
addr, err := tx.Get(nodes, []byte(name))
|
||||||
|
if err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the service entry
|
||||||
|
key := []byte(fmt.Sprintf("%s||%s", name, service))
|
||||||
|
nService := rpc.NodeService{
|
||||||
|
Tag: tag,
|
||||||
|
Port: port,
|
||||||
|
}
|
||||||
|
val, err := rpc.Encode(255, &nService)
|
||||||
|
if err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := tx.Put(services, key, val, 0); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove previous entry if any
|
||||||
|
if exist, ok := existing[service]; ok {
|
||||||
|
key := []byte(fmt.Sprintf("%s||%s||%s", service, exist.Tag, name))
|
||||||
|
if err := tx.Del(index, key, nil); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the index entry
|
||||||
|
key = []byte(fmt.Sprintf("%s||%s||%s", service, tag, name))
|
||||||
|
node := rpc.ServiceNode{
|
||||||
|
Node: name,
|
||||||
|
Address: string(addr),
|
||||||
|
ServiceTag: tag,
|
||||||
|
ServicePort: port,
|
||||||
|
}
|
||||||
|
val, err = rpc.Encode(255, &node)
|
||||||
|
if err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := tx.Put(index, key, val, 0); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return tx.Commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// NodeServices is used to return all the services of a given node
|
// NodeServices is used to return all the services of a given node
|
||||||
func (s *StateStore) NodeServices(name string) rpc.NodeServices {
|
func (s *StateStore) NodeServices(name string) rpc.NodeServices {
|
||||||
stmt := s.prepared[queryNodeServices]
|
tx, dbis, err := s.startTxn(true, dbServices)
|
||||||
return parseNodeServices(stmt.Query(name))
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
defer tx.Abort()
|
||||||
|
return filterNodeServices(tx, dbis[0], name)
|
||||||
|
}
|
||||||
|
|
||||||
|
// filterNodeServices is used to filter the services to a specific node
|
||||||
|
func filterNodeServices(tx *mdb.Txn, services mdb.DBI, name string) rpc.NodeServices {
|
||||||
|
keyPrefix := []byte(fmt.Sprintf("%s||", name))
|
||||||
|
return parseNodeServices(tx, services, keyPrefix)
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseNodeServices is used to parse the results of a queryNodeServices
|
// parseNodeServices is used to parse the results of a queryNodeServices
|
||||||
func parseNodeServices(rows *sql.Rows, err error) rpc.NodeServices {
|
func parseNodeServices(tx *mdb.Txn, dbi mdb.DBI, prefix []byte) rpc.NodeServices {
|
||||||
|
// Create the cursor
|
||||||
|
cursor, err := tx.CursorOpen(dbi)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get node services: %v", err))
|
panic(fmt.Errorf("Failed to get nodes: %v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
services := rpc.NodeServices(make(map[string]rpc.NodeService))
|
services := rpc.NodeServices(make(map[string]rpc.NodeService))
|
||||||
var service string
|
var service string
|
||||||
var entry rpc.NodeService
|
var entry rpc.NodeService
|
||||||
for rows.Next() {
|
var key, val []byte
|
||||||
if err := rows.Scan(&service, &entry.Tag, &entry.Port); err != nil {
|
first := true
|
||||||
|
|
||||||
|
for {
|
||||||
|
if first {
|
||||||
|
first = false
|
||||||
|
key, val, err = cursor.Get(prefix, mdb.SET_RANGE)
|
||||||
|
} else {
|
||||||
|
key, val, err = cursor.Get(nil, mdb.NEXT)
|
||||||
|
}
|
||||||
|
if err == mdb.NotFound {
|
||||||
|
break
|
||||||
|
} else if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get node services: %v", err))
|
panic(fmt.Errorf("Failed to get node services: %v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Bail if this does not match our filter
|
||||||
|
if !bytes.HasPrefix(key, prefix) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// Split to get service name
|
||||||
|
parts := bytes.SplitN(key, []byte("||"), 2)
|
||||||
|
service = string(parts[1])
|
||||||
|
|
||||||
|
// Setup the entry
|
||||||
|
if val[0] != 255 {
|
||||||
|
panic(fmt.Errorf("Bad service value: %v", val))
|
||||||
|
}
|
||||||
|
if err := rpc.Decode(val[1:], &entry); err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node services: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add to the map
|
||||||
services[service] = entry
|
services[service] = entry
|
||||||
}
|
}
|
||||||
return services
|
return services
|
||||||
@ -226,62 +307,168 @@ func parseNodeServices(rows *sql.Rows, err error) rpc.NodeServices {
|
|||||||
|
|
||||||
// DeleteNodeService is used to delete a node service
|
// DeleteNodeService is used to delete a node service
|
||||||
func (s *StateStore) DeleteNodeService(node, service string) error {
|
func (s *StateStore) DeleteNodeService(node, service string) error {
|
||||||
stmt := s.prepared[queryDeleteNodeService]
|
tx, dbis, err := s.startTxn(false, dbServices, dbServiceIndex)
|
||||||
return s.checkDelete(stmt.Exec(node, service))
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
services := dbis[0]
|
||||||
|
index := dbis[1]
|
||||||
|
|
||||||
|
// Get the existing services
|
||||||
|
existing := filterNodeServices(tx, services, node)
|
||||||
|
exist, ok := existing[service]
|
||||||
|
|
||||||
|
// Bail if no existing entry
|
||||||
|
if !ok {
|
||||||
|
tx.Abort()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the node service entry
|
||||||
|
key := []byte(fmt.Sprintf("%s||%s", node, service))
|
||||||
|
if err = tx.Del(services, key, nil); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the sevice index entry
|
||||||
|
key = []byte(fmt.Sprintf("%s||%s||%s", service, exist.Tag, node))
|
||||||
|
if err := tx.Del(index, key, nil); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return tx.Commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteNode is used to delete a node and all it's services
|
// DeleteNode is used to delete a node and all it's services
|
||||||
func (s *StateStore) DeleteNode(node string) error {
|
func (s *StateStore) DeleteNode(node string) error {
|
||||||
stmt := s.prepared[queryDeleteNode]
|
tx, dbis, err := s.startTxn(false, dbNodes, dbServices, dbServiceIndex)
|
||||||
return s.checkDelete(stmt.Exec(node))
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
nodes := dbis[0]
|
||||||
|
services := dbis[1]
|
||||||
|
index := dbis[2]
|
||||||
|
|
||||||
|
// Delete the node
|
||||||
|
err = tx.Del(nodes, []byte(node), nil)
|
||||||
|
if err == mdb.NotFound {
|
||||||
|
err = nil
|
||||||
|
} else if err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the existing services
|
||||||
|
existing := filterNodeServices(tx, services, node)
|
||||||
|
|
||||||
|
// Nuke all the services
|
||||||
|
for service, entry := range existing {
|
||||||
|
// Delete the node service entry
|
||||||
|
key := []byte(fmt.Sprintf("%s||%s", node, service))
|
||||||
|
if err = tx.Del(services, key, nil); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete the sevice index entry
|
||||||
|
key = []byte(fmt.Sprintf("%s||%s||%s", service, entry.Tag, node))
|
||||||
|
if err := tx.Del(index, key, nil); err != nil {
|
||||||
|
tx.Abort()
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tx.Commit()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Services is used to return all the services with a list of associated tags
|
// Services is used to return all the services with a list of associated tags
|
||||||
func (s *StateStore) Services() map[string][]string {
|
func (s *StateStore) Services() map[string][]string {
|
||||||
stmt := s.prepared[queryServices]
|
tx, dbis, err := s.startTxn(false, dbServiceIndex)
|
||||||
rows, err := stmt.Query()
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
index := dbis[0]
|
||||||
|
|
||||||
|
cursor, err := tx.CursorOpen(index)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get services: %v", err))
|
panic(fmt.Errorf("Failed to get services: %v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
services := make(map[string][]string)
|
services := make(map[string][]string)
|
||||||
var service, tag string
|
for {
|
||||||
for rows.Next() {
|
key, _, err := cursor.Get(nil, mdb.NEXT)
|
||||||
if err := rows.Scan(&service, &tag); err != nil {
|
if err == mdb.NotFound {
|
||||||
|
break
|
||||||
|
} else if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get services: %v", err))
|
panic(fmt.Errorf("Failed to get services: %v", err))
|
||||||
}
|
}
|
||||||
|
parts := bytes.SplitN(key, []byte("||"), 3)
|
||||||
|
service := string(parts[0])
|
||||||
|
tag := string(parts[1])
|
||||||
|
|
||||||
tags := services[service]
|
tags := services[service]
|
||||||
tags = append(tags, tag)
|
if !strContains(tags, tag) {
|
||||||
services[service] = tags
|
tags = append(tags, tag)
|
||||||
|
services[service] = tags
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return services
|
return services
|
||||||
}
|
}
|
||||||
|
|
||||||
// ServiceNodes returns the nodes associated with a given service
|
// ServiceNodes returns the nodes associated with a given service
|
||||||
func (s *StateStore) ServiceNodes(service string) rpc.ServiceNodes {
|
func (s *StateStore) ServiceNodes(service string) rpc.ServiceNodes {
|
||||||
stmt := s.prepared[queryServiceNodes]
|
tx, dbis, err := s.startTxn(false, dbServiceIndex)
|
||||||
return parseServiceNodes(stmt.Query(service))
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
defer tx.Abort()
|
||||||
|
prefix := []byte(fmt.Sprintf("%s||", service))
|
||||||
|
return parseServiceNodes(tx, dbis[0], prefix)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ServiceTagNodes returns the nodes associated with a given service matching a tag
|
// ServiceTagNodes returns the nodes associated with a given service matching a tag
|
||||||
func (s *StateStore) ServiceTagNodes(service, tag string) rpc.ServiceNodes {
|
func (s *StateStore) ServiceTagNodes(service, tag string) rpc.ServiceNodes {
|
||||||
stmt := s.prepared[queryServiceTagNodes]
|
tx, dbis, err := s.startTxn(false, dbServiceIndex)
|
||||||
return parseServiceNodes(stmt.Query(service, tag))
|
if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node servicess: %v", err))
|
||||||
|
}
|
||||||
|
defer tx.Abort()
|
||||||
|
prefix := []byte(fmt.Sprintf("%s||%s||", service, tag))
|
||||||
|
return parseServiceNodes(tx, dbis[0], prefix)
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseServiceNodes parses results from the queryServiceNodes / queryServiceTagNodes query
|
// parseServiceNodes parses results ServiceNodes and ServiceTagNodes
|
||||||
func parseServiceNodes(rows *sql.Rows, err error) rpc.ServiceNodes {
|
func parseServiceNodes(tx *mdb.Txn, index mdb.DBI, prefix []byte) rpc.ServiceNodes {
|
||||||
|
cursor, err := tx.CursorOpen(index)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(fmt.Errorf("Failed to get service nodes: %v", err))
|
panic(fmt.Errorf("Failed to get node services: %v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
var nodes rpc.ServiceNodes
|
var nodes rpc.ServiceNodes
|
||||||
var node rpc.ServiceNode
|
var node rpc.ServiceNode
|
||||||
for rows.Next() {
|
for {
|
||||||
if err := rows.Scan(&node.Node, &node.Address, &node.ServiceTag, &node.ServicePort); err != nil {
|
key, val, err := cursor.Get(nil, mdb.NEXT)
|
||||||
panic(fmt.Errorf("Failed to get services: %v", err))
|
if err == mdb.NotFound {
|
||||||
|
break
|
||||||
|
} else if err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node services: %v", err))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Bail if this does not match our filter
|
||||||
|
if !bytes.HasPrefix(key, prefix) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup the node
|
||||||
|
if val[0] != 255 {
|
||||||
|
panic(fmt.Errorf("Bad service value: %v", val))
|
||||||
|
}
|
||||||
|
if err := rpc.Decode(val[1:], &node); err != nil {
|
||||||
|
panic(fmt.Errorf("Failed to get node services: %v", err))
|
||||||
|
}
|
||||||
|
|
||||||
nodes = append(nodes, node)
|
nodes = append(nodes, node)
|
||||||
}
|
}
|
||||||
return nodes
|
return nodes
|
||||||
@ -289,56 +476,5 @@ func parseServiceNodes(rows *sql.Rows, err error) rpc.ServiceNodes {
|
|||||||
|
|
||||||
// Snapshot is used to create a point in time snapshot
|
// Snapshot is used to create a point in time snapshot
|
||||||
func (s *StateStore) Snapshot() (*StateStore, error) {
|
func (s *StateStore) Snapshot() (*StateStore, error) {
|
||||||
defer func(start time.Time) {
|
return s, nil
|
||||||
log.Printf("[INFO] StateStore Snapshot created in %v", time.Now().Sub(start))
|
|
||||||
}(time.Now())
|
|
||||||
|
|
||||||
// Create a new state store
|
|
||||||
state, err := NewStateStore()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start a Tx on the new DB
|
|
||||||
tx, err := state.db.Begin()
|
|
||||||
if err != nil {
|
|
||||||
state.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create the new statements we need
|
|
||||||
ensureNode := tx.Stmt(state.prepared[queryEnsureNode])
|
|
||||||
ensureService := tx.Stmt(state.prepared[queryEnsureService])
|
|
||||||
|
|
||||||
// Copy all the nodes
|
|
||||||
nodes := s.Nodes()
|
|
||||||
for i := 0; i < len(nodes); i += 2 {
|
|
||||||
if _, err := ensureNode.Exec(nodes[i], nodes[i+1]); err != nil {
|
|
||||||
state.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy all the services
|
|
||||||
var node, service, tag string
|
|
||||||
var port int
|
|
||||||
rows, err := s.prepared[queryAllServices].Query()
|
|
||||||
for rows.Next() {
|
|
||||||
if err := rows.Scan(&node, &service, &tag, &port); err != nil {
|
|
||||||
state.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if _, err := ensureService.Exec(node, service, tag, port); err != nil {
|
|
||||||
state.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Commit the Txn
|
|
||||||
if err := tx.Commit(); err != nil {
|
|
||||||
state.Close()
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return state, nil
|
|
||||||
}
|
}
|
||||||
|
@ -50,7 +50,7 @@ func TestGetNodes(t *testing.T) {
|
|||||||
if len(nodes) != 4 {
|
if len(nodes) != 4 {
|
||||||
t.Fatalf("Bad: %v", nodes)
|
t.Fatalf("Bad: %v", nodes)
|
||||||
}
|
}
|
||||||
if nodes[0] != "foo" && nodes[2] != "bar" {
|
if nodes[2] != "foo" && nodes[0] != "bar" {
|
||||||
t.Fatalf("Bad: %v", nodes)
|
t.Fatalf("Bad: %v", nodes)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -63,19 +63,19 @@ func TestEnsureService(t *testing.T) {
|
|||||||
defer store.Close()
|
defer store.Close()
|
||||||
|
|
||||||
if err := store.EnsureNode("foo", "127.0.0.1"); err != nil {
|
if err := store.EnsureNode("foo", "127.0.0.1"); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := store.EnsureService("foo", "api", "", 5000); err != nil {
|
if err := store.EnsureService("foo", "api", "", 5000); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := store.EnsureService("foo", "api", "", 5001); err != nil {
|
if err := store.EnsureService("foo", "api", "", 5001); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := store.EnsureService("foo", "db", "master", 8000); err != nil {
|
if err := store.EnsureService("foo", "db", "master", 8000); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
services := store.NodeServices("foo")
|
services := store.NodeServices("foo")
|
||||||
@ -105,15 +105,15 @@ func TestDeleteNodeService(t *testing.T) {
|
|||||||
defer store.Close()
|
defer store.Close()
|
||||||
|
|
||||||
if err := store.EnsureNode("foo", "127.0.0.1"); err != nil {
|
if err := store.EnsureNode("foo", "127.0.0.1"); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := store.EnsureService("foo", "api", "", 5000); err != nil {
|
if err := store.EnsureService("foo", "api", "", 5000); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := store.DeleteNodeService("foo", "api"); err != nil {
|
if err := store.DeleteNodeService("foo", "api"); err != nil {
|
||||||
t.Fatalf("err: %v")
|
t.Fatalf("err: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
services := store.NodeServices("foo")
|
services := store.NodeServices("foo")
|
||||||
|
Loading…
x
Reference in New Issue
Block a user