diff --git a/api/kv.go b/api/kv.go index d696f941da..3dac2583c1 100644 --- a/api/kv.go +++ b/api/kv.go @@ -35,6 +35,7 @@ const ( KVLock = "lock" KVUnlock = "unlock" KVGet = "get" + KVGetTree = "get-tree" KVCheckSession = "check-session" KVCheckIndex = "check-index" ) diff --git a/command/agent/txn_endpoint_test.go b/command/agent/txn_endpoint_test.go index 95de2e726e..18a6e16313 100644 --- a/command/agent/txn_endpoint_test.go +++ b/command/agent/txn_endpoint_test.go @@ -244,6 +244,12 @@ func TestTxnEndpoint_KV_Actions(t *testing.T) { "Verb": "get", "Key": "key" } + }, + { + "KV": { + "Verb": "get-tree", + "Key": "key" + } } ] `)) @@ -274,9 +280,6 @@ func TestTxnEndpoint_KV_Actions(t *testing.T) { if !ok { t.Fatalf("bad type: %T", obj) } - if len(txnResp.Results) != 1 { - t.Fatalf("bad: %v", txnResp) - } expected := structs.TxnReadResponse{ TxnResponse: structs.TxnResponse{ Results: structs.TxnResults{ @@ -293,6 +296,19 @@ func TestTxnEndpoint_KV_Actions(t *testing.T) { }, }, }, + &structs.TxnResult{ + KV: &structs.DirEntry{ + Key: "key", + Value: []byte("hello world"), + Flags: 23, + Session: id, + LockIndex: 1, + RaftIndex: structs.RaftIndex{ + CreateIndex: index, + ModifyIndex: index, + }, + }, + }, }, }, QueryMeta: structs.QueryMeta{ diff --git a/consul/kvs_endpoint.go b/consul/kvs_endpoint.go index ebe109d194..95ce7576ea 100644 --- a/consul/kvs_endpoint.go +++ b/consul/kvs_endpoint.go @@ -31,7 +31,7 @@ func kvsPreApply(srv *Server, acl acl.ACL, op structs.KVSOp, dirEnt *structs.Dir return false, permissionDeniedErr } - case structs.KVSGet: + case structs.KVSGet, structs.KVSGetTree: // Filtering for GETs is done on the output side. case structs.KVSCheckSession, structs.KVSCheckIndex: diff --git a/consul/state/kvs.go b/consul/state/kvs.go index 2ef9012bf1..3dccdebd31 100644 --- a/consul/state/kvs.go +++ b/consul/state/kvs.go @@ -151,6 +151,12 @@ func (s *StateStore) KVSList(prefix string) (uint64, structs.DirEntries, error) tx := s.db.Txn(false) defer tx.Abort() + return s.kvsListTxn(tx, prefix) +} + +// kvsListTxn is the inner method that gets a list of KVS entries matching a +// prefix. +func (s *StateStore) kvsListTxn(tx *memdb.Txn, prefix string) (uint64, structs.DirEntries, error) { // Get the table indexes. idx := maxIndexTxn(tx, "kvs", "tombstones") diff --git a/consul/state/txn.go b/consul/state/txn.go index 896b73f7f0..00d7905a2c 100644 --- a/consul/state/txn.go +++ b/consul/state/txn.go @@ -60,6 +60,18 @@ func (s *StateStore) txnKVS(tx *memdb.Txn, idx uint64, op *structs.TxnKVOp) (str err = fmt.Errorf("key %q doesn't exist", op.DirEnt.Key) } + case structs.KVSGetTree: + var entries structs.DirEntries + _, entries, err = s.kvsListTxn(tx, op.DirEnt.Key) + if err == nil { + results := make(structs.TxnResults, 0, len(entries)) + for _, e := range entries { + result := structs.TxnResult{KV: e} + results = append(results, &result) + } + return results, nil + } + case structs.KVSCheckSession: entry, err = s.kvsCheckSessionTxn(tx, op.DirEnt.Key, op.DirEnt.Session) diff --git a/consul/state/txn_test.go b/consul/state/txn_test.go index dda793211b..d868c13523 100644 --- a/consul/state/txn_test.go +++ b/consul/state/txn_test.go @@ -27,6 +27,14 @@ func TestStateStore_Txn_KVS(t *testing.T) { // Set up a transaction that hits every operation. ops := structs.TxnOps{ + &structs.TxnOp{ + KV: &structs.TxnKVOp{ + Verb: structs.KVSGetTree, + DirEnt: structs.DirEntry{ + Key: "foo/bar", + }, + }, + }, &structs.TxnOp{ KV: &structs.TxnKVOp{ Verb: structs.KVSSet, @@ -157,6 +165,26 @@ func TestStateStore_Txn_KVS(t *testing.T) { // Make sure the response looks as expected. expected := structs.TxnResults{ + &structs.TxnResult{ + KV: &structs.DirEntry{ + Key: "foo/bar/baz", + Value: []byte("baz"), + RaftIndex: structs.RaftIndex{ + CreateIndex: 2, + ModifyIndex: 2, + }, + }, + }, + &structs.TxnResult{ + KV: &structs.DirEntry{ + Key: "foo/bar/zip", + Value: []byte("zip"), + RaftIndex: structs.RaftIndex{ + CreateIndex: 3, + ModifyIndex: 3, + }, + }, + }, &structs.TxnResult{ KV: &structs.DirEntry{ Key: "foo/new", @@ -517,6 +545,14 @@ func TestStateStore_Txn_KVS_RO(t *testing.T) { // Set up a transaction that hits all the read-only operations. ops := structs.TxnOps{ + &structs.TxnOp{ + KV: &structs.TxnKVOp{ + Verb: structs.KVSGetTree, + DirEnt: structs.DirEntry{ + Key: "foo/bar", + }, + }, + }, &structs.TxnOp{ KV: &structs.TxnKVOp{ Verb: structs.KVSGet, @@ -550,12 +586,29 @@ func TestStateStore_Txn_KVS_RO(t *testing.T) { if len(errors) > 0 { t.Fatalf("err: %v", errors) } - if len(results) != len(ops) { - t.Fatalf("bad len: %d != %d", len(results), len(ops)) - } // Make sure the response looks as expected. expected := structs.TxnResults{ + &structs.TxnResult{ + KV: &structs.DirEntry{ + Key: "foo/bar/baz", + Value: []byte("baz"), + RaftIndex: structs.RaftIndex{ + CreateIndex: 2, + ModifyIndex: 2, + }, + }, + }, + &structs.TxnResult{ + KV: &structs.DirEntry{ + Key: "foo/bar/zip", + Value: []byte("zip"), + RaftIndex: structs.RaftIndex{ + CreateIndex: 3, + ModifyIndex: 3, + }, + }, + }, &structs.TxnResult{ KV: &structs.DirEntry{ Key: "foo", diff --git a/consul/structs/structs.go b/consul/structs/structs.go index f2f86604bb..08c0d8e5cd 100644 --- a/consul/structs/structs.go +++ b/consul/structs/structs.go @@ -538,6 +538,7 @@ const ( // The following operations are only available inside of atomic // transactions via the Txn request. KVSGet = "get" // Read the key during the transaction. + KVSGetTree = "get-tree" // Read all keys with the given prefix during the transaction. KVSCheckSession = "check-session" // Check the session holds the key. KVSCheckIndex = "check-index" // Check the modify index of the key. ) @@ -545,7 +546,7 @@ const ( // IsWrite returns true if the given operation alters the state store. func (op KVSOp) IsWrite() bool { switch op { - case KVSGet, KVSCheckSession, KVSCheckIndex: + case KVSGet, KVSGetTree, KVSCheckSession, KVSCheckIndex: return false default: diff --git a/consul/txn_endpoint_test.go b/consul/txn_endpoint_test.go index 7977184336..b1e60021c8 100644 --- a/consul/txn_endpoint_test.go +++ b/consul/txn_endpoint_test.go @@ -209,6 +209,14 @@ func TestTxn_Apply_ACLDeny(t *testing.T) { }, }, }, + &structs.TxnOp{ + KV: &structs.TxnKVOp{ + Verb: structs.KVSGetTree, + DirEnt: structs.DirEntry{ + Key: "nope", + }, + }, + }, &structs.TxnOp{ KV: &structs.TxnKVOp{ Verb: structs.KVSCheckSession, @@ -239,7 +247,7 @@ func TestTxn_Apply_ACLDeny(t *testing.T) { var expected structs.TxnResponse for i, op := range arg.Ops { switch op.KV.Verb { - case structs.KVSGet: + case structs.KVSGet, structs.KVSGetTree: // These get filtered but won't result in an error. default: @@ -455,6 +463,14 @@ func TestTxn_Read_ACLDeny(t *testing.T) { }, }, }, + &structs.TxnOp{ + KV: &structs.TxnKVOp{ + Verb: structs.KVSGetTree, + DirEnt: structs.DirEntry{ + Key: "nope", + }, + }, + }, &structs.TxnOp{ KV: &structs.TxnKVOp{ Verb: structs.KVSCheckSession, @@ -489,7 +505,7 @@ func TestTxn_Read_ACLDeny(t *testing.T) { } for i, op := range arg.Ops { switch op.KV.Verb { - case structs.KVSGet: + case structs.KVSGet, structs.KVSGetTree: // These get filtered but won't result in an error. default: diff --git a/website/source/docs/agent/http/kv.html.markdown b/website/source/docs/agent/http/kv.html.markdown index b65d3a6a55..55e02800d7 100644 --- a/website/source/docs/agent/http/kv.html.markdown +++ b/website/source/docs/agent/http/kv.html.markdown @@ -262,7 +262,7 @@ operation ("X" means a field is required and "O" means it is optional): unlock - Unlocks the `Key` with the given `Session`. The `Key` will only release the lock if the `Session` is valid and currently has it locked.. + Unlocks the `Key` with the given `Session`. The `Key` will only release the lock if the `Session` is valid and currently has it locked. X X O @@ -271,7 +271,16 @@ operation ("X" means a field is required and "O" means it is optional): get - Gets the `Key` during the transaction. This fails the transaction if the `Key` doesn't exist. + Gets the `Key` during the transaction. This fails the transaction if the `Key` doesn't exist. The key may not be present in the results if ACLs do not permit it to be read. + X + + + + + + + get-tree + Gets all keys with a prefix of `Key` during the transaction. This does not fail the transaction if the `Key` doesn't exist. Not all keys may be present in the results if ACLs do not permit them to be read. X @@ -355,9 +364,9 @@ back. If either of these status codes are returned, the response will look like ``` `Results` has entries for some operations if the transaction was successful. To save -space, the `Value` will be `null` for any `Verb` other than "get". Like the `/v1/kv/` -endpoint, `Value` will be Base64-encoded if it is present. Also, no result entries will be -added for verbs that delete keys. +space, the `Value` will be `null` for any `Verb` other than "get" or "get-tree". Like +the `/v1/kv/` endpoint, `Value` will be Base64-encoded if it is present. Also, +no result entries will be added for verbs that delete keys. `Errors` has entries describing which operations failed if the transaction was rolled back. The `OpIndex` gives the index of the failed operation in the transaction, and