Skip to content

Commit

Permalink
fix: use magicsock DERPHeaders in netcheck (#39)
Browse files Browse the repository at this point in the history
  • Loading branch information
deansheather authored Sep 21, 2023
1 parent 30f543e commit 8d4d23e
Show file tree
Hide file tree
Showing 3 changed files with 105 additions and 0 deletions.
13 changes: 13 additions & 0 deletions net/netcheck/netcheck.go
Original file line number Diff line number Diff line change
Expand Up @@ -203,6 +203,10 @@ type Client struct {
// If false, the default net.Resolver will be used, with no caching.
UseDNSCache bool

// GetDERPHeaders optionally specifies headers to send with all HTTP(S) DERP
// probes.
GetDERPHeaders func() http.Header

// For tests
testEnoughRegions int
testCaptivePortalDelay time.Duration
Expand Down Expand Up @@ -1277,7 +1281,13 @@ func (c *Client) measureHTTPLatency(ctx context.Context, reg *tailcfg.DERPRegion

var ip netip.Addr

derpHeaders := http.Header{}
if c.GetDERPHeaders != nil {
derpHeaders = c.GetDERPHeaders()
}

dc := derphttp.NewNetcheckClient(c.logf)
dc.Header = derpHeaders
defer dc.Close()

var hasForceHTTPNode = false
Expand Down Expand Up @@ -1356,6 +1366,9 @@ func (c *Client) measureHTTPLatency(ctx context.Context, reg *tailcfg.DERPRegion
if err != nil {
return 0, ip, err
}
for k := range derpHeaders {
req.Header.Set(k, derpHeaders.Get(k))
}

resp, err := hc.Do(req)
if err != nil {
Expand Down
85 changes: 85 additions & 0 deletions net/netcheck/netcheck_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1022,6 +1022,91 @@ func TestNodeAddrResolve(t *testing.T) {
}
}

func TestProbeHeaders(t *testing.T) {
logf, closeLogf := logger.LogfCloser(t.Logf)
defer closeLogf()

// Create a DERP server manually, without a STUN server and with a custom
// handler.
derpServer := derp.NewServer(key.NewNode(), logf)
derpHandler := derphttp.Handler(derpServer)

expectedHeaders := http.Header{}
expectedHeaders.Set("X-Cool-Test", "yes")
expectedHeaders.Set("X-Proxy-Auth-Key", "blah blah blah")

var called atomic.Bool
httpsrv := httptest.NewUnstartedServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
called.Store(true)
for k, v := range expectedHeaders {
if got := r.Header[k]; !reflect.DeepEqual(got, v) {
t.Errorf("unexpected header %q: got %q; want %q", k, got, v)
}
}

if r.URL.Path == "/derp/latency-check" {
w.WriteHeader(http.StatusOK)
return
}
if r.URL.Path == "/derp" {
derpHandler.ServeHTTP(w, r)
return
}

t.Errorf("unexpected request: %v", r.URL)
w.WriteHeader(http.StatusNotFound)
}))
httpsrv.Config.ErrorLog = logger.StdLogger(logf)
httpsrv.Config.TLSNextProto = make(map[string]func(*http.Server, *tls.Conn, http.Handler))
httpsrv.StartTLS()
t.Cleanup(func() {
httpsrv.CloseClientConnections()
httpsrv.Close()
derpServer.Close()
})

derpMap := &tailcfg.DERPMap{
Regions: map[int]*tailcfg.DERPRegion{
1: {
RegionID: 1,
RegionCode: "derpy",
Nodes: []*tailcfg.DERPNode{
{
Name: "d1",
RegionID: 1,
HostName: "localhost",
// Don't specify an IP address to avoid ICMP pinging,
// which will bypass the artificial latency.
IPv4: "",
IPv6: "",
STUNPort: -1,
DERPPort: httpsrv.Listener.Addr().(*net.TCPAddr).Port,
InsecureForTests: true,
},
},
},
},
}

c := &Client{
Logf: t.Logf,
UDPBindAddr: "127.0.0.1:0",
GetDERPHeaders: func() http.Header { return expectedHeaders.Clone() },
}

ctx, cancel := context.WithTimeout(context.Background(), 3*time.Second)
defer cancel()

_, err := c.GetReport(ctx, derpMap)
if err != nil {
t.Fatal(err)
}

if !called.Load() {
t.Error("didn't call test handler")
}
}

func TestNeverPickSTUNOnlyRegionAsPreferredDERP(t *testing.T) {
// Create two DERP regions, one with a STUN server only and one with only a
// DERP node. Add artificial latency of 300ms to the DERP region, and test
Expand Down
7 changes: 7 additions & 0 deletions wgengine/magicsock/magicsock.go
Original file line number Diff line number Diff line change
Expand Up @@ -443,6 +443,13 @@ func NewConn(opts Options) (*Conn, error) {
SkipExternalNetwork: inTest(),
PortMapper: c.portMapper,
UseDNSCache: true,
GetDERPHeaders: func() http.Header {
h := c.derpHeader.Load()
if h == nil {
return nil
}
return h.Clone()
},
}

c.ignoreSTUNPackets()
Expand Down

0 comments on commit 8d4d23e

Please sign in to comment.