mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-08 09:07:44 +00:00
e8ae355bb8
Previously the client had heuristics to calculate which DNS search domains to set, based on the peers' names. Unfortunately that prevented us from doing some things we wanted to do server-side related to node sharing. So, bump MapRequest.Version to 9 to signal that the client only uses the explicitly configured DNS search domains and doesn't augment it with its own list. Updates tailscale/corp#1026 Signed-off-by: Brad Fitzpatrick <bradfitz@tailscale.com>
146 lines
3.6 KiB
Go
146 lines
3.6 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 tsdns
|
|
|
|
import (
|
|
"sort"
|
|
"strings"
|
|
|
|
"inet.af/netaddr"
|
|
)
|
|
|
|
// Map is all the data Resolver needs to resolve DNS queries within the Tailscale network.
|
|
type Map struct {
|
|
// nameToIP is a mapping of Tailscale domain names to their IP addresses.
|
|
// For example, monitoring.tailscale.us -> 100.64.0.1.
|
|
nameToIP map[string]netaddr.IP
|
|
// ipToName is the inverse of nameToIP.
|
|
ipToName map[netaddr.IP]string
|
|
// names are the keys of nameToIP in sorted order.
|
|
names []string
|
|
// rootDomains are the domains whose subdomains should always
|
|
// be resolved locally to prevent leakage of sensitive names.
|
|
rootDomains []string // e.g. "user.provider.beta.tailscale.net."
|
|
}
|
|
|
|
// NewMap returns a new Map with name to address mapping given by nameToIP.
|
|
//
|
|
// rootDomains are the domains whose subdomains should always be
|
|
// resolved locally to prevent leakage of sensitive names. They should
|
|
// end in a period ("user-foo.tailscale.net.").
|
|
func NewMap(initNameToIP map[string]netaddr.IP, rootDomains []string) *Map {
|
|
// TODO(dmytro): we have to allocate names and ipToName, but nameToIP can be avoided.
|
|
// It is here because control sends us names not in canonical form. Change this.
|
|
names := make([]string, 0, len(initNameToIP))
|
|
nameToIP := make(map[string]netaddr.IP, len(initNameToIP))
|
|
ipToName := make(map[netaddr.IP]string, len(initNameToIP))
|
|
|
|
for name, ip := range initNameToIP {
|
|
if len(name) == 0 {
|
|
// Nothing useful can be done with empty names.
|
|
continue
|
|
}
|
|
if name[len(name)-1] != '.' {
|
|
name += "."
|
|
}
|
|
names = append(names, name)
|
|
nameToIP[name] = ip
|
|
ipToName[ip] = name
|
|
}
|
|
sort.Strings(names)
|
|
|
|
return &Map{
|
|
nameToIP: nameToIP,
|
|
ipToName: ipToName,
|
|
names: names,
|
|
|
|
rootDomains: rootDomains,
|
|
}
|
|
}
|
|
|
|
func printSingleNameIP(buf *strings.Builder, name string, ip netaddr.IP) {
|
|
buf.WriteString(name)
|
|
buf.WriteByte('\t')
|
|
buf.WriteString(ip.String())
|
|
buf.WriteByte('\n')
|
|
}
|
|
|
|
func (m *Map) Pretty() string {
|
|
buf := new(strings.Builder)
|
|
for _, name := range m.names {
|
|
printSingleNameIP(buf, name, m.nameToIP[name])
|
|
}
|
|
return buf.String()
|
|
}
|
|
|
|
func (m *Map) PrettyDiffFrom(old *Map) string {
|
|
var (
|
|
oldNameToIP map[string]netaddr.IP
|
|
newNameToIP map[string]netaddr.IP
|
|
oldNames []string
|
|
newNames []string
|
|
)
|
|
if old != nil {
|
|
oldNameToIP = old.nameToIP
|
|
oldNames = old.names
|
|
}
|
|
if m != nil {
|
|
newNameToIP = m.nameToIP
|
|
newNames = m.names
|
|
}
|
|
|
|
buf := new(strings.Builder)
|
|
|
|
for len(oldNames) > 0 && len(newNames) > 0 {
|
|
var name string
|
|
|
|
newName, oldName := newNames[0], oldNames[0]
|
|
switch {
|
|
case oldName < newName:
|
|
name = oldName
|
|
oldNames = oldNames[1:]
|
|
case oldName > newName:
|
|
name = newName
|
|
newNames = newNames[1:]
|
|
case oldNames[0] == newNames[0]:
|
|
name = oldNames[0]
|
|
oldNames = oldNames[1:]
|
|
newNames = newNames[1:]
|
|
}
|
|
|
|
ipOld, inOld := oldNameToIP[name]
|
|
ipNew, inNew := newNameToIP[name]
|
|
switch {
|
|
case !inOld:
|
|
buf.WriteByte('+')
|
|
printSingleNameIP(buf, name, ipNew)
|
|
case !inNew:
|
|
buf.WriteByte('-')
|
|
printSingleNameIP(buf, name, ipOld)
|
|
case ipOld != ipNew:
|
|
buf.WriteByte('-')
|
|
printSingleNameIP(buf, name, ipOld)
|
|
buf.WriteByte('+')
|
|
printSingleNameIP(buf, name, ipNew)
|
|
}
|
|
}
|
|
|
|
for _, name := range oldNames {
|
|
if _, ok := newNameToIP[name]; !ok {
|
|
buf.WriteByte('-')
|
|
printSingleNameIP(buf, name, oldNameToIP[name])
|
|
}
|
|
}
|
|
|
|
for _, name := range newNames {
|
|
if _, ok := oldNameToIP[name]; !ok {
|
|
buf.WriteByte('+')
|
|
printSingleNameIP(buf, name, newNameToIP[name])
|
|
}
|
|
}
|
|
|
|
return buf.String()
|
|
}
|