2023-01-27 21:37:20 +00:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2021-11-30 23:53:34 +00:00
|
|
|
|
|
|
|
package tsdial
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-12-02 19:10:35 +00:00
|
|
|
"errors"
|
2021-11-30 23:53:34 +00:00
|
|
|
"fmt"
|
|
|
|
"net"
|
2022-07-26 04:55:44 +01:00
|
|
|
"net/netip"
|
2021-11-30 23:53:34 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"tailscale.com/types/netmap"
|
|
|
|
"tailscale.com/util/dnsname"
|
|
|
|
)
|
|
|
|
|
2021-12-02 19:10:35 +00:00
|
|
|
// dnsMap maps MagicDNS names (both base + FQDN) to their first IP.
|
2021-11-30 23:53:34 +00:00
|
|
|
// It must not be mutated once created.
|
|
|
|
//
|
|
|
|
// Example keys are "foo.domain.tld.beta.tailscale.net" and "foo",
|
2022-03-25 21:27:22 +00:00
|
|
|
// both without trailing dots, and both always lowercase.
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
type dnsMap map[string]netip.Addr
|
2021-11-30 23:53:34 +00:00
|
|
|
|
2022-03-25 21:27:22 +00:00
|
|
|
// canonMapKey canonicalizes its input s to be a dnsMap map key.
|
|
|
|
func canonMapKey(s string) string {
|
|
|
|
return strings.ToLower(strings.TrimSuffix(s, "."))
|
|
|
|
}
|
|
|
|
|
2021-12-02 19:10:35 +00:00
|
|
|
func dnsMapFromNetworkMap(nm *netmap.NetworkMap) dnsMap {
|
|
|
|
if nm == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ret := make(dnsMap)
|
2021-11-30 23:53:34 +00:00
|
|
|
suffix := nm.MagicDNSSuffix()
|
|
|
|
have4 := false
|
2023-09-17 18:53:23 +01:00
|
|
|
addrs := nm.GetAddresses()
|
|
|
|
if nm.Name != "" && addrs.Len() > 0 {
|
|
|
|
ip := addrs.At(0).Addr()
|
2022-03-25 21:27:22 +00:00
|
|
|
ret[canonMapKey(nm.Name)] = ip
|
2021-11-30 23:53:34 +00:00
|
|
|
if dnsname.HasSuffix(nm.Name, suffix) {
|
2022-03-25 21:27:22 +00:00
|
|
|
ret[canonMapKey(dnsname.TrimSuffix(nm.Name, suffix))] = ip
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
2023-09-17 18:53:23 +01:00
|
|
|
for i := range addrs.LenIter() {
|
|
|
|
if addrs.At(i).Addr().Is4() {
|
2021-11-30 23:53:34 +00:00
|
|
|
have4 = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, p := range nm.Peers {
|
2023-08-18 15:57:44 +01:00
|
|
|
if p.Name() == "" {
|
2021-11-30 23:53:34 +00:00
|
|
|
continue
|
|
|
|
}
|
2023-08-18 15:57:44 +01:00
|
|
|
for i := range p.Addresses().LenIter() {
|
|
|
|
a := p.Addresses().At(i)
|
2022-07-25 04:08:42 +01:00
|
|
|
ip := a.Addr()
|
2021-11-30 23:53:34 +00:00
|
|
|
if ip.Is4() && !have4 {
|
|
|
|
continue
|
|
|
|
}
|
2023-08-18 15:57:44 +01:00
|
|
|
ret[canonMapKey(p.Name())] = ip
|
|
|
|
if dnsname.HasSuffix(p.Name(), suffix) {
|
|
|
|
ret[canonMapKey(dnsname.TrimSuffix(p.Name(), suffix))] = ip
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, rec := range nm.DNS.ExtraRecords {
|
|
|
|
if rec.Type != "" {
|
|
|
|
continue
|
|
|
|
}
|
2022-07-26 04:55:44 +01:00
|
|
|
ip, err := netip.ParseAddr(rec.Value)
|
2021-11-30 23:53:34 +00:00
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
2022-03-25 21:27:22 +00:00
|
|
|
ret[canonMapKey(rec.Name)] = ip
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2021-12-02 19:10:35 +00:00
|
|
|
// errUnresolved is a sentinel error returned by dnsMap.resolveMemory.
|
|
|
|
var errUnresolved = errors.New("address well formed but not resolved")
|
|
|
|
|
|
|
|
func splitHostPort(addr string) (host string, port uint16, err error) {
|
|
|
|
host, portStr, err := net.SplitHostPort(addr)
|
|
|
|
if err != nil {
|
|
|
|
return "", 0, err
|
|
|
|
}
|
|
|
|
port16, err := strconv.ParseUint(portStr, 10, 16)
|
|
|
|
if err != nil {
|
|
|
|
return "", 0, fmt.Errorf("invalid port in address %q", addr)
|
|
|
|
}
|
|
|
|
return host, uint16(port16), nil
|
|
|
|
}
|
|
|
|
|
2021-11-30 23:53:34 +00:00
|
|
|
// Resolve resolves addr into an IP:port using first the MagicDNS contents
|
|
|
|
// of m, else using the system resolver.
|
2021-12-02 19:10:35 +00:00
|
|
|
//
|
|
|
|
// The error is [exactly] errUnresolved if the addr is a name that isn't known
|
|
|
|
// in the map.
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
func (m dnsMap) resolveMemory(ctx context.Context, network, addr string) (_ netip.AddrPort, err error) {
|
2021-12-02 19:10:35 +00:00
|
|
|
host, port, err := splitHostPort(addr)
|
2021-11-30 23:53:34 +00:00
|
|
|
if err != nil {
|
2021-12-02 19:10:35 +00:00
|
|
|
// addr malformed or invalid port.
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
return netip.AddrPort{}, err
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
2022-07-26 04:55:44 +01:00
|
|
|
if ip, err := netip.ParseAddr(host); err == nil {
|
2021-12-02 19:10:35 +00:00
|
|
|
// addr was literal ip:port.
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
return netip.AddrPortFrom(ip, port), nil
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Host is not an IP, so assume it's a DNS name.
|
|
|
|
|
|
|
|
// Try MagicDNS first, otherwise a real DNS lookup.
|
2022-03-25 21:27:22 +00:00
|
|
|
ip := m[canonMapKey(host)]
|
2022-07-25 04:08:42 +01:00
|
|
|
if ip.IsValid() {
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
return netip.AddrPortFrom(ip, port), nil
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|
|
|
|
|
all: convert more code to use net/netip directly
perl -i -npe 's,netaddr.IPPrefixFrom,netip.PrefixFrom,' $(git grep -l -F netaddr.)
perl -i -npe 's,netaddr.IPPortFrom,netip.AddrPortFrom,' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPrefix,netip.Prefix,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPPort,netip.AddrPort,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IP\b,netip.Addr,g' $(git grep -l -F netaddr. )
perl -i -npe 's,netaddr.IPv6Raw\b,netip.AddrFrom16,g' $(git grep -l -F netaddr. )
goimports -w .
Then delete some stuff from the net/netaddr shim package which is no
longer neeed.
Updates #5162
Change-Id: Ia7a86893fe21c7e3ee1ec823e8aba288d4566cd8
Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
2022-07-26 05:14:09 +01:00
|
|
|
return netip.AddrPort{}, errUnresolved
|
2021-11-30 23:53:34 +00:00
|
|
|
}
|