control/controlclient: remove Client.SetStatusFunc
It can't change at runtime. Make it an option. Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
This commit is contained in:
parent
70a2797064
commit
ef0d740270
|
@ -6,6 +6,7 @@ package controlclient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
|
@ -46,17 +47,17 @@ var _ Client = (*Auto)(nil)
|
||||||
// Auto connects to a tailcontrol server for a node.
|
// Auto connects to a tailcontrol server for a node.
|
||||||
// It's a concrete implementation of the Client interface.
|
// It's a concrete implementation of the Client interface.
|
||||||
type Auto struct {
|
type Auto struct {
|
||||||
direct *Direct // our interface to the server APIs
|
direct *Direct // our interface to the server APIs
|
||||||
timeNow func() time.Time
|
timeNow func() time.Time
|
||||||
logf logger.Logf
|
logf logger.Logf
|
||||||
expiry *time.Time
|
expiry *time.Time
|
||||||
closed bool
|
closed bool
|
||||||
newMapCh chan struct{} // readable when we must restart a map request
|
newMapCh chan struct{} // readable when we must restart a map request
|
||||||
|
statusFunc func(Status) // called to update Client status; always non-nil
|
||||||
|
|
||||||
unregisterHealthWatch func()
|
unregisterHealthWatch func()
|
||||||
|
|
||||||
mu sync.Mutex // mutex guards the following fields
|
mu sync.Mutex // mutex guards the following fields
|
||||||
statusFunc func(Status) // called to update Client status
|
|
||||||
|
|
||||||
paused bool // whether we should stop making HTTP requests
|
paused bool // whether we should stop making HTTP requests
|
||||||
unpauseWaiters []chan struct{}
|
unpauseWaiters []chan struct{}
|
||||||
|
@ -92,6 +93,9 @@ func NewNoStart(opts Options) (*Auto, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
if opts.Status == nil {
|
||||||
|
return nil, errors.New("missing required Options.Status")
|
||||||
|
}
|
||||||
if opts.Logf == nil {
|
if opts.Logf == nil {
|
||||||
opts.Logf = func(fmt string, args ...any) {}
|
opts.Logf = func(fmt string, args ...any) {}
|
||||||
}
|
}
|
||||||
|
@ -99,13 +103,14 @@ func NewNoStart(opts Options) (*Auto, error) {
|
||||||
opts.TimeNow = time.Now
|
opts.TimeNow = time.Now
|
||||||
}
|
}
|
||||||
c := &Auto{
|
c := &Auto{
|
||||||
direct: direct,
|
direct: direct,
|
||||||
timeNow: opts.TimeNow,
|
timeNow: opts.TimeNow,
|
||||||
logf: opts.Logf,
|
logf: opts.Logf,
|
||||||
newMapCh: make(chan struct{}, 1),
|
newMapCh: make(chan struct{}, 1),
|
||||||
quit: make(chan struct{}),
|
quit: make(chan struct{}),
|
||||||
authDone: make(chan struct{}),
|
authDone: make(chan struct{}),
|
||||||
mapDone: make(chan struct{}),
|
mapDone: make(chan struct{}),
|
||||||
|
statusFunc: opts.Status,
|
||||||
}
|
}
|
||||||
c.authCtx, c.authCancel = context.WithCancel(context.Background())
|
c.authCtx, c.authCancel = context.WithCancel(context.Background())
|
||||||
c.mapCtx, c.mapCancel = context.WithCancel(context.Background())
|
c.mapCtx, c.mapCancel = context.WithCancel(context.Background())
|
||||||
|
@ -533,13 +538,6 @@ func (c *Auto) AuthCantContinue() bool {
|
||||||
return !c.loggedIn && (c.loginGoal == nil || c.loginGoal.url != "")
|
return !c.loggedIn && (c.loginGoal == nil || c.loginGoal.url != "")
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetStatusFunc sets fn as the callback to run on any status change.
|
|
||||||
func (c *Auto) SetStatusFunc(fn func(Status)) {
|
|
||||||
c.mu.Lock()
|
|
||||||
c.statusFunc = fn
|
|
||||||
c.mu.Unlock()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Auto) SetHostinfo(hi *tailcfg.Hostinfo) {
|
func (c *Auto) SetHostinfo(hi *tailcfg.Hostinfo) {
|
||||||
if hi == nil {
|
if hi == nil {
|
||||||
panic("nil Hostinfo")
|
panic("nil Hostinfo")
|
||||||
|
@ -567,10 +565,13 @@ func (c *Auto) SetNetInfo(ni *tailcfg.NetInfo) {
|
||||||
|
|
||||||
func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkMap) {
|
func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkMap) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
|
if c.closed {
|
||||||
|
c.mu.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
state := c.state
|
state := c.state
|
||||||
loggedIn := c.loggedIn
|
loggedIn := c.loggedIn
|
||||||
synced := c.synced
|
synced := c.synced
|
||||||
statusFunc := c.statusFunc
|
|
||||||
c.inSendStatus++
|
c.inSendStatus++
|
||||||
c.mu.Unlock()
|
c.mu.Unlock()
|
||||||
|
|
||||||
|
@ -601,9 +602,7 @@ func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkM
|
||||||
State: state,
|
State: state,
|
||||||
Err: err,
|
Err: err,
|
||||||
}
|
}
|
||||||
if statusFunc != nil {
|
c.statusFunc(new)
|
||||||
statusFunc(new)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
c.inSendStatus--
|
c.inSendStatus--
|
||||||
|
@ -684,7 +683,6 @@ func (c *Auto) Shutdown() {
|
||||||
direct := c.direct
|
direct := c.direct
|
||||||
if !closed {
|
if !closed {
|
||||||
c.closed = true
|
c.closed = true
|
||||||
c.statusFunc = nil
|
|
||||||
}
|
}
|
||||||
c.mu.Unlock()
|
c.mu.Unlock()
|
||||||
|
|
||||||
|
|
|
@ -28,9 +28,6 @@ const (
|
||||||
// Currently this is done through a pair of polling https requests in
|
// Currently this is done through a pair of polling https requests in
|
||||||
// the Auto client, but that might change eventually.
|
// the Auto client, but that might change eventually.
|
||||||
type Client interface {
|
type Client interface {
|
||||||
// SetStatusFunc provides a callback to call when control sends us
|
|
||||||
// a message.
|
|
||||||
SetStatusFunc(func(Status))
|
|
||||||
// Shutdown closes this session, which should not be used any further
|
// Shutdown closes this session, which should not be used any further
|
||||||
// afterwards.
|
// afterwards.
|
||||||
Shutdown()
|
Shutdown()
|
||||||
|
|
|
@ -108,6 +108,9 @@ type Options struct {
|
||||||
PopBrowserURL func(url string) // optional func to open browser
|
PopBrowserURL func(url string) // optional func to open browser
|
||||||
Dialer *tsdial.Dialer // non-nil
|
Dialer *tsdial.Dialer // non-nil
|
||||||
|
|
||||||
|
// Status is called when there's a change in status.
|
||||||
|
Status func(Status)
|
||||||
|
|
||||||
// KeepSharerAndUserSplit controls whether the client
|
// KeepSharerAndUserSplit controls whether the client
|
||||||
// understands Node.Sharer. If false, the Sharer is mapped to the User.
|
// understands Node.Sharer. If false, the Sharer is mapped to the User.
|
||||||
KeepSharerAndUserSplit bool
|
KeepSharerAndUserSplit bool
|
||||||
|
|
|
@ -1056,6 +1056,7 @@ func (b *LocalBackend) Start(opts ipn.Options) error {
|
||||||
Pinger: b,
|
Pinger: b,
|
||||||
PopBrowserURL: b.tellClientToBrowseToURL,
|
PopBrowserURL: b.tellClientToBrowseToURL,
|
||||||
Dialer: b.Dialer(),
|
Dialer: b.Dialer(),
|
||||||
|
Status: b.setClientStatus,
|
||||||
|
|
||||||
// Don't warn about broken Linux IP forwarding when
|
// Don't warn about broken Linux IP forwarding when
|
||||||
// netstack is being used.
|
// netstack is being used.
|
||||||
|
@ -1075,7 +1076,6 @@ func (b *LocalBackend) Start(opts ipn.Options) error {
|
||||||
cc.UpdateEndpoints(endpoints)
|
cc.UpdateEndpoints(endpoints)
|
||||||
}
|
}
|
||||||
|
|
||||||
cc.SetStatusFunc(b.setClientStatus)
|
|
||||||
b.e.SetNetInfoCallback(b.setNetInfo)
|
b.e.SetNetInfoCallback(b.setNetInfo)
|
||||||
|
|
||||||
b.mu.Lock()
|
b.mu.Lock()
|
||||||
|
|
|
@ -114,10 +114,6 @@ func (cc *mockControl) logf(format string, args ...any) {
|
||||||
cc.logfActual(format, args...)
|
cc.logfActual(format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cc *mockControl) SetStatusFunc(fn func(controlclient.Status)) {
|
|
||||||
cc.statusFunc = fn
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cc *mockControl) populateKeys() (newKeys bool) {
|
func (cc *mockControl) populateKeys() (newKeys bool) {
|
||||||
cc.mu.Lock()
|
cc.mu.Lock()
|
||||||
defer cc.mu.Unlock()
|
defer cc.mu.Unlock()
|
||||||
|
@ -295,12 +291,15 @@ func TestStateMachine(t *testing.T) {
|
||||||
}
|
}
|
||||||
t.Cleanup(e.Close)
|
t.Cleanup(e.Close)
|
||||||
|
|
||||||
cc := newMockControl(t)
|
|
||||||
t.Cleanup(func() { cc.preventLog.Set(true) }) // hacky way to pacify issue 3020
|
|
||||||
b, err := NewLocalBackend(logf, "logid", store, nil, e, 0)
|
b, err := NewLocalBackend(logf, "logid", store, nil, e, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("NewLocalBackend: %v", err)
|
t.Fatalf("NewLocalBackend: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cc := newMockControl(t)
|
||||||
|
cc.statusFunc = b.setClientStatus
|
||||||
|
t.Cleanup(func() { cc.preventLog.Set(true) }) // hacky way to pacify issue 3020
|
||||||
|
|
||||||
b.SetControlClientGetterForTesting(func(opts controlclient.Options) (controlclient.Client, error) {
|
b.SetControlClientGetterForTesting(func(opts controlclient.Options) (controlclient.Client, error) {
|
||||||
cc.mu.Lock()
|
cc.mu.Lock()
|
||||||
cc.opts = opts
|
cc.opts = opts
|
||||||
|
|
Loading…
Reference in New Issue