mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-07 08:07:42 +00:00
control/controlclient: store netinfo and hostinfo separately
Currently, when SetNetInfo is called it sets the value on hostinfo.NetInfo. However, when SetHostInfo is called it overwrites the hostinfo field which may mean it also clears out the NetInfo it had just received. This commit stores NetInfo separately and combines it into Hostinfo as needed so that control is always notified of the latest values. Also, remove unused copies of Hostinfo from ipn.Status and controlclient.Auto. Updates #tailscale/corp#4824 (maybe fixes) Signed-off-by: Maisem Ali <maisem@tailscale.com>
This commit is contained in:
parent
ae483d3446
commit
c60cbca371
@ -61,10 +61,9 @@ type Auto struct {
|
|||||||
loggedIn bool // true if currently logged in
|
loggedIn bool // true if currently logged in
|
||||||
loginGoal *LoginGoal // non-nil if some login activity is desired
|
loginGoal *LoginGoal // non-nil if some login activity is desired
|
||||||
synced bool // true if our netmap is up-to-date
|
synced bool // true if our netmap is up-to-date
|
||||||
hostinfo *tailcfg.Hostinfo
|
inPollNetMap bool // true if currently running a PollNetMap
|
||||||
inPollNetMap bool // true if currently running a PollNetMap
|
inLiteMapUpdate bool // true if a lite (non-streaming) map request is outstanding
|
||||||
inLiteMapUpdate bool // true if a lite (non-streaming) map request is outstanding
|
inSendStatus int // number of sendStatus calls currently in progress
|
||||||
inSendStatus int // number of sendStatus calls currently in progress
|
|
||||||
state State
|
state State
|
||||||
|
|
||||||
authCtx context.Context // context used for auth requests
|
authCtx context.Context // context used for auth requests
|
||||||
@ -555,9 +554,8 @@ func (c *Auto) SetNetInfo(ni *tailcfg.NetInfo) {
|
|||||||
if !c.direct.SetNetInfo(ni) {
|
if !c.direct.SetNetInfo(ni) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
c.logf("NetInfo: %v", ni)
|
|
||||||
|
|
||||||
// Send new Hostinfo (which includes NetInfo) to server
|
// Send new NetInfo to server
|
||||||
c.sendNewMapRequest()
|
c.sendNewMapRequest()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -567,7 +565,6 @@ func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkM
|
|||||||
loggedIn := c.loggedIn
|
loggedIn := c.loggedIn
|
||||||
synced := c.synced
|
synced := c.synced
|
||||||
statusFunc := c.statusFunc
|
statusFunc := c.statusFunc
|
||||||
hi := c.hostinfo
|
|
||||||
c.inSendStatus++
|
c.inSendStatus++
|
||||||
c.mu.Unlock()
|
c.mu.Unlock()
|
||||||
|
|
||||||
@ -595,7 +592,6 @@ func (c *Auto) sendStatus(who string, err error, url string, nm *netmap.NetworkM
|
|||||||
URL: url,
|
URL: url,
|
||||||
Persist: p,
|
Persist: p,
|
||||||
NetMap: nm,
|
NetMap: nm,
|
||||||
Hostinfo: hi,
|
|
||||||
State: state,
|
State: state,
|
||||||
Err: err,
|
Err: err,
|
||||||
}
|
}
|
||||||
|
@ -22,7 +22,7 @@ func fieldsOf(t reflect.Type) (fields []string) {
|
|||||||
|
|
||||||
func TestStatusEqual(t *testing.T) {
|
func TestStatusEqual(t *testing.T) {
|
||||||
// Verify that the Equal method stays in sync with reality
|
// Verify that the Equal method stays in sync with reality
|
||||||
equalHandles := []string{"LoginFinished", "LogoutFinished", "Err", "URL", "NetMap", "State", "Persist", "Hostinfo"}
|
equalHandles := []string{"LoginFinished", "LogoutFinished", "Err", "URL", "NetMap", "State", "Persist"}
|
||||||
if have := fieldsOf(reflect.TypeOf(Status{})); !reflect.DeepEqual(have, equalHandles) {
|
if have := fieldsOf(reflect.TypeOf(Status{})); !reflect.DeepEqual(have, equalHandles) {
|
||||||
t.Errorf("Status.Equal check might be out of sync\nfields: %q\nhandled: %q\n",
|
t.Errorf("Status.Equal check might be out of sync\nfields: %q\nhandled: %q\n",
|
||||||
have, equalHandles)
|
have, equalHandles)
|
||||||
|
@ -80,12 +80,12 @@ type Direct struct {
|
|||||||
sfGroup singleflight.Group // protects noiseClient creation.
|
sfGroup singleflight.Group // protects noiseClient creation.
|
||||||
noiseClient *noiseClient
|
noiseClient *noiseClient
|
||||||
|
|
||||||
persist persist.Persist
|
persist persist.Persist
|
||||||
authKey string
|
authKey string
|
||||||
tryingNewKey key.NodePrivate
|
tryingNewKey key.NodePrivate
|
||||||
expiry *time.Time
|
expiry *time.Time
|
||||||
// hostinfo is mutated in-place while mu is held.
|
|
||||||
hostinfo *tailcfg.Hostinfo // always non-nil
|
hostinfo *tailcfg.Hostinfo // always non-nil
|
||||||
|
netinfo *tailcfg.NetInfo
|
||||||
endpoints []tailcfg.Endpoint
|
endpoints []tailcfg.Endpoint
|
||||||
everEndpoints bool // whether we've ever had non-empty endpoints
|
everEndpoints bool // whether we've ever had non-empty endpoints
|
||||||
localPort uint16 // or zero to mean auto
|
localPort uint16 // or zero to mean auto
|
||||||
@ -208,7 +208,12 @@ func NewDirect(opts Options) (*Direct, error) {
|
|||||||
if opts.Hostinfo == nil {
|
if opts.Hostinfo == nil {
|
||||||
c.SetHostinfo(hostinfo.New())
|
c.SetHostinfo(hostinfo.New())
|
||||||
} else {
|
} else {
|
||||||
|
ni := opts.Hostinfo.NetInfo
|
||||||
|
opts.Hostinfo.NetInfo = nil
|
||||||
c.SetHostinfo(opts.Hostinfo)
|
c.SetHostinfo(opts.Hostinfo)
|
||||||
|
if ni != nil {
|
||||||
|
c.SetNetInfo(ni)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return c, nil
|
return c, nil
|
||||||
}
|
}
|
||||||
@ -253,14 +258,11 @@ func (c *Direct) SetNetInfo(ni *tailcfg.NetInfo) bool {
|
|||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
|
|
||||||
if c.hostinfo == nil {
|
if reflect.DeepEqual(ni, c.netinfo) {
|
||||||
c.logf("[unexpected] SetNetInfo called with no HostInfo; ignoring NetInfo update: %+v", ni)
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
if reflect.DeepEqual(ni, c.hostinfo.NetInfo) {
|
c.netinfo = ni.Clone()
|
||||||
return false
|
c.logf("NetInfo: %v", ni)
|
||||||
}
|
|
||||||
c.hostinfo.NetInfo = ni.Clone()
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,6 +339,14 @@ type httpClient interface {
|
|||||||
Do(req *http.Request) (*http.Response, error)
|
Do(req *http.Request) (*http.Response, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// hostInfoLocked returns a Clone of c.hostinfo and c.netinfo.
|
||||||
|
// It must only be called with c.mu held.
|
||||||
|
func (c *Direct) hostInfoLocked() *tailcfg.Hostinfo {
|
||||||
|
hi := c.hostinfo.Clone()
|
||||||
|
hi.NetInfo = c.netinfo.Clone()
|
||||||
|
return hi
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Direct) doLogin(ctx context.Context, opt loginOpt) (mustRegen bool, newURL string, err error) {
|
func (c *Direct) doLogin(ctx context.Context, opt loginOpt) (mustRegen bool, newURL string, err error) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
persist := c.persist
|
persist := c.persist
|
||||||
@ -344,7 +354,7 @@ func (c *Direct) doLogin(ctx context.Context, opt loginOpt) (mustRegen bool, new
|
|||||||
serverKey := c.serverKey
|
serverKey := c.serverKey
|
||||||
serverNoiseKey := c.serverNoiseKey
|
serverNoiseKey := c.serverNoiseKey
|
||||||
authKey := c.authKey
|
authKey := c.authKey
|
||||||
hi := c.hostinfo.Clone()
|
hi := c.hostInfoLocked()
|
||||||
backendLogID := hi.BackendLogID
|
backendLogID := hi.BackendLogID
|
||||||
expired := c.expiry != nil && !c.expiry.IsZero() && c.expiry.Before(c.timeNow())
|
expired := c.expiry != nil && !c.expiry.IsZero() && c.expiry.Before(c.timeNow())
|
||||||
c.mu.Unlock()
|
c.mu.Unlock()
|
||||||
@ -646,7 +656,7 @@ func (c *Direct) sendMapRequest(ctx context.Context, maxPolls int, cb func(*netm
|
|||||||
serverURL := c.serverURL
|
serverURL := c.serverURL
|
||||||
serverKey := c.serverKey
|
serverKey := c.serverKey
|
||||||
serverNoiseKey := c.serverNoiseKey
|
serverNoiseKey := c.serverNoiseKey
|
||||||
hi := c.hostinfo.Clone()
|
hi := c.hostInfoLocked()
|
||||||
backendLogID := hi.BackendLogID
|
backendLogID := hi.BackendLogID
|
||||||
localPort := c.localPort
|
localPort := c.localPort
|
||||||
var epStrs []string
|
var epStrs []string
|
||||||
|
@ -9,7 +9,6 @@
|
|||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
"tailscale.com/tailcfg"
|
|
||||||
"tailscale.com/types/empty"
|
"tailscale.com/types/empty"
|
||||||
"tailscale.com/types/netmap"
|
"tailscale.com/types/netmap"
|
||||||
"tailscale.com/types/persist"
|
"tailscale.com/types/persist"
|
||||||
@ -75,9 +74,8 @@ type Status struct {
|
|||||||
// package, but we have some automated tests elsewhere that need to
|
// package, but we have some automated tests elsewhere that need to
|
||||||
// use them. Please don't use these fields.
|
// use them. Please don't use these fields.
|
||||||
// TODO(apenwarr): Unexport or remove these.
|
// TODO(apenwarr): Unexport or remove these.
|
||||||
State State
|
State State
|
||||||
Persist *persist.Persist // locally persisted configuration
|
Persist *persist.Persist // locally persisted configuration
|
||||||
Hostinfo *tailcfg.Hostinfo // current Hostinfo data
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Equal reports whether s and s2 are equal.
|
// Equal reports whether s and s2 are equal.
|
||||||
@ -92,7 +90,6 @@ func (s *Status) Equal(s2 *Status) bool {
|
|||||||
s.URL == s2.URL &&
|
s.URL == s2.URL &&
|
||||||
reflect.DeepEqual(s.Persist, s2.Persist) &&
|
reflect.DeepEqual(s.Persist, s2.Persist) &&
|
||||||
reflect.DeepEqual(s.NetMap, s2.NetMap) &&
|
reflect.DeepEqual(s.NetMap, s2.NetMap) &&
|
||||||
reflect.DeepEqual(s.Hostinfo, s2.Hostinfo) &&
|
|
||||||
s.State == s2.State
|
s.State == s2.State
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -935,8 +935,7 @@ func (b *LocalBackend) Start(opts ipn.Options) error {
|
|||||||
httpTestClient := b.httpTestClient
|
httpTestClient := b.httpTestClient
|
||||||
|
|
||||||
if b.hostinfo != nil {
|
if b.hostinfo != nil {
|
||||||
hostinfo.Services = b.hostinfo.Services // keep any previous session and netinfo
|
hostinfo.Services = b.hostinfo.Services // keep any previous services
|
||||||
hostinfo.NetInfo = b.hostinfo.NetInfo
|
|
||||||
}
|
}
|
||||||
b.hostinfo = hostinfo
|
b.hostinfo = hostinfo
|
||||||
b.state = ipn.NoState
|
b.state = ipn.NoState
|
||||||
@ -2870,9 +2869,6 @@ func (b *LocalBackend) assertClientLocked() {
|
|||||||
func (b *LocalBackend) setNetInfo(ni *tailcfg.NetInfo) {
|
func (b *LocalBackend) setNetInfo(ni *tailcfg.NetInfo) {
|
||||||
b.mu.Lock()
|
b.mu.Lock()
|
||||||
cc := b.cc
|
cc := b.cc
|
||||||
if b.hostinfo != nil {
|
|
||||||
b.hostinfo.NetInfo = ni.Clone()
|
|
||||||
}
|
|
||||||
b.mu.Unlock()
|
b.mu.Unlock()
|
||||||
|
|
||||||
if cc == nil {
|
if cc == nil {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user