mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-10 01:53:49 +00:00
138055dd70
Upstream wireguard-go decided to use errors.Is(err, net.ErrClosed) instead of checking the error string. It also provided an unsafe linknamed version of net.ErrClosed for clients running Go 1.15. Switch to that. This reduces the time required for the wgengine/magicsock tests on my machine from ~35s back to the ~13s it was before 456cf8a3765948d6f1992162993eaf3844371592. Signed-off-by: Josh Bleecher Snyder <josh@tailscale.com>
137 lines
3.0 KiB
Go
137 lines
3.0 KiB
Go
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Package stuntest provides a STUN test server.
|
|
package stuntest
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"testing"
|
|
|
|
"inet.af/netaddr"
|
|
"tailscale.com/net/stun"
|
|
"tailscale.com/tailcfg"
|
|
"tailscale.com/types/nettype"
|
|
)
|
|
|
|
type stunStats struct {
|
|
mu sync.Mutex
|
|
readIPv4 int
|
|
readIPv6 int
|
|
}
|
|
|
|
func Serve(t *testing.T) (addr *net.UDPAddr, cleanupFn func()) {
|
|
return ServeWithPacketListener(t, nettype.Std{})
|
|
}
|
|
|
|
func ServeWithPacketListener(t *testing.T, ln nettype.PacketListener) (addr *net.UDPAddr, cleanupFn func()) {
|
|
t.Helper()
|
|
|
|
// TODO(crawshaw): use stats to test re-STUN logic
|
|
var stats stunStats
|
|
|
|
pc, err := ln.ListenPacket(context.Background(), "udp4", ":0")
|
|
if err != nil {
|
|
t.Fatalf("failed to open STUN listener: %v", err)
|
|
}
|
|
addr = pc.LocalAddr().(*net.UDPAddr)
|
|
if len(addr.IP) == 0 || addr.IP.IsUnspecified() {
|
|
addr.IP = net.ParseIP("127.0.0.1")
|
|
}
|
|
doneCh := make(chan struct{})
|
|
go runSTUN(t, pc, &stats, doneCh)
|
|
return addr, func() {
|
|
pc.Close()
|
|
<-doneCh
|
|
}
|
|
}
|
|
|
|
func runSTUN(t *testing.T, pc net.PacketConn, stats *stunStats, done chan<- struct{}) {
|
|
defer close(done)
|
|
|
|
var buf [64 << 10]byte
|
|
for {
|
|
n, addr, err := pc.ReadFrom(buf[:])
|
|
if err != nil {
|
|
// TODO: when we switch to Go 1.16, replace this with errors.Is(err, net.ErrClosed)
|
|
if strings.Contains(err.Error(), "closed network connection") {
|
|
t.Logf("STUN server shutdown")
|
|
return
|
|
}
|
|
continue
|
|
}
|
|
ua := addr.(*net.UDPAddr)
|
|
pkt := buf[:n]
|
|
if !stun.Is(pkt) {
|
|
continue
|
|
}
|
|
txid, err := stun.ParseBindingRequest(pkt)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
stats.mu.Lock()
|
|
if ua.IP.To4() != nil {
|
|
stats.readIPv4++
|
|
} else {
|
|
stats.readIPv6++
|
|
}
|
|
stats.mu.Unlock()
|
|
|
|
res := stun.Response(txid, ua.IP, uint16(ua.Port))
|
|
if _, err := pc.WriteTo(res, addr); err != nil {
|
|
t.Logf("STUN server write failed: %v", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func DERPMapOf(stun ...string) *tailcfg.DERPMap {
|
|
m := &tailcfg.DERPMap{
|
|
Regions: map[int]*tailcfg.DERPRegion{},
|
|
}
|
|
for i, hostPortStr := range stun {
|
|
regionID := i + 1
|
|
host, portStr, err := net.SplitHostPort(hostPortStr)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("bogus STUN hostport: %q", hostPortStr))
|
|
}
|
|
port, err := strconv.Atoi(portStr)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("bogus port %q in %q", portStr, hostPortStr))
|
|
}
|
|
var ipv4, ipv6 string
|
|
ip, err := netaddr.ParseIP(host)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("bogus non-IP STUN host %q in %q", host, hostPortStr))
|
|
}
|
|
if ip.Is4() {
|
|
ipv4 = host
|
|
ipv6 = "none"
|
|
}
|
|
if ip.Is6() {
|
|
ipv6 = host
|
|
ipv4 = "none"
|
|
}
|
|
node := &tailcfg.DERPNode{
|
|
Name: fmt.Sprint(regionID) + "a",
|
|
RegionID: regionID,
|
|
HostName: fmt.Sprintf("d%d.invalid", regionID),
|
|
IPv4: ipv4,
|
|
IPv6: ipv6,
|
|
STUNPort: port,
|
|
STUNOnly: true,
|
|
}
|
|
m.Regions[regionID] = &tailcfg.DERPRegion{
|
|
RegionID: regionID,
|
|
Nodes: []*tailcfg.DERPNode{node},
|
|
}
|
|
}
|
|
return m
|
|
}
|