2021-08-03 16:31:20 +01:00
|
|
|
// Copyright (c) 2021 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 resolver
|
|
|
|
|
|
|
|
import (
|
2022-04-18 20:50:26 +01:00
|
|
|
"flag"
|
2021-08-03 16:31:20 +01:00
|
|
|
"fmt"
|
|
|
|
"reflect"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2021-09-19 01:34:33 +01:00
|
|
|
dns "golang.org/x/net/dns/dnsmessage"
|
2021-10-15 03:39:11 +01:00
|
|
|
"tailscale.com/hostinfo"
|
2021-08-03 14:56:31 +01:00
|
|
|
"tailscale.com/types/dnstype"
|
2021-08-03 16:31:20 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func (rr resolverAndDelay) String() string {
|
2021-08-03 14:56:31 +01:00
|
|
|
return fmt.Sprintf("%v+%v", rr.name, rr.startDelay)
|
2021-08-03 16:31:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestResolversWithDelays(t *testing.T) {
|
|
|
|
// query
|
2022-05-03 22:41:58 +01:00
|
|
|
q := func(ss ...string) (ipps []*dnstype.Resolver) {
|
2022-04-19 05:58:00 +01:00
|
|
|
for _, host := range ss {
|
2022-05-03 22:41:58 +01:00
|
|
|
ipps = append(ipps, &dnstype.Resolver{Addr: host})
|
2021-08-03 16:31:20 +01:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// output
|
|
|
|
o := func(ss ...string) (rr []resolverAndDelay) {
|
|
|
|
for _, s := range ss {
|
|
|
|
var d time.Duration
|
2022-03-19 19:42:46 +00:00
|
|
|
s, durStr, hasPlus := strings.Cut(s, "+")
|
|
|
|
if hasPlus {
|
2021-08-03 16:31:20 +01:00
|
|
|
var err error
|
2022-03-19 19:42:46 +00:00
|
|
|
d, err = time.ParseDuration(durStr)
|
2021-08-03 16:31:20 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("parsing duration in %q: %v", s, err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rr = append(rr, resolverAndDelay{
|
2022-05-03 22:41:58 +01:00
|
|
|
name: &dnstype.Resolver{Addr: s},
|
2021-08-03 16:31:20 +01:00
|
|
|
startDelay: d,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
2022-05-03 22:41:58 +01:00
|
|
|
in []*dnstype.Resolver
|
2021-08-03 16:31:20 +01:00
|
|
|
want []resolverAndDelay
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "unknown-no-delays",
|
2022-04-19 05:58:00 +01:00
|
|
|
in: q("1.2.3.4", "2.3.4.5"),
|
|
|
|
want: o("1.2.3.4", "2.3.4.5"),
|
2021-08-03 16:31:20 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "google-all-ipv4",
|
2022-04-19 05:58:00 +01:00
|
|
|
in: q("8.8.8.8", "8.8.4.4"),
|
|
|
|
want: o("https://dns.google/dns-query", "8.8.8.8+0.5s", "8.8.4.4+0.7s"),
|
2021-08-03 16:31:20 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "google-only-ipv6",
|
2022-04-19 05:58:00 +01:00
|
|
|
in: q("2001:4860:4860::8888", "2001:4860:4860::8844"),
|
|
|
|
want: o("https://dns.google/dns-query", "2001:4860:4860::8888+0.5s", "2001:4860:4860::8844+0.7s"),
|
2021-08-03 16:31:20 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "google-all-four",
|
2022-04-19 05:58:00 +01:00
|
|
|
in: q("8.8.8.8", "8.8.4.4", "2001:4860:4860::8888", "2001:4860:4860::8844"),
|
|
|
|
want: o("https://dns.google/dns-query", "8.8.8.8+0.5s", "8.8.4.4+0.7s", "2001:4860:4860::8888+0.5s", "2001:4860:4860::8844+0.7s"),
|
2021-08-03 16:31:20 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "quad9-one-v4-one-v6",
|
2022-04-19 05:58:00 +01:00
|
|
|
in: q("9.9.9.9", "2620:fe::fe"),
|
|
|
|
want: o("https://dns.quad9.net/dns-query", "9.9.9.9+0.5s", "2620:fe::fe+0.5s"),
|
2021-08-03 16:31:20 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := resolversWithDelays(tt.in)
|
|
|
|
if !reflect.DeepEqual(got, tt.want) {
|
|
|
|
t.Errorf("got %v; want %v", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-09-19 01:34:33 +01:00
|
|
|
|
|
|
|
func TestGetRCode(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
packet []byte
|
|
|
|
want dns.RCode
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "empty",
|
|
|
|
packet: []byte{},
|
|
|
|
want: dns.RCode(5),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "too-short",
|
|
|
|
packet: []byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0},
|
|
|
|
want: dns.RCode(5),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "noerror",
|
|
|
|
packet: []byte{0xC4, 0xFE, 0x81, 0xA0, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01},
|
|
|
|
want: dns.RCode(0),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "refused",
|
|
|
|
packet: []byte{0xee, 0xa1, 0x81, 0x05, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01},
|
|
|
|
want: dns.RCode(5),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "nxdomain",
|
|
|
|
packet: []byte{0x34, 0xf4, 0x81, 0x83, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01},
|
|
|
|
want: dns.RCode(3),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
got := getRCode(tt.packet)
|
|
|
|
if got != tt.want {
|
|
|
|
t.Errorf("got %d; want %d", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2021-10-15 03:39:11 +01:00
|
|
|
|
|
|
|
func TestMaxDoHInFlight(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
goos string
|
|
|
|
ver string
|
|
|
|
want int
|
|
|
|
}{
|
|
|
|
{"ios", "", 10},
|
|
|
|
{"ios", "1532", 10},
|
|
|
|
{"ios", "9.3.2", 10},
|
|
|
|
{"ios", "14.3.2", 10},
|
|
|
|
{"ios", "15.3.2", 1000},
|
|
|
|
{"ios", "20.3.2", 1000},
|
|
|
|
{"android", "", 1000},
|
|
|
|
{"darwin", "", 1000},
|
|
|
|
{"linux", "", 1000},
|
|
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
|
|
t.Run(fmt.Sprintf("%s-%s", tc.goos, tc.ver), func(t *testing.T) {
|
|
|
|
hostinfo.SetOSVersion(tc.ver)
|
|
|
|
got := maxDoHInFlight(tc.goos)
|
|
|
|
if got != tc.want {
|
|
|
|
t.Errorf("got %d; want %d", got, tc.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2021-12-19 03:03:38 +00:00
|
|
|
|
2022-04-18 20:50:26 +01:00
|
|
|
var testDNS = flag.Bool("test-dns", false, "run tests that require a working DNS server")
|
|
|
|
|
|
|
|
func TestGetKnownDoHClientForProvider(t *testing.T) {
|
|
|
|
var fwd forwarder
|
|
|
|
c, ok := fwd.getKnownDoHClientForProvider("https://dns.google/dns-query")
|
|
|
|
if !ok {
|
|
|
|
t.Fatal("not found")
|
|
|
|
}
|
|
|
|
if !*testDNS {
|
|
|
|
t.Skip("skipping without --test-dns")
|
|
|
|
}
|
|
|
|
res, err := c.Head("https://dns.google/")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer res.Body.Close()
|
|
|
|
t.Logf("Got: %+v", res)
|
|
|
|
}
|
|
|
|
|
2021-12-19 03:03:38 +00:00
|
|
|
func BenchmarkNameFromQuery(b *testing.B) {
|
|
|
|
builder := dns.NewBuilder(nil, dns.Header{})
|
|
|
|
builder.StartQuestions()
|
|
|
|
builder.Question(dns.Question{
|
|
|
|
Name: dns.MustNewName("foo.example."),
|
|
|
|
Type: dns.TypeA,
|
|
|
|
Class: dns.ClassINET,
|
|
|
|
})
|
|
|
|
msg, err := builder.Finish()
|
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
b.ResetTimer()
|
|
|
|
b.ReportAllocs()
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
_, err := nameFromQuery(msg)
|
|
|
|
if err != nil {
|
|
|
|
b.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|