mirror of https://github.com/hashicorp/consul
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
376 lines
7.0 KiB
376 lines
7.0 KiB
package command |
|
|
|
import ( |
|
"bytes" |
|
"encoding/base64" |
|
"io" |
|
"io/ioutil" |
|
"os" |
|
"strconv" |
|
"strings" |
|
"testing" |
|
|
|
"github.com/hashicorp/consul/api" |
|
"github.com/hashicorp/consul/command/base" |
|
"github.com/mitchellh/cli" |
|
) |
|
|
|
func testKVPutCommand(t *testing.T) (*cli.MockUi, *KVPutCommand) { |
|
ui := new(cli.MockUi) |
|
return ui, &KVPutCommand{ |
|
Command: base.Command{ |
|
Ui: ui, |
|
Flags: base.FlagSetHTTP, |
|
}, |
|
} |
|
} |
|
|
|
func TestKVPutCommand_implements(t *testing.T) { |
|
var _ cli.Command = &KVPutCommand{} |
|
} |
|
|
|
func TestKVPutCommand_noTabs(t *testing.T) { |
|
assertNoTabs(t, new(KVDeleteCommand)) |
|
} |
|
|
|
func TestKVPutCommand_Validation(t *testing.T) { |
|
ui, c := testKVPutCommand(t) |
|
|
|
cases := map[string]struct { |
|
args []string |
|
output string |
|
}{ |
|
"-acquire without -session": { |
|
[]string{"-acquire", "foo"}, |
|
"Missing -session", |
|
}, |
|
"-release without -session": { |
|
[]string{"-release", "foo"}, |
|
"Missing -session", |
|
}, |
|
"-cas no -modify-index": { |
|
[]string{"-cas", "foo"}, |
|
"Must specify -modify-index", |
|
}, |
|
"no key": { |
|
[]string{}, |
|
"Missing KEY argument", |
|
}, |
|
"extra args": { |
|
[]string{"foo", "bar", "baz"}, |
|
"Too many arguments", |
|
}, |
|
} |
|
|
|
for name, tc := range cases { |
|
// Ensure our buffer is always clear |
|
if ui.ErrorWriter != nil { |
|
ui.ErrorWriter.Reset() |
|
} |
|
if ui.OutputWriter != nil { |
|
ui.OutputWriter.Reset() |
|
} |
|
|
|
code := c.Run(tc.args) |
|
if code == 0 { |
|
t.Errorf("%s: expected non-zero exit", name) |
|
} |
|
|
|
output := ui.ErrorWriter.String() |
|
if !strings.Contains(output, tc.output) { |
|
t.Errorf("%s: expected %q to contain %q", name, output, tc.output) |
|
} |
|
} |
|
} |
|
|
|
func TestKVPutCommand_Run(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"foo", "bar", |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte("bar")) { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_RunEmptyDataQuoted(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"foo", "", |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if data.Value != nil { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_RunBase64(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
const encodedString = "aGVsbG8gd29ybGQK" |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"-base64", |
|
"foo", encodedString, |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
expected, err := base64.StdEncoding.DecodeString(encodedString) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte(expected)) { |
|
t.Errorf("bad: %#v, %s", data.Value, data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_File(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
f, err := ioutil.TempFile("", "kv-put-command-file") |
|
if err != nil { |
|
t.Fatalf("err: %#v", err) |
|
} |
|
defer os.Remove(f.Name()) |
|
if _, err := f.WriteString("bar"); err != nil { |
|
t.Fatalf("err: %#v", err) |
|
} |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"foo", "@" + f.Name(), |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte("bar")) { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_FileNoExist(t *testing.T) { |
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"foo", "@/nope/definitely/not-a-real-file.txt", |
|
} |
|
|
|
code := c.Run(args) |
|
if code == 0 { |
|
t.Fatal("bad: expected error") |
|
} |
|
|
|
output := ui.ErrorWriter.String() |
|
if !strings.Contains(output, "Failed to read file") { |
|
t.Errorf("bad: %#v", output) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_Stdin(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
stdinR, stdinW := io.Pipe() |
|
|
|
ui, c := testKVPutCommand(t) |
|
c.testStdin = stdinR |
|
|
|
go func() { |
|
stdinW.Write([]byte("bar")) |
|
stdinW.Close() |
|
}() |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"foo", "-", |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte("bar")) { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_NegativeVal(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"foo", "-2", |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte("-2")) { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_Flags(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"-flags", "12345", |
|
"foo", |
|
} |
|
|
|
code := c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if data.Flags != 12345 { |
|
t.Errorf("bad: %#v", data.Flags) |
|
} |
|
} |
|
|
|
func TestKVPutCommand_CAS(t *testing.T) { |
|
srv, client := testAgentWithAPIClient(t) |
|
defer srv.Shutdown() |
|
waitForLeader(t, srv.httpAddr) |
|
|
|
// Create the initial pair so it has a ModifyIndex. |
|
pair := &api.KVPair{ |
|
Key: "foo", |
|
Value: []byte("bar"), |
|
} |
|
if _, err := client.KV().Put(pair, nil); err != nil { |
|
t.Fatalf("err: %#v", err) |
|
} |
|
|
|
ui, c := testKVPutCommand(t) |
|
|
|
args := []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"-cas", |
|
"-modify-index", "123", |
|
"foo", "a", |
|
} |
|
|
|
code := c.Run(args) |
|
if code == 0 { |
|
t.Fatalf("bad: expected error") |
|
} |
|
|
|
data, _, err := client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
// Reset buffers |
|
ui.OutputWriter.Reset() |
|
ui.ErrorWriter.Reset() |
|
|
|
args = []string{ |
|
"-http-addr=" + srv.httpAddr, |
|
"-cas", |
|
"-modify-index", strconv.FormatUint(data.ModifyIndex, 10), |
|
"foo", "a", |
|
} |
|
|
|
code = c.Run(args) |
|
if code != 0 { |
|
t.Fatalf("bad: %d. %#v", code, ui.ErrorWriter.String()) |
|
} |
|
|
|
data, _, err = client.KV().Get("foo", nil) |
|
if err != nil { |
|
t.Fatal(err) |
|
} |
|
|
|
if !bytes.Equal(data.Value, []byte("a")) { |
|
t.Errorf("bad: %#v", data.Value) |
|
} |
|
}
|
|
|