File: watch_test.go

package info (click to toggle)
golang-etcd 0.2.0~rc1%2Bgit20140717-1
  • links: PTS, VCS
  • area: main
  • in suites: jessie, jessie-backports, jessie-kfreebsd
  • size: 188 kB
  • ctags: 165
  • sloc: makefile: 5
file content (119 lines) | stat: -rw-r--r-- 2,649 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
package etcd

import (
	"fmt"
	"runtime"
	"testing"
	"time"
)

func TestWatch(t *testing.T) {
	c := NewClient(nil)
	defer func() {
		c.Delete("watch_foo", true)
	}()

	go setHelper("watch_foo", "bar", c)

	resp, err := c.Watch("watch_foo", 0, false, nil, nil)
	if err != nil {
		t.Fatal(err)
	}
	if !(resp.Node.Key == "/watch_foo" && resp.Node.Value == "bar") {
		t.Fatalf("Watch 1 failed: %#v", resp)
	}

	go setHelper("watch_foo", "bar", c)

	resp, err = c.Watch("watch_foo", resp.Node.ModifiedIndex+1, false, nil, nil)
	if err != nil {
		t.Fatal(err)
	}
	if !(resp.Node.Key == "/watch_foo" && resp.Node.Value == "bar") {
		t.Fatalf("Watch 2 failed: %#v", resp)
	}

	routineNum := runtime.NumGoroutine()

	ch := make(chan *Response, 10)
	stop := make(chan bool, 1)

	go setLoop("watch_foo", "bar", c)

	go receiver(ch, stop)

	_, err = c.Watch("watch_foo", 0, false, ch, stop)
	if err != ErrWatchStoppedByUser {
		t.Fatalf("Watch returned a non-user stop error")
	}

	if newRoutineNum := runtime.NumGoroutine(); newRoutineNum != routineNum {
		t.Fatalf("Routine numbers differ after watch stop: %v, %v", routineNum, newRoutineNum)
	}
}

func TestWatchAll(t *testing.T) {
	c := NewClient(nil)
	defer func() {
		c.Delete("watch_foo", true)
	}()

	go setHelper("watch_foo/foo", "bar", c)

	resp, err := c.Watch("watch_foo", 0, true, nil, nil)
	if err != nil {
		t.Fatal(err)
	}
	if !(resp.Node.Key == "/watch_foo/foo" && resp.Node.Value == "bar") {
		t.Fatalf("WatchAll 1 failed: %#v", resp)
	}

	go setHelper("watch_foo/foo", "bar", c)

	resp, err = c.Watch("watch_foo", resp.Node.ModifiedIndex+1, true, nil, nil)
	if err != nil {
		t.Fatal(err)
	}
	if !(resp.Node.Key == "/watch_foo/foo" && resp.Node.Value == "bar") {
		t.Fatalf("WatchAll 2 failed: %#v", resp)
	}

	ch := make(chan *Response, 10)
	stop := make(chan bool, 1)

	routineNum := runtime.NumGoroutine()

	go setLoop("watch_foo/foo", "bar", c)

	go receiver(ch, stop)

	_, err = c.Watch("watch_foo", 0, true, ch, stop)
	if err != ErrWatchStoppedByUser {
		t.Fatalf("Watch returned a non-user stop error")
	}

	if newRoutineNum := runtime.NumGoroutine(); newRoutineNum != routineNum {
		t.Fatalf("Routine numbers differ after watch stop: %v, %v", routineNum, newRoutineNum)
	}
}

func setHelper(key, value string, c *Client) {
	time.Sleep(time.Second)
	c.Set(key, value, 100)
}

func setLoop(key, value string, c *Client) {
	time.Sleep(time.Second)
	for i := 0; i < 10; i++ {
		newValue := fmt.Sprintf("%s_%v", value, i)
		c.Set(key, newValue, 100)
		time.Sleep(time.Second / 10)
	}
}

func receiver(c chan *Response, stop chan bool) {
	for i := 0; i < 10; i++ {
		<-c
	}
	stop <- true
}