mirror of
https://github.com/tailscale/tailscale.git
synced 2025-04-03 23:05:50 +00:00
151 lines
4.2 KiB
Go
151 lines
4.2 KiB
Go
![]() |
// Copyright (c) Tailscale Inc & AUTHORS
|
||
|
// SPDX-License-Identifier: BSD-3-Clause
|
||
|
|
||
|
package magicsock
|
||
|
|
||
|
import (
|
||
|
"encoding/binary"
|
||
|
"net"
|
||
|
"net/netip"
|
||
|
"testing"
|
||
|
|
||
|
"golang.org/x/net/ipv4"
|
||
|
"golang.org/x/net/ipv6"
|
||
|
"golang.org/x/sys/cpu"
|
||
|
"golang.org/x/sys/unix"
|
||
|
"tailscale.com/disco"
|
||
|
"tailscale.com/net/packet"
|
||
|
"tailscale.com/types/ipproto"
|
||
|
)
|
||
|
|
||
|
func TestDecodeDiscoPacket(t *testing.T) {
|
||
|
mk4 := func(proto ipproto.Proto, src, dst netip.Addr, data []byte) []byte {
|
||
|
if !src.Is4() || !dst.Is4() {
|
||
|
panic("not an IPv4 address")
|
||
|
}
|
||
|
iph := &ipv4.Header{
|
||
|
Version: ipv4.Version,
|
||
|
Len: ipv4.HeaderLen,
|
||
|
TotalLen: ipv4.HeaderLen + len(data),
|
||
|
TTL: 64,
|
||
|
Protocol: int(proto),
|
||
|
Src: net.IP(src.AsSlice()),
|
||
|
Dst: net.IP(dst.AsSlice()),
|
||
|
}
|
||
|
hdr, err := iph.Marshal()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return append(hdr, data...)
|
||
|
}
|
||
|
mk6 := func(proto ipproto.Proto, src, dst netip.Addr, data []byte) []byte {
|
||
|
if !src.Is6() || !dst.Is6() {
|
||
|
panic("not an IPv6 address")
|
||
|
}
|
||
|
// The ipv6 package doesn't have a Marshal method, so just do
|
||
|
// the most basic thing and construct the header manually.
|
||
|
buf := make([]byte, ipv6.HeaderLen, ipv6.HeaderLen+len(data))
|
||
|
buf[0] = 6 << 4 // version
|
||
|
binary.BigEndian.PutUint16(buf[4:6], uint16(len(data)))
|
||
|
buf[6] = byte(proto)
|
||
|
copy(buf[8:24], src.AsSlice())
|
||
|
copy(buf[24:40], dst.AsSlice())
|
||
|
return append(buf, data...)
|
||
|
}
|
||
|
|
||
|
mkUDP := func(srcPort, dstPort uint16, data []byte) []byte {
|
||
|
udp := make([]byte, 8, 8+len(data))
|
||
|
binary.BigEndian.PutUint16(udp[0:2], srcPort)
|
||
|
binary.BigEndian.PutUint16(udp[2:4], dstPort)
|
||
|
binary.BigEndian.PutUint16(udp[4:6], uint16(8+len(data)))
|
||
|
return append(udp, data...)
|
||
|
}
|
||
|
mkUDP4 := func(src, dst netip.AddrPort, data []byte) []byte {
|
||
|
return mk4(ipproto.UDP, src.Addr(), dst.Addr(), mkUDP(src.Port(), dst.Port(), data))
|
||
|
}
|
||
|
mkUDP6 := func(src, dst netip.AddrPort, data []byte) []byte {
|
||
|
return mk6(ipproto.UDP, src.Addr(), dst.Addr(), mkUDP(src.Port(), dst.Port(), data))
|
||
|
}
|
||
|
|
||
|
ip4 := netip.MustParseAddrPort("127.0.0.10:12345")
|
||
|
ip4_2 := netip.MustParseAddrPort("127.0.0.99:54321")
|
||
|
ip6 := netip.MustParseAddrPort("[::1]:12345")
|
||
|
|
||
|
testCases := []struct {
|
||
|
name string
|
||
|
in []byte
|
||
|
is6 bool
|
||
|
want bool
|
||
|
}{
|
||
|
{
|
||
|
name: "too_short_4",
|
||
|
in: mkUDP4(ip4, ip4_2, append([]byte(disco.Magic), 0x00, 0x00)),
|
||
|
is6: false,
|
||
|
want: false,
|
||
|
},
|
||
|
{
|
||
|
name: "too_short_6",
|
||
|
in: mkUDP6(ip6, ip6, append([]byte(disco.Magic), 0x00, 0x00)),
|
||
|
is6: true,
|
||
|
want: false,
|
||
|
},
|
||
|
{
|
||
|
name: "valid_4",
|
||
|
in: mkUDP4(ip4, ip4_2, testDiscoPacket),
|
||
|
is6: false,
|
||
|
want: true,
|
||
|
},
|
||
|
{
|
||
|
name: "valid_6",
|
||
|
in: mkUDP6(ip6, ip6, testDiscoPacket),
|
||
|
is6: true,
|
||
|
want: true,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
for _, tc := range testCases {
|
||
|
t.Run(tc.name, func(t *testing.T) {
|
||
|
var pkt packet.Parsed
|
||
|
got := decodeDiscoPacket(&pkt, t.Logf, tc.in, tc.is6)
|
||
|
if got != tc.want {
|
||
|
t.Errorf("got %v; want %v", got, tc.want)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func TestEthernetProto(t *testing.T) {
|
||
|
htons := func(x uint16) int {
|
||
|
// Network byte order is big-endian; write the value as
|
||
|
// big-endian to a byte slice and read it back in the native
|
||
|
// endian-ness. This is a no-op on a big-endian platform and a
|
||
|
// byte swap on a little-endian platform.
|
||
|
var b [2]byte
|
||
|
binary.BigEndian.PutUint16(b[:], x)
|
||
|
return int(binary.NativeEndian.Uint16(b[:]))
|
||
|
}
|
||
|
|
||
|
if v4 := ethernetProtoIPv4(); v4 != htons(unix.ETH_P_IP) {
|
||
|
t.Errorf("ethernetProtoIPv4 = 0x%04x; want 0x%04x", v4, htons(unix.ETH_P_IP))
|
||
|
}
|
||
|
if v6 := ethernetProtoIPv6(); v6 != htons(unix.ETH_P_IPV6) {
|
||
|
t.Errorf("ethernetProtoIPv6 = 0x%04x; want 0x%04x", v6, htons(unix.ETH_P_IPV6))
|
||
|
}
|
||
|
|
||
|
// As a way to verify that the htons function is working correctly,
|
||
|
// assert that the ETH_P_IP value returned from our function matches
|
||
|
// the value defined in the unix package based on whether the host is
|
||
|
// big-endian (network byte order) or little-endian.
|
||
|
if cpu.IsBigEndian {
|
||
|
if v4 := ethernetProtoIPv4(); v4 != unix.ETH_P_IP {
|
||
|
t.Errorf("ethernetProtoIPv4 = 0x%04x; want 0x%04x", v4, unix.ETH_P_IP)
|
||
|
}
|
||
|
} else {
|
||
|
if v4 := ethernetProtoIPv4(); v4 == unix.ETH_P_IP {
|
||
|
t.Errorf("ethernetProtoIPv4 = 0x%04x; want 0x%04x", v4, htons(unix.ETH_P_IP))
|
||
|
} else {
|
||
|
t.Logf("ethernetProtoIPv4 = 0x%04x, correctly different from 0x%04x", v4, unix.ETH_P_IP)
|
||
|
}
|
||
|
}
|
||
|
}
|