blob: e7b28a4662be1a7a94352981182217d7f46bfa43 [file] [log] [blame]
Robin Thellendac59e972014-08-19 18:26:11 -07001package stats_test
2
3import (
4 "reflect"
5 "testing"
6 "time"
7
Jiri Simsa519c5072014-09-17 21:37:57 -07008 libstats "veyron.io/veyron/veyron/lib/stats"
9 "veyron.io/veyron/veyron/lib/stats/counter"
10 "veyron.io/veyron/veyron/lib/stats/histogram"
11 istats "veyron.io/veyron/veyron/services/mgmt/stats"
Robin Thellendac59e972014-08-19 18:26:11 -070012)
13
Robin Thellend14a09ef2014-08-29 13:55:19 -070014func doGlob(root, pattern string, since time.Time, includeValues bool) ([]libstats.KeyValue, error) {
15 it := libstats.Glob(root, pattern, since, includeValues)
Robin Thellend7a442432014-08-22 09:05:08 -070016 out := []libstats.KeyValue{}
Robin Thellendac59e972014-08-19 18:26:11 -070017 for it.Advance() {
Robin Thellendfa70aaa2014-10-09 17:13:05 -070018 out = append(out, it.Value())
Robin Thellendac59e972014-08-19 18:26:11 -070019 }
20 if err := it.Err(); err != nil {
21 return nil, err
22 }
23 return out, nil
24}
25
26func TestStats(t *testing.T) {
Robin Thellendac59e972014-08-19 18:26:11 -070027 now := time.Unix(1, 0)
Jungho Ahn58e85b82014-12-01 10:01:02 -080028 counter.TimeNow = func() time.Time { return now }
Robin Thellendac59e972014-08-19 18:26:11 -070029
Robin Thellend7a442432014-08-22 09:05:08 -070030 a := libstats.NewInteger("ipc/test/aaa")
31 b := libstats.NewFloat("ipc/test/bbb")
32 c := libstats.NewString("ipc/test/ccc")
33 d := libstats.NewCounter("ipc/test/ddd")
Robin Thellendac59e972014-08-19 18:26:11 -070034
35 a.Set(1)
36 b.Set(2)
37 c.Set("Hello")
38 d.Set(4)
39
Robin Thellend7a442432014-08-22 09:05:08 -070040 got, err := libstats.Value("ipc/test/aaa")
Robin Thellendac59e972014-08-19 18:26:11 -070041 if err != nil {
42 t.Errorf("unexpected error: %v", err)
43 }
44 if expected := int64(1); got != expected {
45 t.Errorf("unexpected result. Got %v, want %v", got, expected)
46 }
47
Robin Thellend7a442432014-08-22 09:05:08 -070048 if _, err := libstats.Value(""); err != libstats.ErrNotFound {
Robin Thellendac59e972014-08-19 18:26:11 -070049 t.Errorf("expected error, got err=%v", err)
50 }
Robin Thellend7a442432014-08-22 09:05:08 -070051 if _, err := libstats.Value("does/not/exist"); err != libstats.ErrNotFound {
Robin Thellendac59e972014-08-19 18:26:11 -070052 t.Errorf("expected error, got err=%v", err)
53 }
54
Robin Thellend7a442432014-08-22 09:05:08 -070055 root := libstats.NewInteger("")
Robin Thellendac59e972014-08-19 18:26:11 -070056 root.Set(42)
Robin Thellend7a442432014-08-22 09:05:08 -070057 got, err = libstats.Value("")
Robin Thellendac59e972014-08-19 18:26:11 -070058 if err != nil {
59 t.Errorf("unexpected error: %v", err)
60 }
61 if expected := int64(42); got != expected {
62 t.Errorf("unexpected result. Got %v, want %v", got, expected)
63 }
64
Robin Thellend7a442432014-08-22 09:05:08 -070065 foo := libstats.NewInteger("foo")
Robin Thellendac59e972014-08-19 18:26:11 -070066 foo.Set(55)
Robin Thellend7a442432014-08-22 09:05:08 -070067 got, err = libstats.Value("foo")
Robin Thellendac59e972014-08-19 18:26:11 -070068 if err != nil {
69 t.Errorf("unexpected error: %v", err)
70 }
71 if expected := int64(55); got != expected {
72 t.Errorf("unexpected result. Got %v, want %v", got, expected)
73 }
74
Robin Thellend7a442432014-08-22 09:05:08 -070075 bar := libstats.NewInteger("foo/bar")
Robin Thellendac59e972014-08-19 18:26:11 -070076 bar.Set(44)
Robin Thellend7a442432014-08-22 09:05:08 -070077 got, err = libstats.Value("foo/bar")
Robin Thellendac59e972014-08-19 18:26:11 -070078 if err != nil {
79 t.Errorf("unexpected error: %v", err)
80 }
81 if expected := int64(44); got != expected {
82 t.Errorf("unexpected result. Got %v, want %v", got, expected)
83 }
84
Robin Thellend14a09ef2014-08-29 13:55:19 -070085 // Glob showing only nodes with a value.
86 result, err := doGlob("", "...", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -070087 if err != nil {
88 t.Errorf("unexpected error: %v", err)
89 }
Robin Thellend7a442432014-08-22 09:05:08 -070090 expected := []libstats.KeyValue{
91 libstats.KeyValue{Key: "", Value: int64(42)},
92 libstats.KeyValue{Key: "foo", Value: int64(55)},
93 libstats.KeyValue{Key: "foo/bar", Value: int64(44)},
94 libstats.KeyValue{Key: "ipc/test/aaa", Value: int64(1)},
95 libstats.KeyValue{Key: "ipc/test/bbb", Value: float64(2)},
96 libstats.KeyValue{Key: "ipc/test/ccc", Value: string("Hello")},
97 libstats.KeyValue{Key: "ipc/test/ddd", Value: int64(4)},
Robin Thellend1eb81692014-09-03 10:48:24 -070098 libstats.KeyValue{Key: "ipc/test/ddd/delta10m", Value: int64(4)},
99 libstats.KeyValue{Key: "ipc/test/ddd/delta1h", Value: int64(4)},
100 libstats.KeyValue{Key: "ipc/test/ddd/delta1m", Value: int64(4)},
Robin Thellend7a442432014-08-22 09:05:08 -0700101 libstats.KeyValue{Key: "ipc/test/ddd/rate10m", Value: float64(0)},
102 libstats.KeyValue{Key: "ipc/test/ddd/rate1h", Value: float64(0)},
103 libstats.KeyValue{Key: "ipc/test/ddd/rate1m", Value: float64(0)},
Robin Thellendac59e972014-08-19 18:26:11 -0700104 }
105 if !reflect.DeepEqual(result, expected) {
106 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
107 }
108
Robin Thellend14a09ef2014-08-29 13:55:19 -0700109 result, err = doGlob("", "ipc/test/*", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700110 if err != nil {
111 t.Errorf("unexpected error: %v", err)
112 }
Robin Thellend7a442432014-08-22 09:05:08 -0700113 expected = []libstats.KeyValue{
114 libstats.KeyValue{Key: "ipc/test/aaa", Value: int64(1)},
115 libstats.KeyValue{Key: "ipc/test/bbb", Value: float64(2)},
116 libstats.KeyValue{Key: "ipc/test/ccc", Value: string("Hello")},
117 libstats.KeyValue{Key: "ipc/test/ddd", Value: int64(4)},
Robin Thellendac59e972014-08-19 18:26:11 -0700118 }
119 if !reflect.DeepEqual(result, expected) {
120 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
121 }
122
Robin Thellend14a09ef2014-08-29 13:55:19 -0700123 // Glob showing all nodes without values
124 result, err = doGlob("", "ipc/...", time.Time{}, false)
125 if err != nil {
126 t.Errorf("unexpected error: %v", err)
127 }
128 expected = []libstats.KeyValue{
129 libstats.KeyValue{Key: "ipc"},
130 libstats.KeyValue{Key: "ipc/test"},
131 libstats.KeyValue{Key: "ipc/test/aaa"},
132 libstats.KeyValue{Key: "ipc/test/bbb"},
133 libstats.KeyValue{Key: "ipc/test/ccc"},
134 libstats.KeyValue{Key: "ipc/test/ddd"},
135 libstats.KeyValue{Key: "ipc/test/ddd/delta10m"},
136 libstats.KeyValue{Key: "ipc/test/ddd/delta1h"},
137 libstats.KeyValue{Key: "ipc/test/ddd/delta1m"},
138 libstats.KeyValue{Key: "ipc/test/ddd/rate10m"},
139 libstats.KeyValue{Key: "ipc/test/ddd/rate1h"},
140 libstats.KeyValue{Key: "ipc/test/ddd/rate1m"},
141 }
142 if !reflect.DeepEqual(result, expected) {
143 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
144 }
145
Robin Thellendac59e972014-08-19 18:26:11 -0700146 // Test the rate counter.
147 now = now.Add(10 * time.Second)
148 d.Incr(100)
Robin Thellend14a09ef2014-08-29 13:55:19 -0700149 result, err = doGlob("", "ipc/test/ddd/*", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700150 if err != nil {
151 t.Errorf("unexpected error: %v", err)
152 }
Robin Thellend7a442432014-08-22 09:05:08 -0700153 expected = []libstats.KeyValue{
Robin Thellend1eb81692014-09-03 10:48:24 -0700154 libstats.KeyValue{Key: "ipc/test/ddd/delta10m", Value: int64(104)},
155 libstats.KeyValue{Key: "ipc/test/ddd/delta1h", Value: int64(104)},
156 libstats.KeyValue{Key: "ipc/test/ddd/delta1m", Value: int64(104)},
157 libstats.KeyValue{Key: "ipc/test/ddd/rate10m", Value: float64(10.4)},
Robin Thellend7a442432014-08-22 09:05:08 -0700158 libstats.KeyValue{Key: "ipc/test/ddd/rate1h", Value: float64(0)},
Robin Thellend1eb81692014-09-03 10:48:24 -0700159 libstats.KeyValue{Key: "ipc/test/ddd/rate1m", Value: float64(10.4)},
Robin Thellendac59e972014-08-19 18:26:11 -0700160 }
161 if !reflect.DeepEqual(result, expected) {
162 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
163 }
164
165 // Test Glob on non-root object.
Robin Thellend14a09ef2014-08-29 13:55:19 -0700166 result, err = doGlob("ipc/test", "*", time.Time{}, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700167 if err != nil {
168 t.Errorf("unexpected error: %v", err)
169 }
Robin Thellend7a442432014-08-22 09:05:08 -0700170 expected = []libstats.KeyValue{
171 libstats.KeyValue{Key: "aaa", Value: int64(1)},
172 libstats.KeyValue{Key: "bbb", Value: float64(2)},
173 libstats.KeyValue{Key: "ccc", Value: string("Hello")},
174 libstats.KeyValue{Key: "ddd", Value: int64(104)},
Robin Thellendac59e972014-08-19 18:26:11 -0700175 }
176 if !reflect.DeepEqual(result, expected) {
177 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
178 }
179
Robin Thellend14a09ef2014-08-29 13:55:19 -0700180 result, err = doGlob("ipc/test/aaa", "", time.Time{}, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700181 if err != nil {
182 t.Errorf("unexpected error: %v", err)
183 }
Robin Thellend7a442432014-08-22 09:05:08 -0700184 expected = []libstats.KeyValue{
185 libstats.KeyValue{Key: "", Value: int64(1)},
Robin Thellendac59e972014-08-19 18:26:11 -0700186 }
187 if !reflect.DeepEqual(result, expected) {
188 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
189 }
190
191 // Test LastUpdate. The test only works on Counters.
Robin Thellend14a09ef2014-08-29 13:55:19 -0700192 result, err = doGlob("ipc/test", "ddd", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700193 if err != nil {
194 t.Errorf("unexpected error: %v", err)
195 }
Robin Thellend7a442432014-08-22 09:05:08 -0700196 expected = []libstats.KeyValue{
197 libstats.KeyValue{Key: "ddd", Value: int64(104)},
Robin Thellendac59e972014-08-19 18:26:11 -0700198 }
199 if !reflect.DeepEqual(result, expected) {
200 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
201 }
202
Robin Thellend14a09ef2014-08-29 13:55:19 -0700203 result, err = doGlob("ipc/test", "ddd", now.Add(time.Second), true)
Robin Thellendac59e972014-08-19 18:26:11 -0700204 if err != nil {
205 t.Errorf("unexpected error: %v", err)
206 }
Robin Thellend7a442432014-08-22 09:05:08 -0700207 expected = []libstats.KeyValue{}
Robin Thellendac59e972014-08-19 18:26:11 -0700208 if !reflect.DeepEqual(result, expected) {
209 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
210 }
211
212 // Test histogram
Robin Thellend7a442432014-08-22 09:05:08 -0700213 h := libstats.NewHistogram("ipc/test/hhh", histogram.Options{NumBuckets: 5, GrowthFactor: 0})
Robin Thellendac59e972014-08-19 18:26:11 -0700214 h.Add(1)
215 h.Add(2)
216
Robin Thellend14a09ef2014-08-29 13:55:19 -0700217 result, err = doGlob("", "ipc/test/hhh", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700218 if err != nil {
219 t.Errorf("unexpected error: %v", err)
220 }
Robin Thellend7a442432014-08-22 09:05:08 -0700221 expected = []libstats.KeyValue{
222 libstats.KeyValue{
Robin Thellendac59e972014-08-19 18:26:11 -0700223 Key: "ipc/test/hhh",
Robin Thellend7a442432014-08-22 09:05:08 -0700224 Value: istats.HistogramValue{
Robin Thellendac59e972014-08-19 18:26:11 -0700225 Count: 2,
226 Sum: 3,
Jungho Ahn58e85b82014-12-01 10:01:02 -0800227 Min: 1,
228 Max: 2,
Robin Thellend7a442432014-08-22 09:05:08 -0700229 Buckets: []istats.HistogramBucket{
230 istats.HistogramBucket{LowBound: 0, Count: 0},
231 istats.HistogramBucket{LowBound: 1, Count: 1},
232 istats.HistogramBucket{LowBound: 2, Count: 1},
233 istats.HistogramBucket{LowBound: 3, Count: 0},
234 istats.HistogramBucket{LowBound: 4, Count: 0},
Robin Thellendac59e972014-08-19 18:26:11 -0700235 },
236 },
237 },
238 }
239 if !reflect.DeepEqual(result, expected) {
240 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
241 }
242
243 now = now.Add(30 * time.Second)
244 h.Add(2)
245 now = now.Add(30 * time.Second)
246 h.Add(3)
247
Robin Thellend14a09ef2014-08-29 13:55:19 -0700248 result, err = doGlob("", "ipc/test/hhh/delta1m", now, true)
Robin Thellendac59e972014-08-19 18:26:11 -0700249 if err != nil {
250 t.Errorf("unexpected error: %v", err)
251 }
Robin Thellend7a442432014-08-22 09:05:08 -0700252 expected = []libstats.KeyValue{
253 libstats.KeyValue{
Robin Thellendac59e972014-08-19 18:26:11 -0700254 Key: "ipc/test/hhh/delta1m",
Robin Thellend7a442432014-08-22 09:05:08 -0700255 Value: istats.HistogramValue{
Robin Thellendac59e972014-08-19 18:26:11 -0700256 Count: 2,
257 Sum: 5,
Jungho Ahn58e85b82014-12-01 10:01:02 -0800258 Min: 2,
259 Max: 3,
Robin Thellend7a442432014-08-22 09:05:08 -0700260 Buckets: []istats.HistogramBucket{
261 istats.HistogramBucket{LowBound: 0, Count: 0},
262 istats.HistogramBucket{LowBound: 1, Count: 0},
263 istats.HistogramBucket{LowBound: 2, Count: 1},
264 istats.HistogramBucket{LowBound: 3, Count: 1},
265 istats.HistogramBucket{LowBound: 4, Count: 0},
Robin Thellendac59e972014-08-19 18:26:11 -0700266 },
267 },
268 },
269 }
270 if !reflect.DeepEqual(result, expected) {
271 t.Errorf("unexpected result. Got %#v, want %#v", result, expected)
272 }
273}
Robin Thellenddf428232014-10-06 12:50:44 -0700274
Robin Thellendfa70aaa2014-10-09 17:13:05 -0700275func TestMap(t *testing.T) {
276 m := libstats.NewMap("testing/foo")
277 m.Set([]libstats.KeyValue{{"a", 1}, {"b", 2}})
278
279 // Test the Value of the map.
280 {
281 got, err := libstats.Value("testing/foo")
282 if err != nil {
283 t.Errorf("unexpected error: %v", err)
284 }
285 if expected := interface{}(nil); got != expected {
286 t.Errorf("unexpected result. Got %v, want %v", got, expected)
287 }
288 }
289 // Test Glob on the map object.
290 {
291 got, err := doGlob("testing", "foo/...", time.Time{}, true)
292 if err != nil {
293 t.Errorf("unexpected error: %v", err)
294 }
295 expected := []libstats.KeyValue{
296 libstats.KeyValue{Key: "foo", Value: nil},
297 libstats.KeyValue{Key: "foo/a", Value: int64(1)},
298 libstats.KeyValue{Key: "foo/b", Value: int64(2)},
299 }
300 if !reflect.DeepEqual(got, expected) {
301 t.Errorf("unexpected result. Got %#v, want %#v", got, expected)
302 }
303 }
304
305 m.Delete([]string{"a"})
306
307 // Test Glob on the map object.
308 {
309 got, err := doGlob("testing", "foo/...", time.Time{}, true)
310 if err != nil {
311 t.Errorf("unexpected error: %v", err)
312 }
313 expected := []libstats.KeyValue{
314 libstats.KeyValue{Key: "foo", Value: nil},
315 libstats.KeyValue{Key: "foo/b", Value: int64(2)},
316 }
317 if !reflect.DeepEqual(got, expected) {
318 t.Errorf("unexpected result. Got %#v, want %#v", got, expected)
319 }
320 }
321}
322
Robin Thellend8cc0aee2014-10-16 10:58:24 -0700323func TestFunc(t *testing.T) {
324 libstats.NewIntegerFunc("testing/integer", func() int64 { return 123 })
325 libstats.NewFloatFunc("testing/float", func() float64 { return 456.789 })
326 libstats.NewStringFunc("testing/string", func() string { return "Hello World" })
327 ch := make(chan int64, 5)
328 libstats.NewIntegerFunc("testing/slowint", func() int64 {
329 return <-ch
330 })
331
332 testcases := []struct {
333 name string
334 expected interface{}
335 }{
336 {"testing/integer", int64(123)},
337 {"testing/float", float64(456.789)},
338 {"testing/string", "Hello World"},
339 {"testing/slowint", nil}, // Times out
340 }
341 for _, tc := range testcases {
342 checkVariable(t, tc.name, tc.expected)
343 }
344 checkVariable(t, "testing/slowint", nil) // Times out
345 checkVariable(t, "testing/slowint", nil) // Times out
346 ch <- int64(0)
347 checkVariable(t, "testing/slowint", int64(0)) // New value
348 checkVariable(t, "testing/slowint", int64(0)) // Times out
349 for i := 1; i <= 5; i++ {
350 ch <- int64(i)
351 }
352 for i := 1; i <= 5; i++ {
353 checkVariable(t, "testing/slowint", int64(i)) // New value each time
354 }
355}
356
357func checkVariable(t *testing.T, name string, expected interface{}) {
358 got, err := libstats.Value(name)
359 if err != nil {
360 t.Errorf("unexpected error for %q: %v", name, err)
361 }
362 if got != expected {
363 t.Errorf("unexpected result for %q. Got %v, want %v", name, got, expected)
364 }
365}
366
Robin Thellenddf428232014-10-06 12:50:44 -0700367func TestDelete(t *testing.T) {
368 _ = libstats.NewInteger("a/b/c/d")
369 if _, err := libstats.GetStatsObject("a/b/c/d"); err != nil {
370 t.Errorf("unexpected error value: %v", err)
371 }
372 if err := libstats.Delete("a/b/c/d"); err != nil {
373 t.Errorf("unexpected error value: %v", err)
374 }
375 if _, err := libstats.GetStatsObject("a/b/c/d"); err != libstats.ErrNotFound {
376 t.Errorf("unexpected error value: Got %v, want %v", err, libstats.ErrNotFound)
377 }
378 if err := libstats.Delete("a/b"); err != nil {
379 t.Errorf("unexpected error value: %v", err)
380 }
381 if _, err := libstats.GetStatsObject("a/b"); err != libstats.ErrNotFound {
382 t.Errorf("unexpected error value: Got %v, want %v", err, libstats.ErrNotFound)
383 }
384 if _, err := libstats.GetStatsObject("a/b/c"); err != libstats.ErrNotFound {
385 t.Errorf("unexpected error value: Got %v, want %v", err, libstats.ErrNotFound)
386 }
387}