mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
185 lines
5.5 KiB
Go
185 lines
5.5 KiB
Go
// Copyright 2016 The etcd Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package e2e
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
)
|
|
|
|
func TestCtlV3Watch(t *testing.T) { testCtl(t, watchTest) }
|
|
func TestCtlV3WatchNoTLS(t *testing.T) { testCtl(t, watchTest, withCfg(configNoTLS)) }
|
|
func TestCtlV3WatchClientTLS(t *testing.T) { testCtl(t, watchTest, withCfg(configClientTLS)) }
|
|
func TestCtlV3WatchPeerTLS(t *testing.T) { testCtl(t, watchTest, withCfg(configPeerTLS)) }
|
|
func TestCtlV3WatchTimeout(t *testing.T) { testCtl(t, watchTest, withDialTimeout(0)) }
|
|
|
|
func TestCtlV3WatchInteractive(t *testing.T) {
|
|
testCtl(t, watchTest, withInteractive())
|
|
}
|
|
func TestCtlV3WatchInteractiveNoTLS(t *testing.T) {
|
|
testCtl(t, watchTest, withInteractive(), withCfg(configNoTLS))
|
|
}
|
|
func TestCtlV3WatchInteractiveClientTLS(t *testing.T) {
|
|
testCtl(t, watchTest, withInteractive(), withCfg(configClientTLS))
|
|
}
|
|
func TestCtlV3WatchInteractivePeerTLS(t *testing.T) {
|
|
testCtl(t, watchTest, withInteractive(), withCfg(configPeerTLS))
|
|
}
|
|
|
|
type kvExec struct {
|
|
key, val string
|
|
execOutput string
|
|
}
|
|
|
|
func watchTest(cx ctlCtx) {
|
|
tests := []struct {
|
|
puts []kv
|
|
args []string
|
|
|
|
wkv []kvExec
|
|
}{
|
|
{ // watch 1 key
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"sample", "--rev", "1"},
|
|
[]kvExec{{key: "sample", val: "value"}},
|
|
},
|
|
{ // watch 1 key with "echo watch event received"
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"sample", "--rev", "1", "--", "echo", "watch event received"},
|
|
[]kvExec{{key: "sample", val: "value", execOutput: "watch event received"}},
|
|
},
|
|
{ // watch 1 key with "echo watch event received"
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"--rev", "1", "sample", "--", "echo", "watch event received"},
|
|
[]kvExec{{key: "sample", val: "value", execOutput: "watch event received"}},
|
|
},
|
|
{ // watch 1 key with "echo \"Hello World!\""
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"--rev", "1", "sample", "--", "echo", "\"Hello World!\""},
|
|
[]kvExec{{key: "sample", val: "value", execOutput: "Hello World!"}},
|
|
},
|
|
{ // watch 1 key with "echo watch event received"
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"sample", "samplx", "--rev", "1", "--", "echo", "watch event received"},
|
|
[]kvExec{{key: "sample", val: "value", execOutput: "watch event received"}},
|
|
},
|
|
{ // watch 1 key with "echo watch event received"
|
|
[]kv{{"sample", "value"}},
|
|
[]string{"sample", "--rev", "1", "samplx", "--", "echo", "watch event received"},
|
|
[]kvExec{{key: "sample", val: "value", execOutput: "watch event received"}},
|
|
},
|
|
{ // watch 3 keys by prefix
|
|
[]kv{{"key1", "val1"}, {"key2", "val2"}, {"key3", "val3"}},
|
|
[]string{"key", "--rev", "1", "--prefix"},
|
|
[]kvExec{{key: "key1", val: "val1"}, {key: "key2", val: "val2"}, {key: "key3", val: "val3"}},
|
|
},
|
|
{ // watch by revision
|
|
[]kv{{"etcd", "revision_1"}, {"etcd", "revision_2"}, {"etcd", "revision_3"}},
|
|
[]string{"etcd", "--rev", "2"},
|
|
[]kvExec{{key: "etcd", val: "revision_2"}, {key: "etcd", val: "revision_3"}},
|
|
},
|
|
{ // watch 3 keys by range
|
|
[]kv{{"key1", "val1"}, {"key3", "val3"}, {"key2", "val2"}},
|
|
[]string{"key", "key3", "--rev", "1"},
|
|
[]kvExec{{key: "key1", val: "val1"}, {key: "key2", val: "val2"}},
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
donec := make(chan struct{})
|
|
go func(i int, puts []kv) {
|
|
for j := range puts {
|
|
if err := ctlV3Put(cx, puts[j].key, puts[j].val, ""); err != nil {
|
|
cx.t.Fatalf("watchTest #%d-%d: ctlV3Put error (%v)", i, j, err)
|
|
}
|
|
}
|
|
close(donec)
|
|
}(i, tt.puts)
|
|
if err := ctlV3Watch(cx, tt.args, tt.wkv...); err != nil {
|
|
if cx.dialTimeout > 0 && !isGRPCTimedout(err) {
|
|
cx.t.Errorf("watchTest #%d: ctlV3Watch error (%v)", i, err)
|
|
}
|
|
}
|
|
<-donec
|
|
}
|
|
}
|
|
|
|
func setupWatchArgs(cx ctlCtx, args []string) []string {
|
|
cmdArgs := append(cx.PrefixArgs(), "watch")
|
|
if cx.interactive {
|
|
cmdArgs = append(cmdArgs, "--interactive")
|
|
} else {
|
|
cmdArgs = append(cmdArgs, args...)
|
|
}
|
|
|
|
return cmdArgs
|
|
}
|
|
|
|
func ctlV3Watch(cx ctlCtx, args []string, kvs ...kvExec) error {
|
|
cmdArgs := setupWatchArgs(cx, args)
|
|
|
|
proc, err := spawnCmd(cmdArgs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if cx.interactive {
|
|
wl := strings.Join(append([]string{"watch"}, args...), " ") + "\r"
|
|
if err = proc.Send(wl); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, elem := range kvs {
|
|
if _, err = proc.Expect(elem.key); err != nil {
|
|
return err
|
|
}
|
|
if _, err = proc.Expect(elem.val); err != nil {
|
|
return err
|
|
}
|
|
if elem.execOutput != "" {
|
|
if _, err = proc.Expect(elem.execOutput); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return proc.Stop()
|
|
}
|
|
|
|
func ctlV3WatchFailPerm(cx ctlCtx, args []string) error {
|
|
cmdArgs := setupWatchArgs(cx, args)
|
|
|
|
proc, err := spawnCmd(cmdArgs)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if cx.interactive {
|
|
wl := strings.Join(append([]string{"watch"}, args...), " ") + "\r"
|
|
if err = proc.Send(wl); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// TODO(mitake): after printing accurate error message that includes
|
|
// "permission denied", the above string argument of proc.Expect()
|
|
// should be updated.
|
|
_, err = proc.Expect("watch is canceled by the server")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return proc.Close()
|
|
}
|