consul: First pass at KVS endpoints for RPC
This commit is contained in:
parent
a55ebaa353
commit
2067783692
|
@ -1,8 +1,103 @@
|
||||||
package consul
|
package consul
|
||||||
|
|
||||||
import ()
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/armon/go-metrics"
|
||||||
|
"github.com/hashicorp/consul/consul/structs"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
// KVS endpoint is used to manipulate the Key-Value store
|
// KVS endpoint is used to manipulate the Key-Value store
|
||||||
type KVS struct {
|
type KVS struct {
|
||||||
srv *Server
|
srv *Server
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Apply is used to apply a KVS request to the data store. This should
|
||||||
|
// only be used for operations that modify the data
|
||||||
|
func (c *Catalog) Apply(args *structs.KVSRequest, reply *bool) error {
|
||||||
|
if done, err := c.srv.forward("KVS.Apply", args.Datacenter, args, reply); done {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer metrics.MeasureSince([]string{"consul", "kvs", "apply"}, time.Now())
|
||||||
|
|
||||||
|
// Verify the args
|
||||||
|
if args.DirEnt.Key == "" {
|
||||||
|
return fmt.Errorf("Must provide key")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply the update
|
||||||
|
resp, err := c.srv.raftApply(structs.KVSRequestType, args)
|
||||||
|
if err != nil {
|
||||||
|
c.srv.logger.Printf("[ERR] consul.kvs: Apply failed: %v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if respErr, ok := resp.(error); ok {
|
||||||
|
return respErr
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the return type is a bool
|
||||||
|
if respBool, ok := resp.(bool); ok {
|
||||||
|
*reply = respBool
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get is used to lookup a single key
|
||||||
|
func (k *KVS) Get(args *structs.KeyRequest, reply *structs.IndexedDirEntries) error {
|
||||||
|
if done, err := k.srv.forward("KVS.Get", args.Datacenter, args, reply); done {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the local state
|
||||||
|
state := k.srv.fsm.State()
|
||||||
|
return k.srv.blockingRPC(&args.BlockingQuery,
|
||||||
|
state.QueryTables("KVSGet"),
|
||||||
|
func() (uint64, error) {
|
||||||
|
index, ent, err := state.KVSGet(args.Key)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
if ent == nil {
|
||||||
|
reply.Index = index
|
||||||
|
reply.Entries = nil
|
||||||
|
} else {
|
||||||
|
reply.Index = ent.ModifyIndex
|
||||||
|
reply.Entries = structs.DirEntries{ent}
|
||||||
|
}
|
||||||
|
return reply.Index, nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// List is used to list all keys with a given prefix
|
||||||
|
func (k *KVS) List(args *structs.KeyRequest, reply *structs.IndexedDirEntries) error {
|
||||||
|
if done, err := k.srv.forward("KVS.List", args.Datacenter, args, reply); done {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the local state
|
||||||
|
state := k.srv.fsm.State()
|
||||||
|
return k.srv.blockingRPC(&args.BlockingQuery,
|
||||||
|
state.QueryTables("KVSList"),
|
||||||
|
func() (uint64, error) {
|
||||||
|
index, ent, err := state.KVSList(args.Key)
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
if len(ent) == 0 {
|
||||||
|
reply.Index = index
|
||||||
|
reply.Entries = nil
|
||||||
|
} else {
|
||||||
|
// Determine the maximum affected index
|
||||||
|
var maxIndex uint64
|
||||||
|
for _, e := range ent {
|
||||||
|
if e.ModifyIndex > maxIndex {
|
||||||
|
maxIndex = e.ModifyIndex
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
reply.Index = maxIndex
|
||||||
|
reply.Entries = ent
|
||||||
|
}
|
||||||
|
return reply.Index, nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -231,6 +231,8 @@ func (s *StateStore) initialize() error {
|
||||||
"NodeChecks": MDBTables{s.checkTable},
|
"NodeChecks": MDBTables{s.checkTable},
|
||||||
"ServiceChecks": MDBTables{s.checkTable},
|
"ServiceChecks": MDBTables{s.checkTable},
|
||||||
"CheckServiceNodes": MDBTables{s.nodeTable, s.serviceTable, s.checkTable},
|
"CheckServiceNodes": MDBTables{s.nodeTable, s.serviceTable, s.checkTable},
|
||||||
|
"KVSGet": MDBTables{s.kvsTable},
|
||||||
|
"KVSList": MDBTables{s.kvsTable},
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -188,8 +188,6 @@ type KVSOp string
|
||||||
|
|
||||||
const (
|
const (
|
||||||
KVSSet KVSOp = "set"
|
KVSSet KVSOp = "set"
|
||||||
KVSGet = "get" // Key must match
|
|
||||||
KVSList = "list" // Key is only a prefix
|
|
||||||
KVSDelete = "delete"
|
KVSDelete = "delete"
|
||||||
KVSDeleteTree = "delete-tree"
|
KVSDeleteTree = "delete-tree"
|
||||||
KVSCAS = "cas" // Check-and-set
|
KVSCAS = "cas" // Check-and-set
|
||||||
|
@ -202,6 +200,18 @@ type KVSRequest struct {
|
||||||
DirEnt DirEntry // Which directory entry
|
DirEnt DirEntry // Which directory entry
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// KeyRequest is used to request a key, or key prefix
|
||||||
|
type KeyRequest struct {
|
||||||
|
Datacenter string
|
||||||
|
Key string
|
||||||
|
BlockingQuery
|
||||||
|
}
|
||||||
|
|
||||||
|
type IndexedDirEntries struct {
|
||||||
|
Index uint64
|
||||||
|
Entries DirEntries
|
||||||
|
}
|
||||||
|
|
||||||
// Decode is used to decode a MsgPack encoded object
|
// Decode is used to decode a MsgPack encoded object
|
||||||
func Decode(buf []byte, out interface{}) error {
|
func Decode(buf []byte, out interface{}) error {
|
||||||
var handle codec.MsgpackHandle
|
var handle codec.MsgpackHandle
|
||||||
|
|
Loading…
Reference in New Issue