types/logger: add key grinder stats lines to rate-limiting exemption list

Updates #1749

Co-authored-by: Brad Fitzpatrick <bradfitz@tailscale.com>
Signed-off-by: Josh Bleecher Snyder <josh@tailscale.com>
This commit is contained in:
Josh Bleecher Snyder 2021-05-04 20:36:10 -07:00 committed by Brad Fitzpatrick
parent f116a4c44f
commit 78d4c561b5
3 changed files with 42 additions and 8 deletions

View File

@ -15,6 +15,7 @@
"tailscale.com/tailcfg" "tailscale.com/tailcfg"
"tailscale.com/tstest" "tailscale.com/tstest"
"tailscale.com/types/key" "tailscale.com/types/key"
"tailscale.com/types/logger"
"tailscale.com/types/persist" "tailscale.com/types/persist"
"tailscale.com/wgengine" "tailscale.com/wgengine"
) )
@ -30,6 +31,12 @@ func TestLocalLogLines(t *testing.T) {
}) })
defer logListen.Close() defer logListen.Close()
// Put a rate-limiter with a burst of 0 between the components below.
// This instructs the rate-limiter to eliminate all logging that
// isn't explicitly exempt from rate-limiting.
// This lets the logListen tracker verify that the rate-limiter allows these key lines.
logf := logger.RateLimitedFnWithClock(logListen.Logf, 5*time.Second, 0, 10, time.Now)
logid := func(hex byte) logtail.PublicID { logid := func(hex byte) logtail.PublicID {
var ret logtail.PublicID var ret logtail.PublicID
for i := 0; i < len(ret); i++ { for i := 0; i < len(ret); i++ {
@ -41,12 +48,12 @@ func TestLocalLogLines(t *testing.T) {
// set up a LocalBackend, super bare bones. No functional data. // set up a LocalBackend, super bare bones. No functional data.
store := &ipn.MemoryStore{} store := &ipn.MemoryStore{}
e, err := wgengine.NewFakeUserspaceEngine(logListen.Logf, 0) e, err := wgengine.NewFakeUserspaceEngine(logf, 0)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
lb, err := NewLocalBackend(logListen.Logf, idA.String(), store, e) lb, err := NewLocalBackend(logf, idA.String(), store, e)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -61,6 +68,7 @@ func TestLocalLogLines(t *testing.T) {
testWantRemain := func(wantRemain ...string) func(t *testing.T) { testWantRemain := func(wantRemain ...string) func(t *testing.T) {
return func(t *testing.T) { return func(t *testing.T) {
if remain := logListen.Check(); !reflect.DeepEqual(remain, wantRemain) { if remain := logListen.Check(); !reflect.DeepEqual(remain, wantRemain) {
t.Helper()
t.Errorf("remain %q, want %q", remain, wantRemain) t.Errorf("remain %q, want %q", remain, wantRemain)
} }
} }
@ -75,17 +83,30 @@ func TestLocalLogLines(t *testing.T) {
t.Run("after_prefs", testWantRemain("[v1] peer keys: %s", "[v1] v%v peers: %v")) t.Run("after_prefs", testWantRemain("[v1] peer keys: %s", "[v1] v%v peers: %v"))
// log peers, peer keys // log peers, peer keys
status := &wgengine.Status{ lb.mu.Lock()
lb.parseWgStatusLocked(&wgengine.Status{
Peers: []ipnstate.PeerStatusLite{{ Peers: []ipnstate.PeerStatusLite{{
TxBytes: 10, TxBytes: 10,
RxBytes: 10, RxBytes: 10,
LastHandshake: time.Now(), LastHandshake: time.Now(),
NodeKey: tailcfg.NodeKey(key.NewPrivate()), NodeKey: tailcfg.NodeKey(key.NewPrivate()),
}}, }},
} })
lb.mu.Lock()
lb.parseWgStatusLocked(status)
lb.mu.Unlock() lb.mu.Unlock()
t.Run("after_peers", testWantRemain()) t.Run("after_peers", testWantRemain())
// Log it again with different stats to ensure it's not dup-suppressed.
logListen.Reset()
lb.mu.Lock()
lb.parseWgStatusLocked(&wgengine.Status{
Peers: []ipnstate.PeerStatusLite{{
TxBytes: 11,
RxBytes: 12,
LastHandshake: time.Now(),
NodeKey: tailcfg.NodeKey(key.NewPrivate()),
}},
})
lb.mu.Unlock()
t.Run("after_second_peer_status", testWantRemain("SetPrefs: %v"))
} }

View File

@ -107,6 +107,15 @@ func (lt *LogLineTracker) Check() []string {
return notSeen return notSeen
} }
// Reset forgets everything that it's seen.
func (lt *LogLineTracker) Reset() {
lt.mu.Lock()
defer lt.mu.Unlock()
for _, line := range lt.listenFor {
lt.seen[line] = false
}
}
// Close closes lt. After calling Close, calls to Logf become no-ops. // Close closes lt. After calling Close, calls to Logf become no-ops.
func (lt *LogLineTracker) Close() { func (lt *LogLineTracker) Close() {
lt.mu.Lock() lt.mu.Lock()

View File

@ -63,10 +63,15 @@ type limitData struct {
var disableRateLimit = os.Getenv("TS_DEBUG_LOG_RATE") == "all" var disableRateLimit = os.Getenv("TS_DEBUG_LOG_RATE") == "all"
// rateFree are format string substrings that are exempt from rate limiting. // rateFree are format string substrings that are exempt from rate limiting.
// Things should not be added to this unless they're already limited otherwise. // Things should not be added to this unless they're already limited otherwise
// or are critical for generating important stats from the logs.
var rateFree = []string{ var rateFree = []string{
"magicsock: disco: ", "magicsock: disco: ",
"magicsock: ParseEndpoint:", "magicsock: ParseEndpoint:",
// grinder stats lines
"SetPrefs: %v",
"peer keys: %s",
"v%v peers: %v",
} }
// RateLimitedFn is a wrapper for RateLimitedFnWithClock that includes the // RateLimitedFn is a wrapper for RateLimitedFnWithClock that includes the
@ -179,7 +184,6 @@ func LogOnChange(logf Logf, maxInterval time.Duration, timeNow func() time.Time)
// as it might contain formatting directives.) // as it might contain formatting directives.)
logf(format, args...) logf(format, args...)
} }
} }
// ArgWriter is a fmt.Formatter that can be passed to any Logf func to // ArgWriter is a fmt.Formatter that can be passed to any Logf func to