2023-01-27 13:37:20 -08:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2021-03-01 21:34:41 -08:00
|
|
|
|
2023-04-18 14:26:58 -07:00
|
|
|
package netmon
|
2021-03-01 21:34:41 -08:00
|
|
|
|
|
|
|
import (
|
2021-03-02 10:29:49 -08:00
|
|
|
"flag"
|
2023-04-17 11:05:58 -07:00
|
|
|
"sync/atomic"
|
2021-03-01 21:34:41 -08:00
|
|
|
"testing"
|
2021-03-03 20:58:09 -08:00
|
|
|
"time"
|
2021-03-02 10:29:49 -08:00
|
|
|
|
|
|
|
"tailscale.com/net/interfaces"
|
2021-03-01 21:34:41 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMonitorStartClose(t *testing.T) {
|
|
|
|
mon, err := New(t.Logf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
mon.Start()
|
|
|
|
if err := mon.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMonitorJustClose(t *testing.T) {
|
|
|
|
mon, err := New(t.Logf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := mon.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
2021-03-02 10:29:49 -08:00
|
|
|
|
2021-03-03 20:58:09 -08:00
|
|
|
func TestMonitorInjectEvent(t *testing.T) {
|
|
|
|
mon, err := New(t.Logf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer mon.Close()
|
|
|
|
got := make(chan bool, 1)
|
|
|
|
mon.RegisterChangeCallback(func(changed bool, state *interfaces.State) {
|
|
|
|
select {
|
|
|
|
case got <- true:
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
})
|
|
|
|
mon.Start()
|
|
|
|
mon.InjectEvent()
|
|
|
|
select {
|
|
|
|
case <-got:
|
|
|
|
// Pass.
|
|
|
|
case <-time.After(5 * time.Second):
|
|
|
|
t.Fatal("timeout waiting for callback")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-17 11:05:58 -07:00
|
|
|
var (
|
|
|
|
monitor = flag.String("monitor", "", `go into monitor mode like 'route monitor'; test never terminates. Value can be either "raw" or "callback"`)
|
|
|
|
monitorDuration = flag.Duration("monitor-duration", 0, "if non-zero, how long to run TestMonitorMode. Zero means forever.")
|
|
|
|
)
|
2021-03-02 10:29:49 -08:00
|
|
|
|
|
|
|
func TestMonitorMode(t *testing.T) {
|
|
|
|
switch *monitor {
|
|
|
|
case "":
|
|
|
|
t.Skip("skipping non-test without --monitor")
|
|
|
|
case "raw", "callback":
|
|
|
|
default:
|
|
|
|
t.Skipf(`invalid --monitor value: must be "raw" or "callback"`)
|
|
|
|
}
|
|
|
|
mon, err := New(t.Logf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
switch *monitor {
|
|
|
|
case "raw":
|
2023-04-17 11:05:58 -07:00
|
|
|
var closed atomic.Bool
|
|
|
|
if *monitorDuration != 0 {
|
|
|
|
t := time.AfterFunc(*monitorDuration, func() {
|
|
|
|
closed.Store(true)
|
|
|
|
mon.Close()
|
|
|
|
})
|
|
|
|
defer t.Stop()
|
|
|
|
}
|
2021-03-02 10:29:49 -08:00
|
|
|
for {
|
|
|
|
msg, err := mon.om.Receive()
|
2023-04-17 11:05:58 -07:00
|
|
|
if closed.Load() {
|
|
|
|
return
|
|
|
|
}
|
2021-03-02 10:29:49 -08:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
t.Logf("msg: %#v", msg)
|
|
|
|
}
|
|
|
|
case "callback":
|
2023-04-17 11:05:58 -07:00
|
|
|
var done <-chan time.Time
|
|
|
|
if *monitorDuration != 0 {
|
|
|
|
t := time.NewTimer(*monitorDuration)
|
|
|
|
defer t.Stop()
|
|
|
|
done = t.C
|
|
|
|
}
|
|
|
|
n := 0
|
2021-03-02 10:29:49 -08:00
|
|
|
mon.RegisterChangeCallback(func(changed bool, st *interfaces.State) {
|
2023-04-17 11:05:58 -07:00
|
|
|
n++
|
2021-03-02 10:29:49 -08:00
|
|
|
t.Logf("cb: changed=%v, ifSt=%v", changed, st)
|
|
|
|
})
|
|
|
|
mon.Start()
|
2023-04-17 11:05:58 -07:00
|
|
|
<-done
|
|
|
|
t.Logf("%v callbacks", n)
|
2021-03-02 10:29:49 -08:00
|
|
|
}
|
|
|
|
}
|