cmd/tailscale,feature/relayserver,ipn: add relay-server-static-endpoints set flag
Updates tailscale/corp#31489 Updates #17791 Signed-off-by: Jordan Whited <jordan@tailscale.com>
This commit is contained in:
committed by
Jordan Whited
parent
755309c04e
commit
7426eca163
@@ -9,6 +9,7 @@ import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"net/netip"
|
||||
|
||||
"tailscale.com/disco"
|
||||
"tailscale.com/feature"
|
||||
@@ -23,6 +24,7 @@ import (
|
||||
"tailscale.com/types/key"
|
||||
"tailscale.com/types/logger"
|
||||
"tailscale.com/types/ptr"
|
||||
"tailscale.com/types/views"
|
||||
"tailscale.com/util/eventbus"
|
||||
"tailscale.com/wgengine/magicsock"
|
||||
)
|
||||
@@ -85,6 +87,7 @@ type relayServer interface {
|
||||
AllocateEndpoint(discoA, discoB key.DiscoPublic) (endpoint.ServerEndpoint, error)
|
||||
GetSessions() []status.ServerSession
|
||||
SetDERPMapView(tailcfg.DERPMapView)
|
||||
SetStaticAddrPorts(addrPorts views.Slice[netip.AddrPort])
|
||||
}
|
||||
|
||||
// extension is an [ipnext.Extension] managing the relay server on platforms
|
||||
@@ -95,12 +98,13 @@ type extension struct {
|
||||
ec *eventbus.Client
|
||||
respPub *eventbus.Publisher[magicsock.UDPRelayAllocResp]
|
||||
|
||||
mu syncs.Mutex // guards the following fields
|
||||
shutdown bool // true if Shutdown() has been called
|
||||
rs relayServer // nil when disabled
|
||||
port *int // ipn.Prefs.RelayServerPort, nil if disabled
|
||||
derpMapView tailcfg.DERPMapView // latest seen over the eventbus
|
||||
hasNodeAttrDisableRelayServer bool // [tailcfg.NodeAttrDisableRelayServer]
|
||||
mu syncs.Mutex // guards the following fields
|
||||
shutdown bool // true if Shutdown() has been called
|
||||
rs relayServer // nil when disabled
|
||||
port *int // ipn.Prefs.RelayServerPort, nil if disabled
|
||||
staticEndpoints views.Slice[netip.AddrPort] // ipn.Prefs.RelayServerStaticEndpoints
|
||||
derpMapView tailcfg.DERPMapView // latest seen over the eventbus
|
||||
hasNodeAttrDisableRelayServer bool // [tailcfg.NodeAttrDisableRelayServer]
|
||||
}
|
||||
|
||||
// Name implements [ipnext.Extension].
|
||||
@@ -186,6 +190,7 @@ func (e *extension) relayServerShouldBeRunningLocked() bool {
|
||||
|
||||
// handleRelayServerLifetimeLocked handles the lifetime of [e.rs].
|
||||
func (e *extension) handleRelayServerLifetimeLocked() {
|
||||
defer e.handleRelayServerStaticAddrPortsLocked()
|
||||
if !e.relayServerShouldBeRunningLocked() {
|
||||
e.stopRelayServerLocked()
|
||||
return
|
||||
@@ -195,6 +200,13 @@ func (e *extension) handleRelayServerLifetimeLocked() {
|
||||
e.tryStartRelayServerLocked()
|
||||
}
|
||||
|
||||
func (e *extension) handleRelayServerStaticAddrPortsLocked() {
|
||||
if e.rs != nil {
|
||||
// TODO(jwhited): env var support
|
||||
e.rs.SetStaticAddrPorts(e.staticEndpoints)
|
||||
}
|
||||
}
|
||||
|
||||
func (e *extension) selfNodeViewChanged(nodeView tailcfg.NodeView) {
|
||||
e.mu.Lock()
|
||||
defer e.mu.Unlock()
|
||||
@@ -205,6 +217,7 @@ func (e *extension) selfNodeViewChanged(nodeView tailcfg.NodeView) {
|
||||
func (e *extension) profileStateChanged(_ ipn.LoginProfileView, prefs ipn.PrefsView, sameNode bool) {
|
||||
e.mu.Lock()
|
||||
defer e.mu.Unlock()
|
||||
e.staticEndpoints = prefs.RelayServerStaticEndpoints()
|
||||
newPort, ok := prefs.RelayServerPort().GetOk()
|
||||
enableOrDisableServer := ok != (e.port != nil)
|
||||
portChanged := ok && e.port != nil && newPort != *e.port
|
||||
|
||||
@@ -5,7 +5,9 @@ package relayserver
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"net/netip"
|
||||
"reflect"
|
||||
"slices"
|
||||
"testing"
|
||||
|
||||
"tailscale.com/ipn"
|
||||
@@ -17,15 +19,21 @@ import (
|
||||
"tailscale.com/types/key"
|
||||
"tailscale.com/types/logger"
|
||||
"tailscale.com/types/ptr"
|
||||
"tailscale.com/types/views"
|
||||
)
|
||||
|
||||
func Test_extension_profileStateChanged(t *testing.T) {
|
||||
prefsWithPortOne := ipn.Prefs{RelayServerPort: ptr.To(1)}
|
||||
prefsWithNilPort := ipn.Prefs{RelayServerPort: nil}
|
||||
prefsWithPortOneRelayEndpoints := ipn.Prefs{
|
||||
RelayServerPort: ptr.To(1),
|
||||
RelayServerStaticEndpoints: []netip.AddrPort{netip.MustParseAddrPort("127.0.0.1:7777")},
|
||||
}
|
||||
|
||||
type fields struct {
|
||||
port *int
|
||||
rs relayServer
|
||||
port *int
|
||||
staticEndpoints views.Slice[netip.AddrPort]
|
||||
rs relayServer
|
||||
}
|
||||
type args struct {
|
||||
prefs ipn.PrefsView
|
||||
@@ -38,6 +46,7 @@ func Test_extension_profileStateChanged(t *testing.T) {
|
||||
wantPort *int
|
||||
wantRelayServerFieldNonNil bool
|
||||
wantRelayServerFieldMutated bool
|
||||
wantEndpoints []netip.AddrPort
|
||||
}{
|
||||
{
|
||||
name: "no changes non-nil port previously running",
|
||||
@@ -53,6 +62,52 @@ func Test_extension_profileStateChanged(t *testing.T) {
|
||||
wantRelayServerFieldNonNil: true,
|
||||
wantRelayServerFieldMutated: false,
|
||||
},
|
||||
{
|
||||
name: "set addr ports unchanged port previously running",
|
||||
fields: fields{
|
||||
port: ptr.To(1),
|
||||
rs: mockRelayServerNotZeroVal(),
|
||||
},
|
||||
args: args{
|
||||
prefs: prefsWithPortOneRelayEndpoints.View(),
|
||||
sameNode: true,
|
||||
},
|
||||
wantPort: ptr.To(1),
|
||||
wantRelayServerFieldNonNil: true,
|
||||
wantRelayServerFieldMutated: false,
|
||||
wantEndpoints: prefsWithPortOneRelayEndpoints.RelayServerStaticEndpoints,
|
||||
},
|
||||
{
|
||||
name: "set addr ports not previously running",
|
||||
fields: fields{
|
||||
port: nil,
|
||||
rs: nil,
|
||||
},
|
||||
args: args{
|
||||
prefs: prefsWithPortOneRelayEndpoints.View(),
|
||||
sameNode: true,
|
||||
},
|
||||
wantPort: ptr.To(1),
|
||||
wantRelayServerFieldNonNil: true,
|
||||
wantRelayServerFieldMutated: true,
|
||||
wantEndpoints: prefsWithPortOneRelayEndpoints.RelayServerStaticEndpoints,
|
||||
},
|
||||
{
|
||||
name: "clear addr ports unchanged port previously running",
|
||||
fields: fields{
|
||||
port: ptr.To(1),
|
||||
staticEndpoints: views.SliceOf(prefsWithPortOneRelayEndpoints.RelayServerStaticEndpoints),
|
||||
rs: mockRelayServerNotZeroVal(),
|
||||
},
|
||||
args: args{
|
||||
prefs: prefsWithPortOne.View(),
|
||||
sameNode: true,
|
||||
},
|
||||
wantPort: ptr.To(1),
|
||||
wantRelayServerFieldNonNil: true,
|
||||
wantRelayServerFieldMutated: false,
|
||||
wantEndpoints: nil,
|
||||
},
|
||||
{
|
||||
name: "prefs port nil",
|
||||
fields: fields{
|
||||
@@ -160,6 +215,7 @@ func Test_extension_profileStateChanged(t *testing.T) {
|
||||
return &mockRelayServer{}, nil
|
||||
}
|
||||
e.port = tt.fields.port
|
||||
e.staticEndpoints = tt.fields.staticEndpoints
|
||||
e.rs = tt.fields.rs
|
||||
defer e.Shutdown()
|
||||
e.profileStateChanged(ipn.LoginProfileView{}, tt.args.prefs, tt.args.sameNode)
|
||||
@@ -174,24 +230,34 @@ func Test_extension_profileStateChanged(t *testing.T) {
|
||||
if tt.wantRelayServerFieldMutated != !reflect.DeepEqual(tt.fields.rs, e.rs) {
|
||||
t.Errorf("wantRelayServerFieldMutated: %v != !reflect.DeepEqual(tt.fields.rs, e.rs): %v", tt.wantRelayServerFieldMutated, !reflect.DeepEqual(tt.fields.rs, e.rs))
|
||||
}
|
||||
if !slices.Equal(tt.wantEndpoints, e.staticEndpoints.AsSlice()) {
|
||||
t.Errorf("wantEndpoints: %v != %v", tt.wantEndpoints, e.staticEndpoints.AsSlice())
|
||||
}
|
||||
if e.rs != nil && !slices.Equal(tt.wantEndpoints, e.rs.(*mockRelayServer).addrPorts.AsSlice()) {
|
||||
t.Errorf("wantEndpoints: %v != %v", tt.wantEndpoints, e.rs.(*mockRelayServer).addrPorts.AsSlice())
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func mockRelayServerNotZeroVal() *mockRelayServer {
|
||||
return &mockRelayServer{true}
|
||||
return &mockRelayServer{set: true}
|
||||
}
|
||||
|
||||
type mockRelayServer struct {
|
||||
set bool
|
||||
set bool
|
||||
addrPorts views.Slice[netip.AddrPort]
|
||||
}
|
||||
|
||||
func (mockRelayServer) Close() error { return nil }
|
||||
func (mockRelayServer) AllocateEndpoint(_, _ key.DiscoPublic) (endpoint.ServerEndpoint, error) {
|
||||
func (m *mockRelayServer) Close() error { return nil }
|
||||
func (m *mockRelayServer) AllocateEndpoint(_, _ key.DiscoPublic) (endpoint.ServerEndpoint, error) {
|
||||
return endpoint.ServerEndpoint{}, errors.New("not implemented")
|
||||
}
|
||||
func (mockRelayServer) GetSessions() []status.ServerSession { return nil }
|
||||
func (mockRelayServer) SetDERPMapView(tailcfg.DERPMapView) { return }
|
||||
func (m *mockRelayServer) GetSessions() []status.ServerSession { return nil }
|
||||
func (m *mockRelayServer) SetDERPMapView(tailcfg.DERPMapView) { return }
|
||||
func (m *mockRelayServer) SetStaticAddrPorts(aps views.Slice[netip.AddrPort]) {
|
||||
m.addrPorts = aps
|
||||
}
|
||||
|
||||
type mockSafeBackend struct {
|
||||
sys *tsd.System
|
||||
|
||||
Reference in New Issue
Block a user