2023-01-27 13:37:20 -08:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2021-11-22 21:47:30 -08:00
|
|
|
|
|
|
|
package clientmetric
|
|
|
|
|
|
|
|
import (
|
2024-09-25 17:20:56 +02:00
|
|
|
"expvar"
|
2021-11-22 21:47:30 -08:00
|
|
|
"testing"
|
|
|
|
"time"
|
2024-09-25 17:20:56 +02:00
|
|
|
|
|
|
|
qt "github.com/frankban/quicktest"
|
2021-11-22 21:47:30 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestDeltaEncBuf(t *testing.T) {
|
|
|
|
var enc deltaEncBuf
|
2022-05-11 16:09:27 -07:00
|
|
|
enc.writeName("one_one", TypeCounter)
|
2021-11-22 21:47:30 -08:00
|
|
|
enc.writeValue(1, 1)
|
2022-05-11 16:09:27 -07:00
|
|
|
enc.writeName("two_zero", TypeGauge)
|
2021-11-22 21:47:30 -08:00
|
|
|
enc.writeValue(2, 0)
|
|
|
|
|
|
|
|
enc.writeDelta(1, 63)
|
|
|
|
enc.writeDelta(2, 64)
|
|
|
|
enc.writeDelta(1, -65)
|
|
|
|
enc.writeDelta(2, -64)
|
|
|
|
|
|
|
|
got := enc.buf.String()
|
2022-05-11 16:09:27 -07:00
|
|
|
const want = "N0eone_oneS0202N1cgauge_two_zeroS0400I027eI048001I028101I047f"
|
2021-11-22 21:47:30 -08:00
|
|
|
if got != want {
|
|
|
|
t.Errorf("error\n got %q\nwant %q\n", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func clearMetrics() {
|
|
|
|
mu.Lock()
|
|
|
|
defer mu.Unlock()
|
|
|
|
metrics = map[string]*Metric{}
|
|
|
|
numWireID = 0
|
|
|
|
lastDelta = time.Time{}
|
|
|
|
sorted = nil
|
|
|
|
lastLogVal = nil
|
|
|
|
unsorted = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func advanceTime() {
|
|
|
|
mu.Lock()
|
|
|
|
defer mu.Unlock()
|
|
|
|
lastDelta = time.Time{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestEncodeLogTailMetricsDelta(t *testing.T) {
|
|
|
|
clearMetrics()
|
|
|
|
|
|
|
|
c1 := NewCounter("foo")
|
2022-05-11 16:09:27 -07:00
|
|
|
c2 := NewGauge("bar")
|
2021-11-22 21:47:30 -08:00
|
|
|
c1.Add(123)
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "N06fooS02f601"; got != want {
|
|
|
|
t.Errorf("first = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
c2.Add(456)
|
|
|
|
advanceTime()
|
2022-05-11 16:09:27 -07:00
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "N12gauge_barS049007"; got != want {
|
2021-11-22 21:47:30 -08:00
|
|
|
t.Errorf("second = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
advanceTime()
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), ""; got != want {
|
|
|
|
t.Errorf("with no changes = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
c1.Add(1)
|
|
|
|
c2.Add(2)
|
|
|
|
advanceTime()
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "I0202I0404"; got != want {
|
|
|
|
t.Errorf("with increments = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
}
|
2023-04-05 14:19:40 -07:00
|
|
|
|
|
|
|
func TestDisableDeltas(t *testing.T) {
|
|
|
|
clearMetrics()
|
|
|
|
|
|
|
|
c := NewCounter("foo")
|
|
|
|
c.DisableDeltas()
|
|
|
|
c.Set(123)
|
|
|
|
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "N06fooS02f601"; got != want {
|
|
|
|
t.Errorf("first = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Set(456)
|
|
|
|
advanceTime()
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "S029007"; got != want {
|
|
|
|
t.Errorf("second = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWithFunc(t *testing.T) {
|
|
|
|
clearMetrics()
|
|
|
|
|
|
|
|
v := int64(123)
|
|
|
|
NewCounterFunc("foo", func() int64 { return v })
|
|
|
|
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "N06fooS02f601"; got != want {
|
|
|
|
t.Errorf("first = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
|
|
|
|
v = 456
|
|
|
|
advanceTime()
|
|
|
|
if got, want := EncodeLogTailMetricsDelta(), "I029a05"; got != want {
|
|
|
|
t.Errorf("second = %q; want %q", got, want)
|
|
|
|
}
|
|
|
|
}
|
2024-09-25 17:20:56 +02:00
|
|
|
|
|
|
|
func TestAggregateCounter(t *testing.T) {
|
|
|
|
clearMetrics()
|
|
|
|
|
|
|
|
c := qt.New(t)
|
|
|
|
|
|
|
|
expv1 := &expvar.Int{}
|
|
|
|
expv2 := &expvar.Int{}
|
|
|
|
expv3 := &expvar.Int{}
|
|
|
|
|
|
|
|
aggCounter := NewAggregateCounter("agg_counter")
|
|
|
|
|
|
|
|
aggCounter.Register(expv1)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(0))
|
|
|
|
|
|
|
|
expv1.Add(1)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(1))
|
|
|
|
|
|
|
|
aggCounter.Register(expv2)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(1))
|
|
|
|
|
|
|
|
expv1.Add(1)
|
|
|
|
expv2.Add(1)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(3))
|
|
|
|
|
|
|
|
// Adding a new expvar should not change the value
|
|
|
|
// and any value the counter already had is reset
|
|
|
|
expv3.Set(5)
|
|
|
|
aggCounter.Register(expv3)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(3))
|
|
|
|
|
|
|
|
// Registering the same expvar multiple times should not change the value
|
|
|
|
aggCounter.Register(expv3)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(3))
|
|
|
|
|
|
|
|
aggCounter.UnregisterAll()
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(0))
|
|
|
|
|
|
|
|
// Start over
|
|
|
|
expv3.Set(5)
|
|
|
|
aggCounter.Register(expv3)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(0))
|
|
|
|
|
|
|
|
expv3.Set(5)
|
|
|
|
c.Assert(aggCounter.Value(), qt.Equals, int64(5))
|
|
|
|
}
|