2023-01-27 21:37:20 +00:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
2021-02-26 16:28:31 +00:00
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"expvar"
|
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2022-08-04 18:43:49 +01:00
|
|
|
|
|
|
|
"tailscale.com/syncs"
|
2023-03-03 18:15:56 +00:00
|
|
|
"tailscale.com/util/slicesx"
|
2021-02-26 16:28:31 +00:00
|
|
|
)
|
|
|
|
|
2022-09-02 19:48:30 +01:00
|
|
|
const refreshTimeout = time.Minute
|
2021-02-26 16:28:31 +00:00
|
|
|
|
2022-09-02 19:48:30 +01:00
|
|
|
type dnsEntryMap map[string][]net.IP
|
|
|
|
|
|
|
|
var (
|
|
|
|
dnsCache syncs.AtomicValue[dnsEntryMap]
|
|
|
|
dnsCacheBytes syncs.AtomicValue[[]byte] // of JSON
|
|
|
|
unpublishedDNSCache syncs.AtomicValue[dnsEntryMap]
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
bootstrapDNSRequests = expvar.NewInt("counter_bootstrap_dns_requests")
|
|
|
|
publishedDNSHits = expvar.NewInt("counter_bootstrap_dns_published_hits")
|
|
|
|
publishedDNSMisses = expvar.NewInt("counter_bootstrap_dns_published_misses")
|
|
|
|
unpublishedDNSHits = expvar.NewInt("counter_bootstrap_dns_unpublished_hits")
|
|
|
|
unpublishedDNSMisses = expvar.NewInt("counter_bootstrap_dns_unpublished_misses")
|
|
|
|
)
|
2021-02-26 16:28:31 +00:00
|
|
|
|
|
|
|
func refreshBootstrapDNSLoop() {
|
2022-09-02 19:48:30 +01:00
|
|
|
if *bootstrapDNS == "" && *unpublishedDNS == "" {
|
2021-02-26 16:28:31 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
refreshBootstrapDNS()
|
2022-09-02 19:48:30 +01:00
|
|
|
refreshUnpublishedDNS()
|
2021-02-26 16:28:31 +00:00
|
|
|
time.Sleep(10 * time.Minute)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func refreshBootstrapDNS() {
|
|
|
|
if *bootstrapDNS == "" {
|
|
|
|
return
|
|
|
|
}
|
2022-09-02 19:48:30 +01:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), refreshTimeout)
|
|
|
|
defer cancel()
|
|
|
|
dnsEntries := resolveList(ctx, strings.Split(*bootstrapDNS, ","))
|
2023-03-03 04:36:12 +00:00
|
|
|
// Randomize the order of the IPs for each name to avoid the client biasing
|
|
|
|
// to IPv6
|
|
|
|
for k := range dnsEntries {
|
|
|
|
ips := dnsEntries[k]
|
2023-03-03 18:15:56 +00:00
|
|
|
slicesx.Shuffle(ips)
|
2023-03-03 04:36:12 +00:00
|
|
|
dnsEntries[k] = ips
|
|
|
|
}
|
2022-09-02 19:48:30 +01:00
|
|
|
j, err := json.MarshalIndent(dnsEntries, "", "\t")
|
|
|
|
if err != nil {
|
|
|
|
// leave the old values in place
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
dnsCache.Store(dnsEntries)
|
|
|
|
dnsCacheBytes.Store(j)
|
|
|
|
}
|
|
|
|
|
|
|
|
func refreshUnpublishedDNS() {
|
|
|
|
if *unpublishedDNS == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), refreshTimeout)
|
2021-02-26 16:28:31 +00:00
|
|
|
defer cancel()
|
2022-09-02 19:48:30 +01:00
|
|
|
|
|
|
|
dnsEntries := resolveList(ctx, strings.Split(*unpublishedDNS, ","))
|
|
|
|
unpublishedDNSCache.Store(dnsEntries)
|
|
|
|
}
|
|
|
|
|
|
|
|
func resolveList(ctx context.Context, names []string) dnsEntryMap {
|
|
|
|
dnsEntries := make(dnsEntryMap)
|
|
|
|
|
2021-02-26 16:28:31 +00:00
|
|
|
var r net.Resolver
|
|
|
|
for _, name := range names {
|
|
|
|
addrs, err := r.LookupIP(ctx, "ip", name)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("bootstrap DNS lookup %q: %v", name, err)
|
|
|
|
continue
|
|
|
|
}
|
2022-02-11 20:30:36 +00:00
|
|
|
dnsEntries[name] = addrs
|
2021-02-26 16:28:31 +00:00
|
|
|
}
|
2022-09-02 19:48:30 +01:00
|
|
|
return dnsEntries
|
2022-02-11 20:30:36 +00:00
|
|
|
}
|
2021-02-26 16:28:31 +00:00
|
|
|
|
2022-02-11 20:30:36 +00:00
|
|
|
func handleBootstrapDNS(w http.ResponseWriter, r *http.Request) {
|
|
|
|
bootstrapDNSRequests.Add(1)
|
2022-09-02 19:48:30 +01:00
|
|
|
|
2022-02-11 22:14:04 +00:00
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2022-09-02 19:48:30 +01:00
|
|
|
// Bootstrap DNS requests occur cross-regions, and are randomized per
|
|
|
|
// request, so keeping a connection open is pointlessly expensive.
|
2022-02-11 22:06:53 +00:00
|
|
|
w.Header().Set("Connection", "close")
|
2022-09-02 19:48:30 +01:00
|
|
|
|
|
|
|
// Try answering a query from our hidden map first
|
|
|
|
if q := r.URL.Query().Get("q"); q != "" {
|
|
|
|
if ips, ok := unpublishedDNSCache.Load()[q]; ok && len(ips) > 0 {
|
|
|
|
unpublishedDNSHits.Add(1)
|
|
|
|
|
|
|
|
// Only return the specific query, not everything.
|
|
|
|
m := dnsEntryMap{q: ips}
|
|
|
|
j, err := json.MarshalIndent(m, "", "\t")
|
|
|
|
if err == nil {
|
|
|
|
w.Write(j)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we have a "q" query for a name in the published cache
|
|
|
|
// list, then track whether that's a hit/miss.
|
|
|
|
if m, ok := dnsCache.Load()[q]; ok {
|
|
|
|
if len(m) > 0 {
|
|
|
|
publishedDNSHits.Add(1)
|
|
|
|
} else {
|
|
|
|
publishedDNSMisses.Add(1)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// If it wasn't in either cache, treat this as a query
|
|
|
|
// for the unpublished cache, and thus a cache miss.
|
|
|
|
unpublishedDNSMisses.Add(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fall back to returning the public set of cached DNS names
|
|
|
|
j := dnsCacheBytes.Load()
|
2021-02-26 16:28:31 +00:00
|
|
|
w.Write(j)
|
|
|
|
}
|