| 
									
										
										
										
											2023-01-27 13:37:20 -08:00
										 |  |  | // Copyright (c) Tailscale Inc & AUTHORS | 
					
						
							|  |  |  | // SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | package controlclient | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2022-03-10 10:28:42 -08:00
										 |  |  | 	"net/http" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	"sync/atomic" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 13:40:32 +01:00
										 |  |  | 	"tailscale.com/health" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	"tailscale.com/logtail/backoff" | 
					
						
							| 
									
										
										
										
											2023-02-03 12:07:58 -08:00
										 |  |  | 	"tailscale.com/net/sockstats" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	"tailscale.com/tailcfg" | 
					
						
							| 
									
										
										
										
											2023-08-04 19:29:44 -04:00
										 |  |  | 	"tailscale.com/tstime" | 
					
						
							| 
									
										
										
										
											2021-10-28 14:22:51 -07:00
										 |  |  | 	"tailscale.com/types/key" | 
					
						
							| 
									
										
										
										
											2020-02-14 19:23:16 -08:00
										 |  |  | 	"tailscale.com/types/logger" | 
					
						
							| 
									
										
										
										
											2021-02-05 15:44:46 -08:00
										 |  |  | 	"tailscale.com/types/netmap" | 
					
						
							| 
									
										
										
										
											2021-02-05 15:23:01 -08:00
										 |  |  | 	"tailscale.com/types/persist" | 
					
						
							| 
									
										
										
										
											2020-05-03 13:58:39 -07:00
										 |  |  | 	"tailscale.com/types/structs" | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 	"tailscale.com/util/clientmetric" | 
					
						
							| 
									
										
										
										
											2024-01-18 10:33:20 -08:00
										 |  |  | 	"tailscale.com/util/execqueue" | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type LoginGoal struct { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	_     structs.Incomparable | 
					
						
							| 
									
										
										
										
											2024-05-17 14:54:41 -07:00
										 |  |  | 	flags LoginFlags // flags to use when logging in | 
					
						
							|  |  |  | 	url   string     // auth url that needs to be visited | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-19 15:06:33 -07:00
										 |  |  | var _ Client = (*Auto)(nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | // waitUnpause waits until either the client is unpaused or the Auto client is | 
					
						
							|  |  |  | // shut down. It reports whether the client should keep running (i.e. it's not | 
					
						
							|  |  |  | // closed). | 
					
						
							|  |  |  | func (c *Auto) waitUnpause(routineLogName string) (keepRunning bool) { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-09-05 12:12:08 -07:00
										 |  |  | 	if !c.paused || c.closed { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		defer c.mu.Unlock() | 
					
						
							|  |  |  | 		return !c.closed | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	unpaused := c.unpausedChanLocked() | 
					
						
							|  |  |  | 	c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.logf("%s: awaiting unpause", routineLogName) | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	return <-unpaused | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // updateRoutine is responsible for informing the server of worthy changes to | 
					
						
							|  |  |  | // our local state. It runs in its own goroutine. | 
					
						
							|  |  |  | func (c *Auto) updateRoutine() { | 
					
						
							|  |  |  | 	defer close(c.updateDone) | 
					
						
							|  |  |  | 	bo := backoff.NewBackoff("updateRoutine", c.logf, 30*time.Second) | 
					
						
							| 
									
										
										
										
											2023-08-13 08:31:15 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// lastUpdateGenInformed is the value of lastUpdateAt that we've successfully | 
					
						
							|  |  |  | 	// informed the server of. | 
					
						
							|  |  |  | 	var lastUpdateGenInformed updateGen | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		if !c.waitUnpause("updateRoutine") { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 			c.logf("updateRoutine: exiting") | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		c.mu.Lock() | 
					
						
							|  |  |  | 		gen := c.lastUpdateGen | 
					
						
							|  |  |  | 		ctx := c.mapCtx | 
					
						
							| 
									
										
										
										
											2023-08-13 08:31:15 -07:00
										 |  |  | 		needUpdate := gen > 0 && gen != lastUpdateGenInformed && c.loggedIn | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		if !needUpdate { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 			// Nothing to do, wait for a signal. | 
					
						
							|  |  |  | 			select { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			case <-ctx.Done(): | 
					
						
							|  |  |  | 				continue | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 			case <-c.updateCh: | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		t0 := c.clock.Now() | 
					
						
							|  |  |  | 		err := c.direct.SendUpdate(ctx) | 
					
						
							|  |  |  | 		d := time.Since(t0).Round(time.Millisecond) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			if ctx.Err() == nil { | 
					
						
							|  |  |  | 				c.direct.logf("lite map update error after %v: %v", d, err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			bo.BackOff(ctx, err) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		bo.BackOff(ctx, nil) | 
					
						
							|  |  |  | 		c.direct.logf("[v1] successful lite map update in %v", d) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-13 08:31:15 -07:00
										 |  |  | 		lastUpdateGenInformed = gen | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // atomicGen is an atomic int64 generator. It is used to generate monotonically | 
					
						
							|  |  |  | // increasing numbers for updateGen. | 
					
						
							|  |  |  | var atomicGen atomic.Int64 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func nextUpdateGen() updateGen { | 
					
						
							|  |  |  | 	return updateGen(atomicGen.Add(1)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // updateGen is a monotonically increasing number that represents a particular | 
					
						
							|  |  |  | // update to the local state. | 
					
						
							|  |  |  | type updateGen int64 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | // Auto connects to a tailcontrol server for a node. | 
					
						
							|  |  |  | // It's a concrete implementation of the Client interface. | 
					
						
							|  |  |  | type Auto struct { | 
					
						
							| 
									
										
										
										
											2023-09-01 09:56:31 -07:00
										 |  |  | 	direct        *Direct // our interface to the server APIs | 
					
						
							|  |  |  | 	clock         tstime.Clock | 
					
						
							|  |  |  | 	logf          logger.Logf | 
					
						
							|  |  |  | 	closed        bool | 
					
						
							|  |  |  | 	updateCh      chan struct{} // readable when we should inform the server of a change | 
					
						
							|  |  |  | 	observer      Observer      // called to update Client status; always non-nil | 
					
						
							| 
									
										
										
										
											2024-01-18 10:33:20 -08:00
										 |  |  | 	observerQueue execqueue.ExecQueue | 
					
						
							| 
									
										
										
										
											2025-03-12 10:37:03 -04:00
										 |  |  | 	shutdownFn    func() // to be called prior to shutdown or nil | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-18 08:58:13 -08:00
										 |  |  | 	unregisterHealthWatch func() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 	mu sync.Mutex // mutex guards the following fields | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	wantLoggedIn bool   // whether the user wants to be logged in per last method call | 
					
						
							|  |  |  | 	urlToVisit   string // the last url we were told to visit | 
					
						
							|  |  |  | 	expiry       time.Time | 
					
						
							| 
									
										
										
										
											2023-08-28 22:19:23 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	// lastUpdateGen is the gen of last update we had an update worth sending to | 
					
						
							|  |  |  | 	// the server. | 
					
						
							|  |  |  | 	lastUpdateGen updateGen | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 	lastStatus atomic.Pointer[Status] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	paused         bool        // whether we should stop making HTTP requests | 
					
						
							|  |  |  | 	unpauseWaiters []chan bool // chans that gets sent true (once) on wake, or false on Shutdown | 
					
						
							|  |  |  | 	loggedIn       bool        // true if currently logged in | 
					
						
							|  |  |  | 	loginGoal      *LoginGoal  // non-nil if some login activity is desired | 
					
						
							|  |  |  | 	inMapPoll      bool        // true once we get the first MapResponse in a stream; false when HTTP response ends | 
					
						
							|  |  |  | 	state          State       // TODO(bradfitz): delete this, make it computed by method from other state | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	authCtx    context.Context // context used for auth requests | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	mapCtx     context.Context // context used for netmap and update requests | 
					
						
							|  |  |  | 	authCancel func()          // cancel authCtx | 
					
						
							|  |  |  | 	mapCancel  func()          // cancel mapCtx | 
					
						
							|  |  |  | 	authDone   chan struct{}   // when closed, authRoutine is done | 
					
						
							|  |  |  | 	mapDone    chan struct{}   // when closed, mapRoutine is done | 
					
						
							|  |  |  | 	updateDone chan struct{}   // when closed, updateRoutine is done | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | // New creates and starts a new Auto. | 
					
						
							|  |  |  | func New(opts Options) (*Auto, error) { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c, err := NewNoStart(opts) | 
					
						
							|  |  |  | 	if c != nil { | 
					
						
							|  |  |  | 		c.Start() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return c, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | // NewNoStart creates a new Auto, but without calling Start on it. | 
					
						
							| 
									
										
										
										
											2022-09-27 00:37:27 +02:00
										 |  |  | func NewNoStart(opts Options) (_ *Auto, err error) { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	direct, err := NewDirect(opts) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-09-27 00:37:27 +02:00
										 |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			direct.Close() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 15:27:39 -07:00
										 |  |  | 	if opts.Observer == nil { | 
					
						
							|  |  |  | 		return nil, errors.New("missing required Options.Observer") | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-02-11 15:08:07 -05:00
										 |  |  | 	if opts.Logf == nil { | 
					
						
							| 
									
										
										
										
											2022-03-16 16:27:57 -07:00
										 |  |  | 		opts.Logf = func(fmt string, args ...any) {} | 
					
						
							| 
									
										
										
										
											2020-02-11 15:08:07 -05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-04 19:29:44 -04:00
										 |  |  | 	if opts.Clock == nil { | 
					
						
							|  |  |  | 		opts.Clock = tstime.StdClock{} | 
					
						
							| 
									
										
										
										
											2020-03-08 08:40:56 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | 	c := &Auto{ | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 		direct:     direct, | 
					
						
							| 
									
										
										
										
											2023-08-04 19:29:44 -04:00
										 |  |  | 		clock:      opts.Clock, | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 		logf:       opts.Logf, | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 		updateCh:   make(chan struct{}, 1), | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 		authDone:   make(chan struct{}), | 
					
						
							|  |  |  | 		mapDone:    make(chan struct{}), | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 		updateDone: make(chan struct{}), | 
					
						
							| 
									
										
										
										
											2023-08-28 15:27:39 -07:00
										 |  |  | 		observer:   opts.Observer, | 
					
						
							| 
									
										
										
										
											2025-03-12 10:37:03 -04:00
										 |  |  | 		shutdownFn: opts.Shutdown, | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	c.authCtx, c.authCancel = context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2023-04-12 18:23:22 -07:00
										 |  |  | 	c.authCtx = sockstats.WithSockStats(c.authCtx, sockstats.LabelControlClientAuto, opts.Logf) | 
					
						
							| 
									
										
										
										
											2023-02-03 12:07:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mapCtx, c.mapCancel = context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2023-04-12 18:23:22 -07:00
										 |  |  | 	c.mapCtx = sockstats.WithSockStats(c.mapCtx, sockstats.LabelControlClientAuto, opts.Logf) | 
					
						
							| 
									
										
										
										
											2023-02-03 12:07:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 13:40:32 +01:00
										 |  |  | 	c.unregisterHealthWatch = opts.HealthTracker.RegisterWatcher(func(c health.Change) { | 
					
						
							|  |  |  | 		if c.WarnableChanged { | 
					
						
							|  |  |  | 			direct.ReportWarnableChange(c.Warnable, c.UnhealthyState) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	return c, nil | 
					
						
							| 
									
										
										
										
											2021-02-18 08:58:13 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | // SetPaused controls whether HTTP activity should be paused. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The client can be paused and unpaused repeatedly, unlike Start and Shutdown, which can only be used once. | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) SetPaused(paused bool) { | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	defer c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2023-08-31 14:11:09 -07:00
										 |  |  | 	if paused == c.paused || c.closed { | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-30 06:08:26 -04:00
										 |  |  | 	c.logf("setPaused(%v)", paused) | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 	c.paused = paused | 
					
						
							|  |  |  | 	if paused { | 
					
						
							| 
									
										
										
										
											2023-08-31 14:11:09 -07:00
										 |  |  | 		c.cancelMapCtxLocked() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		c.cancelAuthCtxLocked() | 
					
						
							|  |  |  | 		return | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	for _, ch := range c.unpauseWaiters { | 
					
						
							|  |  |  | 		ch <- true | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	c.unpauseWaiters = nil | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | // Start starts the client's goroutines. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It should only be called for clients created by NewNoStart. | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) Start() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	go c.authRoutine() | 
					
						
							|  |  |  | 	go c.mapRoutine() | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	go c.updateRoutine() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | // updateControl sends a new OmitPeers, non-streaming map request (to just send | 
					
						
							|  |  |  | // Hostinfo/Netinfo/Endpoints info, while keeping an existing streaming response | 
					
						
							|  |  |  | // open). | 
					
						
							| 
									
										
										
										
											2020-12-23 13:03:16 -08:00
										 |  |  | // | 
					
						
							|  |  |  | // It should be called whenever there's something new to tell the server. | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | func (c *Auto) updateControl() { | 
					
						
							|  |  |  | 	gen := nextUpdateGen() | 
					
						
							| 
									
										
										
										
											2020-12-23 13:03:16 -08:00
										 |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	if gen < c.lastUpdateGen { | 
					
						
							|  |  |  | 		// This update is out of date. | 
					
						
							| 
									
										
										
										
											2020-12-23 13:03:16 -08:00
										 |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.lastUpdateGen = gen | 
					
						
							|  |  |  | 	c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2020-12-23 13:03:16 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	select { | 
					
						
							|  |  |  | 	case c.updateCh <- struct{}{}: | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2023-03-08 17:15:47 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-12-23 13:03:16 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-31 14:11:09 -07:00
										 |  |  | // cancelAuthCtxLocked is like cancelAuthCtx, but assumes the caller holds c.mu. | 
					
						
							|  |  |  | func (c *Auto) cancelAuthCtxLocked() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	if c.authCancel != nil { | 
					
						
							|  |  |  | 		c.authCancel() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !c.closed { | 
					
						
							|  |  |  | 		c.authCtx, c.authCancel = context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2023-04-12 18:23:22 -07:00
										 |  |  | 		c.authCtx = sockstats.WithSockStats(c.authCtx, sockstats.LabelControlClientAuto, c.logf) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-31 14:11:09 -07:00
										 |  |  | // cancelMapCtxLocked is like cancelMapCtx, but assumes the caller holds c.mu. | 
					
						
							|  |  |  | func (c *Auto) cancelMapCtxLocked() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	if c.mapCancel != nil { | 
					
						
							|  |  |  | 		c.mapCancel() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !c.closed { | 
					
						
							|  |  |  | 		c.mapCtx, c.mapCancel = context.WithCancel(context.Background()) | 
					
						
							| 
									
										
										
										
											2023-04-12 18:23:22 -07:00
										 |  |  | 		c.mapCtx = sockstats.WithSockStats(c.mapCtx, sockstats.LabelControlClientAuto, c.logf) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | // restartMap cancels the existing mapPoll and liteUpdates, and then starts a | 
					
						
							|  |  |  | // new one. | 
					
						
							|  |  |  | func (c *Auto) restartMap() { | 
					
						
							| 
									
										
										
										
											2023-08-11 18:49:40 -07:00
										 |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-31 14:11:09 -07:00
										 |  |  | 	c.cancelMapCtxLocked() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	synced := c.inMapPoll | 
					
						
							| 
									
										
										
										
											2023-08-11 18:49:40 -07:00
										 |  |  | 	c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2023-03-08 17:15:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-11 18:49:40 -07:00
										 |  |  | 	c.logf("[v1] restartMap: synced=%v", synced) | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.updateControl() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) authRoutine() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	defer close(c.authDone) | 
					
						
							| 
									
										
										
										
											2020-08-08 21:03:20 -07:00
										 |  |  | 	bo := backoff.NewBackoff("authRoutine", c.logf, 30*time.Second) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		if !c.waitUnpause("authRoutine") { | 
					
						
							|  |  |  | 			c.logf("authRoutine: exiting") | 
					
						
							|  |  |  | 			return | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 		c.mu.Lock() | 
					
						
							|  |  |  | 		goal := c.loginGoal | 
					
						
							|  |  |  | 		ctx := c.authCtx | 
					
						
							| 
									
										
										
										
											2020-12-14 10:10:01 -08:00
										 |  |  | 		if goal != nil { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			c.logf("[v1] authRoutine: %s; wantLoggedIn=%v", c.state, true) | 
					
						
							| 
									
										
										
										
											2020-12-14 10:10:01 -08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2022-02-12 08:05:24 -08:00
										 |  |  | 			c.logf("[v1] authRoutine: %s; goal=nil paused=%v", c.state, c.paused) | 
					
						
							| 
									
										
										
										
											2020-12-14 10:10:01 -08:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		report := func(err error, msg string) { | 
					
						
							| 
									
										
										
										
											2020-12-21 10:58:06 -08:00
										 |  |  | 			c.logf("[v1] %s: %v", msg, err) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			// don't send status updates for context errors, | 
					
						
							|  |  |  | 			// since context cancelation is always on purpose. | 
					
						
							|  |  |  | 			if ctx.Err() == nil { | 
					
						
							| 
									
										
										
										
											2020-10-13 15:03:56 -07:00
										 |  |  | 				c.sendStatus("authRoutine-report", err, "", nil) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if goal == nil { | 
					
						
							| 
									
										
										
										
											2024-04-26 09:49:30 -07:00
										 |  |  | 			c.direct.health.SetAuthRoutineInError(nil) | 
					
						
							| 
									
										
										
										
											2020-12-14 10:10:01 -08:00
										 |  |  | 			// Wait for user to Login or Logout. | 
					
						
							|  |  |  | 			<-ctx.Done() | 
					
						
							| 
									
										
										
										
											2020-12-21 10:58:06 -08:00
										 |  |  | 			c.logf("[v1] authRoutine: context done.") | 
					
						
							| 
									
										
										
										
											2020-12-14 10:10:01 -08:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		c.mu.Lock() | 
					
						
							|  |  |  | 		c.urlToVisit = goal.url | 
					
						
							|  |  |  | 		if goal.url != "" { | 
					
						
							|  |  |  | 			c.state = StateURLVisitRequired | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			c.state = StateAuthenticating | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		var url string | 
					
						
							|  |  |  | 		var err error | 
					
						
							|  |  |  | 		var f string | 
					
						
							|  |  |  | 		if goal.url != "" { | 
					
						
							|  |  |  | 			url, err = c.direct.WaitLoginURL(ctx, goal.url) | 
					
						
							|  |  |  | 			f = "WaitLoginURL" | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2024-05-17 14:54:41 -07:00
										 |  |  | 			url, err = c.direct.TryLogin(ctx, goal.flags) | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			f = "TryLogin" | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2024-04-26 09:49:30 -07:00
										 |  |  | 			c.direct.health.SetAuthRoutineInError(err) | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			report(err, f) | 
					
						
							|  |  |  | 			bo.BackOff(ctx, err) | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if url != "" { | 
					
						
							| 
									
										
										
										
											2024-05-06 21:33:37 -07:00
										 |  |  | 			// goal.url ought to be empty here. However, not all control servers | 
					
						
							|  |  |  | 			// get this right, and logging about it here just generates noise. | 
					
						
							|  |  |  | 			// | 
					
						
							|  |  |  | 			// TODO(bradfitz): I don't follow that comment. Our own testcontrol | 
					
						
							|  |  |  | 			// used by tstest/integration hits this path, in fact. | 
					
						
							|  |  |  | 			if c.direct.panicOnUse { | 
					
						
							|  |  |  | 				panic("tainted client") | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			c.urlToVisit = url | 
					
						
							|  |  |  | 			c.loginGoal = &LoginGoal{ | 
					
						
							|  |  |  | 				flags: LoginDefault, | 
					
						
							|  |  |  | 				url:   url, | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			c.state = StateURLVisitRequired | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			c.sendStatus("authRoutine-url", err, url, nil) | 
					
						
							|  |  |  | 			if goal.url == url { | 
					
						
							|  |  |  | 				// The server sent us the same URL we already tried, | 
					
						
							|  |  |  | 				// backoff to avoid a busy loop. | 
					
						
							|  |  |  | 				bo.BackOff(ctx, errors.New("login URL not changing")) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 				bo.BackOff(ctx, nil) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		// success | 
					
						
							| 
									
										
										
										
											2024-04-26 09:49:30 -07:00
										 |  |  | 		c.direct.health.SetAuthRoutineInError(nil) | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		c.mu.Lock() | 
					
						
							|  |  |  | 		c.urlToVisit = "" | 
					
						
							|  |  |  | 		c.loggedIn = true | 
					
						
							|  |  |  | 		c.loginGoal = nil | 
					
						
							|  |  |  | 		c.state = StateAuthenticated | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		c.sendStatus("authRoutine-success", nil, "", nil) | 
					
						
							|  |  |  | 		c.restartMap() | 
					
						
							|  |  |  | 		bo.BackOff(ctx, nil) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:19:23 -07:00
										 |  |  | // ExpiryForTests returns the credential expiration time, or the zero value if | 
					
						
							|  |  |  | // the expiration time isn't known. It's used in tests only. | 
					
						
							|  |  |  | func (c *Auto) ExpiryForTests() time.Time { | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	defer c.mu.Unlock() | 
					
						
							|  |  |  | 	return c.expiry | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:19:23 -07:00
										 |  |  | // DirectForTest returns the underlying direct client object. | 
					
						
							|  |  |  | // It's used in tests only. | 
					
						
							|  |  |  | func (c *Auto) DirectForTest() *Direct { | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | 	return c.direct | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | // unpausedChanLocked returns a new channel that gets sent | 
					
						
							|  |  |  | // either a true when unpaused or false on Auto.Shutdown. | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | // | 
					
						
							|  |  |  | // c.mu must be held | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | func (c *Auto) unpausedChanLocked() <-chan bool { | 
					
						
							|  |  |  | 	unpaused := make(chan bool, 1) | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 	c.unpauseWaiters = append(c.unpauseWaiters, unpaused) | 
					
						
							|  |  |  | 	return unpaused | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | // mapRoutineState is the state of Auto.mapRoutine while it's running. | 
					
						
							|  |  |  | type mapRoutineState struct { | 
					
						
							|  |  |  | 	c  *Auto | 
					
						
							|  |  |  | 	bo *backoff.Backoff | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-01 19:28:00 -07:00
										 |  |  | var _ NetmapDeltaUpdater = mapRoutineState{} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | func (mrs mapRoutineState) UpdateFullNetmap(nm *netmap.NetworkMap) { | 
					
						
							|  |  |  | 	c := mrs.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	ctx := c.mapCtx | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	c.inMapPoll = true | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | 	if c.loggedIn { | 
					
						
							|  |  |  | 		c.state = StateSynchronized | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-28 22:19:23 -07:00
										 |  |  | 	c.expiry = nm.Expiry | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | 	stillAuthed := c.loggedIn | 
					
						
							|  |  |  | 	c.logf("[v1] mapRoutine: netmap received: %s", c.state) | 
					
						
							|  |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if stillAuthed { | 
					
						
							|  |  |  | 		c.sendStatus("mapRoutine-got-netmap", nil, "", nm) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// Reset the backoff timer if we got a netmap. | 
					
						
							|  |  |  | 	mrs.bo.BackOff(ctx, nil) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-01 19:28:00 -07:00
										 |  |  | func (mrs mapRoutineState) UpdateNetmapDelta(muts []netmap.NodeMutation) bool { | 
					
						
							|  |  |  | 	c := mrs.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	goodState := c.loggedIn && c.inMapPoll | 
					
						
							|  |  |  | 	ndu, canDelta := c.observer.(NetmapDeltaUpdater) | 
					
						
							|  |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if !goodState || !canDelta { | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ctx, cancel := context.WithTimeout(c.mapCtx, 2*time.Second) | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var ok bool | 
					
						
							|  |  |  | 	err := c.observerQueue.RunSync(ctx, func() { | 
					
						
							|  |  |  | 		ok = ndu.UpdateNetmapDelta(muts) | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	return err == nil && ok | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | // mapRoutine is responsible for keeping a read-only streaming connection to the | 
					
						
							|  |  |  | // control server, and keeping the netmap up to date. | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) mapRoutine() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	defer close(c.mapDone) | 
					
						
							| 
									
										
										
										
											2023-11-05 07:31:06 -08:00
										 |  |  | 	mrs := mapRoutineState{ | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | 		c:  c, | 
					
						
							|  |  |  | 		bo: backoff.NewBackoff("mapRoutine", c.logf, 30*time.Second), | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		if !c.waitUnpause("mapRoutine") { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 			c.logf("mapRoutine: exiting") | 
					
						
							|  |  |  | 			return | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2022-02-12 08:05:24 -08:00
										 |  |  | 		c.logf("[v1] mapRoutine: %s", c.state) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 		loggedIn := c.loggedIn | 
					
						
							|  |  |  | 		ctx := c.mapCtx | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		report := func(err error, msg string) { | 
					
						
							| 
									
										
										
										
											2020-12-21 10:58:06 -08:00
										 |  |  | 			c.logf("[v1] %s: %v", msg, err) | 
					
						
							| 
									
										
										
										
											2021-10-26 10:19:35 -07:00
										 |  |  | 			err = fmt.Errorf("%s: %w", msg, err) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			// don't send status updates for context errors, | 
					
						
							|  |  |  | 			// since context cancelation is always on purpose. | 
					
						
							|  |  |  | 			if ctx.Err() == nil { | 
					
						
							|  |  |  | 				c.sendStatus("mapRoutine1", err, "", nil) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if !loggedIn { | 
					
						
							|  |  |  | 			// Wait for something interesting to happen | 
					
						
							|  |  |  | 			c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			c.inMapPoll = false | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 			c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			<-ctx.Done() | 
					
						
							|  |  |  | 			c.logf("[v1] mapRoutine: context done.") | 
					
						
							|  |  |  | 			continue | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2024-04-26 09:49:30 -07:00
										 |  |  | 		c.direct.health.SetOutOfPollNetMap() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		err := c.direct.PollNetMap(ctx, mrs) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-26 09:49:30 -07:00
										 |  |  | 		c.direct.health.SetOutOfPollNetMap() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		c.mu.Lock() | 
					
						
							|  |  |  | 		c.inMapPoll = false | 
					
						
							|  |  |  | 		if c.state == StateSynchronized { | 
					
						
							|  |  |  | 			c.state = StateAuthenticated | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		paused := c.paused | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2020-07-31 09:39:45 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		if paused { | 
					
						
							|  |  |  | 			mrs.bo.BackOff(ctx, nil) | 
					
						
							|  |  |  | 			c.logf("mapRoutine: paused") | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2023-08-12 08:18:10 -07:00
										 |  |  | 			mrs.bo.BackOff(ctx, err) | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 			report(err, "PollNetMap") | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) AuthCantContinue() bool { | 
					
						
							| 
									
										
										
										
											2021-04-21 12:57:48 -07:00
										 |  |  | 	if c == nil { | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	defer c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return !c.loggedIn && (c.loginGoal == nil || c.loginGoal.url != "") | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) SetHostinfo(hi *tailcfg.Hostinfo) { | 
					
						
							| 
									
										
										
										
											2020-02-25 10:04:20 -08:00
										 |  |  | 	if hi == nil { | 
					
						
							|  |  |  | 		panic("nil Hostinfo") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-04-02 11:18:39 +11:00
										 |  |  | 	if !c.direct.SetHostinfo(hi) { | 
					
						
							| 
									
										
										
										
											2020-07-28 15:13:34 -07:00
										 |  |  | 		// No changes. Don't log. | 
					
						
							| 
									
										
										
										
											2020-04-02 11:18:39 +11:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	// Send new Hostinfo to server | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.updateControl() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) SetNetInfo(ni *tailcfg.NetInfo) { | 
					
						
							| 
									
										
										
										
											2020-03-03 22:21:56 -08:00
										 |  |  | 	if ni == nil { | 
					
						
							|  |  |  | 		panic("nil NetInfo") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-04-02 11:18:39 +11:00
										 |  |  | 	if !c.direct.SetNetInfo(ni) { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-03 15:07:30 -07:00
										 |  |  | 	// Send new NetInfo to server | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	c.updateControl() | 
					
						
							| 
									
										
										
										
											2020-03-03 22:21:56 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-27 13:40:31 -07:00
										 |  |  | // SetTKAHead updates the TKA head hash that map-request infrastructure sends. | 
					
						
							|  |  |  | func (c *Auto) SetTKAHead(headHash string) { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 	if !c.direct.SetTKAHead(headHash) { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Send new TKAHead to server | 
					
						
							|  |  |  | 	c.updateControl() | 
					
						
							| 
									
										
										
										
											2022-10-27 13:40:31 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | // sendStatus can not be called with the c.mu held. | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkMap) { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2022-06-19 18:14:45 -07:00
										 |  |  | 	if c.closed { | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	state := c.state | 
					
						
							|  |  |  | 	loggedIn := c.loggedIn | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	inMapPoll := c.inMapPoll | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 10:58:06 -08:00
										 |  |  | 	c.logf("[v1] sendStatus: %s: %v", who, state) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 11:35:43 -07:00
										 |  |  | 	var p persist.PersistView | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	if nm != nil && loggedIn && inMapPoll { | 
					
						
							| 
									
										
										
										
											2023-08-30 11:35:43 -07:00
										 |  |  | 		p = c.direct.GetPersist() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		// don't send netmap status, as it's misleading when we're | 
					
						
							|  |  |  | 		// not logged in. | 
					
						
							|  |  |  | 		nm = nil | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 	newSt := &Status{ | 
					
						
							| 
									
										
										
										
											2023-08-30 11:09:36 -07:00
										 |  |  | 		URL:     url, | 
					
						
							|  |  |  | 		Persist: p, | 
					
						
							|  |  |  | 		NetMap:  nm, | 
					
						
							|  |  |  | 		Err:     err, | 
					
						
							|  |  |  | 		state:   state, | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 	c.lastStatus.Store(newSt) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-31 14:31:23 -07:00
										 |  |  | 	// Launch a new goroutine to avoid blocking the caller while the observer | 
					
						
							|  |  |  | 	// does its thing, which may result in a call back into the client. | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 	metricQueued.Add(1) | 
					
						
							| 
									
										
										
										
											2023-09-01 09:56:31 -07:00
										 |  |  | 	c.observerQueue.Add(func() { | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 		if canSkipStatus(newSt, c.lastStatus.Load()) { | 
					
						
							|  |  |  | 			metricSkippable.Add(1) | 
					
						
							|  |  |  | 			if !c.direct.controlKnobs.DisableSkipStatusQueue.Load() { | 
					
						
							|  |  |  | 				metricSkipped.Add(1) | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		c.observer.SetControlClientStatus(c, *newSt) | 
					
						
							| 
									
										
										
										
											2025-01-30 07:23:36 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Best effort stop retaining the memory now that we've sent it to the | 
					
						
							|  |  |  | 		// observer (LocalBackend). We CAS here because the caller goroutine is | 
					
						
							|  |  |  | 		// doing a Store which we want to win a race. This is only a memory | 
					
						
							|  |  |  | 		// optimization and is not for correctness. | 
					
						
							|  |  |  | 		// | 
					
						
							|  |  |  | 		// If the CAS fails, that means somebody else's Store replaced our | 
					
						
							|  |  |  | 		// pointer (so mission accomplished: our netmap is no longer retained in | 
					
						
							|  |  |  | 		// any case) and that Store caller will be responsible for removing | 
					
						
							|  |  |  | 		// their own netmap (or losing their race too, down the chain). | 
					
						
							|  |  |  | 		// Eventually the last caller will win this CAS and zero lastStatus. | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | 		c.lastStatus.CompareAndSwap(newSt, nil) | 
					
						
							| 
									
										
										
										
											2023-09-01 09:56:31 -07:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-24 13:09:21 -08:00
										 |  |  | var ( | 
					
						
							|  |  |  | 	metricQueued    = clientmetric.NewCounter("controlclient_auto_status_queued") | 
					
						
							|  |  |  | 	metricSkippable = clientmetric.NewCounter("controlclient_auto_status_queue_skippable") | 
					
						
							|  |  |  | 	metricSkipped   = clientmetric.NewCounter("controlclient_auto_status_queue_skipped") | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // canSkipStatus reports whether we can skip sending s1, knowing | 
					
						
							|  |  |  | // that s2 is enqueued sometime in the future after s1. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // s1 must be non-nil. s2 may be nil. | 
					
						
							|  |  |  | func canSkipStatus(s1, s2 *Status) bool { | 
					
						
							|  |  |  | 	if s2 == nil { | 
					
						
							|  |  |  | 		// Nothing in the future. | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if s1 == s2 { | 
					
						
							|  |  |  | 		// If the last item in the queue is the same as s1, | 
					
						
							|  |  |  | 		// we can't skip it. | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if s1.Err != nil || s1.URL != "" { | 
					
						
							|  |  |  | 		// If s1 has an error or a URL, we shouldn't skip it, lest the error go | 
					
						
							|  |  |  | 		// away in s2 or in-between. We want to make sure all the subsystems see | 
					
						
							|  |  |  | 		// it. Plus there aren't many of these, so not worth skipping. | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !s1.Persist.Equals(s2.Persist) || s1.state != s2.state { | 
					
						
							|  |  |  | 		// If s1 has a different Persist or state than s2, | 
					
						
							|  |  |  | 		// don't skip it. We only care about skipping the typical | 
					
						
							|  |  |  | 		// entries where the only difference is the NetMap. | 
					
						
							|  |  |  | 		return false | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// If nothing above precludes it, and both s1 and s2 have NetMaps, then | 
					
						
							|  |  |  | 	// we can skip it, because s2's NetMap is a newer version and we can | 
					
						
							|  |  |  | 	// jump straight from whatever state we had before to s2's state, | 
					
						
							|  |  |  | 	// without passing through s1's state first. A NetMap is regrettably a | 
					
						
							|  |  |  | 	// full snapshot of the state, not an incremental delta. We're slowly | 
					
						
							|  |  |  | 	// moving towards passing around only deltas around internally at all | 
					
						
							|  |  |  | 	// layers, but this is explicitly the case where we didn't have a delta | 
					
						
							|  |  |  | 	// path for the message we received over the wire and had to resort | 
					
						
							|  |  |  | 	// to the legacy full NetMap path. And then we can get behind processing | 
					
						
							|  |  |  | 	// these full NetMap snapshots in LocalBackend/wgengine/magicsock/netstack | 
					
						
							|  |  |  | 	// and this path (when it returns true) lets us skip over useless work | 
					
						
							|  |  |  | 	// and not get behind in the queue. This matters in particular for tailnets | 
					
						
							|  |  |  | 	// that are both very large + very churny. | 
					
						
							|  |  |  | 	return s1.NetMap != nil && s2.NetMap != nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-17 14:54:41 -07:00
										 |  |  | func (c *Auto) Login(flags LoginFlags) { | 
					
						
							|  |  |  | 	c.logf("client.Login(%v)", flags) | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	defer c.mu.Unlock() | 
					
						
							|  |  |  | 	if c.closed { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-06 21:33:37 -07:00
										 |  |  | 	if c.direct != nil && c.direct.panicOnUse { | 
					
						
							|  |  |  | 		panic("tainted client") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	c.wantLoggedIn = true | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.loginGoal = &LoginGoal{ | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 		flags: flags, | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	c.cancelMapCtxLocked() | 
					
						
							|  |  |  | 	c.cancelAuthCtxLocked() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | var ErrClientClosed = errors.New("client closed") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) Logout(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2021-04-07 21:06:31 -07:00
										 |  |  | 	c.logf("client.Logout()") | 
					
						
							|  |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	c.wantLoggedIn = false | 
					
						
							|  |  |  | 	c.loginGoal = nil | 
					
						
							|  |  |  | 	closed := c.closed | 
					
						
							| 
									
										
										
										
											2024-05-06 21:33:37 -07:00
										 |  |  | 	if c.direct != nil && c.direct.panicOnUse { | 
					
						
							|  |  |  | 		panic("tainted client") | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2021-04-07 21:06:31 -07:00
										 |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	if closed { | 
					
						
							|  |  |  | 		return ErrClientClosed | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if err := c.direct.TryLogout(ctx); err != nil { | 
					
						
							| 
									
										
										
										
											2021-04-07 21:06:31 -07:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-08-30 18:18:10 -07:00
										 |  |  | 	c.mu.Lock() | 
					
						
							|  |  |  | 	c.loggedIn = false | 
					
						
							|  |  |  | 	c.state = StateNotAuthenticated | 
					
						
							|  |  |  | 	c.cancelAuthCtxLocked() | 
					
						
							|  |  |  | 	c.cancelMapCtxLocked() | 
					
						
							|  |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	c.sendStatus("authRoutine-wantout", nil, "", nil) | 
					
						
							|  |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-09 14:42:42 -08:00
										 |  |  | func (c *Auto) SetExpirySooner(ctx context.Context, expiry time.Time) error { | 
					
						
							|  |  |  | 	return c.direct.SetExpirySooner(ctx, expiry) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-31 08:25:39 -07:00
										 |  |  | // UpdateEndpoints sets the client's discovered endpoints and sends | 
					
						
							|  |  |  | // them to the control server if they've changed. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // It does not retain the provided slice. | 
					
						
							| 
									
										
										
										
											2022-06-19 16:31:54 -07:00
										 |  |  | func (c *Auto) UpdateEndpoints(endpoints []tailcfg.Endpoint) { | 
					
						
							|  |  |  | 	changed := c.direct.SetEndpoints(endpoints) | 
					
						
							| 
									
										
										
										
											2020-02-14 09:28:29 -08:00
										 |  |  | 	if changed { | 
					
						
							| 
									
										
										
										
											2023-08-09 19:56:43 -07:00
										 |  |  | 		c.updateControl() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) Shutdown() { | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mu.Lock() | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 	if c.closed { | 
					
						
							|  |  |  | 		c.mu.Unlock() | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	c.logf("client.Shutdown ...") | 
					
						
							| 
									
										
										
										
											2025-03-12 10:37:03 -04:00
										 |  |  | 	shutdownFn := c.shutdownFn | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 11:45:48 -08:00
										 |  |  | 	direct := c.direct | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 	c.closed = true | 
					
						
							|  |  |  | 	c.observerQueue.Shutdown() | 
					
						
							|  |  |  | 	c.cancelAuthCtxLocked() | 
					
						
							|  |  |  | 	c.cancelMapCtxLocked() | 
					
						
							|  |  |  | 	for _, w := range c.unpauseWaiters { | 
					
						
							|  |  |  | 		w <- false | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 	c.unpauseWaiters = nil | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	c.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 10:37:03 -04:00
										 |  |  | 	if shutdownFn != nil { | 
					
						
							|  |  |  | 		shutdownFn() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 	c.unregisterHealthWatch() | 
					
						
							|  |  |  | 	<-c.authDone | 
					
						
							|  |  |  | 	<-c.mapDone | 
					
						
							|  |  |  | 	<-c.updateDone | 
					
						
							|  |  |  | 	if direct != nil { | 
					
						
							|  |  |  | 		direct.Close() | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2024-05-07 10:55:26 -07:00
										 |  |  | 	ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) | 
					
						
							|  |  |  | 	defer cancel() | 
					
						
							|  |  |  | 	c.observerQueue.Wait(ctx) | 
					
						
							|  |  |  | 	c.logf("Client.Shutdown done.") | 
					
						
							| 
									
										
										
										
											2020-02-05 14:16:58 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | // NodePublicKey returns the node public key currently in use. This is | 
					
						
							|  |  |  | // used exclusively in tests. | 
					
						
							| 
									
										
										
										
											2021-10-28 14:22:51 -07:00
										 |  |  | func (c *Auto) TestOnlyNodePublicKey() key.NodePublic { | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | 	priv := c.direct.GetPersist() | 
					
						
							| 
									
										
										
										
											2022-11-09 10:58:10 +05:00
										 |  |  | 	return priv.PrivateNodeKey().Public() | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) TestOnlySetAuthKey(authkey string) { | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | 	c.direct.mu.Lock() | 
					
						
							|  |  |  | 	defer c.direct.mu.Unlock() | 
					
						
							|  |  |  | 	c.direct.authKey = authkey | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-29 23:27:00 -04:00
										 |  |  | func (c *Auto) TestOnlyTimeNow() time.Time { | 
					
						
							| 
									
										
										
										
											2023-08-04 19:29:44 -04:00
										 |  |  | 	return c.clock.Now() | 
					
						
							| 
									
										
										
										
											2020-05-27 18:46:09 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-06-07 16:03:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // SetDNS sends the SetDNSRequest request to the control plane server, | 
					
						
							|  |  |  | // requesting a DNS record be created or updated. | 
					
						
							|  |  |  | func (c *Auto) SetDNS(ctx context.Context, req *tailcfg.SetDNSRequest) error { | 
					
						
							|  |  |  | 	return c.direct.SetDNS(ctx, req) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-03-10 10:28:42 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func (c *Auto) DoNoiseRequest(req *http.Request) (*http.Response, error) { | 
					
						
							|  |  |  | 	return c.direct.DoNoiseRequest(req) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-11-01 20:37:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // GetSingleUseNoiseRoundTripper returns a RoundTripper that can be only be used | 
					
						
							|  |  |  | // once (and must be used once) to make a single HTTP request over the noise | 
					
						
							|  |  |  | // channel to the coordination server. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // In addition to the RoundTripper, it returns the HTTP/2 channel's early noise | 
					
						
							|  |  |  | // payload, if any. | 
					
						
							|  |  |  | func (c *Auto) GetSingleUseNoiseRoundTripper(ctx context.Context) (http.RoundTripper, *tailcfg.EarlyNoise, error) { | 
					
						
							|  |  |  | 	return c.direct.GetSingleUseNoiseRoundTripper(ctx) | 
					
						
							|  |  |  | } |