mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-05 23:07:44 +00:00
net/dns: fix IsZero and Equal methods on OSConfig
Discovered this while investigating the following issue; I think it's unrelated, but might as well fix it. Also, add a test helper for checking things that have an IsZero method using the reflect package. Updates tailscale/support-escalations#55 Signed-off-by: Andrew Dunham <andrew@du.nham.ca> Change-Id: I57b7adde43bcef9483763b561da173b4c35f49e2
This commit is contained in:
parent
13dee9db7b
commit
7dcf65a10a
@ -8,6 +8,7 @@
|
||||
"errors"
|
||||
"fmt"
|
||||
"net/netip"
|
||||
"slices"
|
||||
"strings"
|
||||
|
||||
"tailscale.com/types/logger"
|
||||
@ -103,10 +104,16 @@ func (o *OSConfig) WriteToBufioWriter(w *bufio.Writer) {
|
||||
}
|
||||
|
||||
func (o OSConfig) IsZero() bool {
|
||||
return len(o.Nameservers) == 0 && len(o.SearchDomains) == 0 && len(o.MatchDomains) == 0
|
||||
return len(o.Hosts) == 0 &&
|
||||
len(o.Nameservers) == 0 &&
|
||||
len(o.SearchDomains) == 0 &&
|
||||
len(o.MatchDomains) == 0
|
||||
}
|
||||
|
||||
func (a OSConfig) Equal(b OSConfig) bool {
|
||||
if len(a.Hosts) != len(b.Hosts) {
|
||||
return false
|
||||
}
|
||||
if len(a.Nameservers) != len(b.Nameservers) {
|
||||
return false
|
||||
}
|
||||
@ -117,6 +124,15 @@ func (a OSConfig) Equal(b OSConfig) bool {
|
||||
return false
|
||||
}
|
||||
|
||||
for i := range a.Hosts {
|
||||
ha, hb := a.Hosts[i], b.Hosts[i]
|
||||
if ha.Addr != hb.Addr {
|
||||
return false
|
||||
}
|
||||
if !slices.Equal(ha.Hosts, hb.Hosts) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
for i := range a.Nameservers {
|
||||
if a.Nameservers[i] != b.Nameservers[i] {
|
||||
return false
|
||||
|
@ -6,8 +6,10 @@
|
||||
import (
|
||||
"fmt"
|
||||
"net/netip"
|
||||
"reflect"
|
||||
"testing"
|
||||
|
||||
"tailscale.com/tstest"
|
||||
"tailscale.com/util/dnsname"
|
||||
)
|
||||
|
||||
@ -41,3 +43,13 @@ func TestOSConfigPrintable(t *testing.T) {
|
||||
t.Errorf("format mismatch:\n got: %s\n want: %s", s, expected)
|
||||
}
|
||||
}
|
||||
|
||||
func TestIsZero(t *testing.T) {
|
||||
tstest.CheckIsZero[OSConfig](t, map[reflect.Type]any{
|
||||
reflect.TypeFor[dnsname.FQDN](): dnsname.FQDN("foo.bar."),
|
||||
reflect.TypeFor[*HostEntry](): &HostEntry{
|
||||
Addr: netip.AddrFrom4([4]byte{100, 1, 2, 3}),
|
||||
Hosts: []string{"foo", "bar"},
|
||||
},
|
||||
})
|
||||
}
|
||||
|
114
tstest/reflect.go
Normal file
114
tstest/reflect.go
Normal file
@ -0,0 +1,114 @@
|
||||
// Copyright (c) Tailscale Inc & AUTHORS
|
||||
// SPDX-License-Identifier: BSD-3-Clause
|
||||
|
||||
package tstest
|
||||
|
||||
import (
|
||||
"net/netip"
|
||||
"reflect"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"tailscale.com/types/ptr"
|
||||
)
|
||||
|
||||
// IsZeroable is the interface for things with an IsZero method.
|
||||
type IsZeroable interface {
|
||||
IsZero() bool
|
||||
}
|
||||
|
||||
var (
|
||||
netipAddrType = reflect.TypeFor[netip.Addr]()
|
||||
netipAddrPortType = reflect.TypeFor[netip.AddrPort]()
|
||||
netipPrefixType = reflect.TypeFor[netip.Prefix]()
|
||||
timeType = reflect.TypeFor[time.Time]()
|
||||
timePtrType = reflect.TypeFor[*time.Time]()
|
||||
)
|
||||
|
||||
// CheckIsZero checks that the IsZero method of a given type functions
|
||||
// correctly, by instantiating a new value of that type, changing a field, and
|
||||
// then checking that the IsZero method returns false.
|
||||
//
|
||||
// The nonzeroValues map should contain non-zero values for each type that
|
||||
// exists in the type T or any contained types. Basic types like string, bool,
|
||||
// and numeric types are handled automatically.
|
||||
func CheckIsZero[T IsZeroable](t testing.TB, nonzeroValues map[reflect.Type]any) {
|
||||
t.Helper()
|
||||
|
||||
var zero T
|
||||
if !zero.IsZero() {
|
||||
t.Errorf("zero value of %T is not IsZero", zero)
|
||||
return
|
||||
}
|
||||
|
||||
var nonEmptyValue func(t reflect.Type) reflect.Value
|
||||
nonEmptyValue = func(ty reflect.Type) reflect.Value {
|
||||
if v, ok := nonzeroValues[ty]; ok {
|
||||
return reflect.ValueOf(v)
|
||||
}
|
||||
|
||||
switch ty {
|
||||
// Given that we're a networking company, probably fine to have
|
||||
// a special case for netip.Addr :)
|
||||
case netipAddrType:
|
||||
return reflect.ValueOf(netip.MustParseAddr("1.2.3.4"))
|
||||
case netipAddrPortType:
|
||||
return reflect.ValueOf(netip.MustParseAddrPort("1.2.3.4:9999"))
|
||||
case netipPrefixType:
|
||||
return reflect.ValueOf(netip.MustParsePrefix("1.2.3.4/24"))
|
||||
|
||||
case timeType:
|
||||
return reflect.ValueOf(time.Unix(1704067200, 0))
|
||||
case timePtrType:
|
||||
return reflect.ValueOf(ptr.To(time.Unix(1704067200, 0)))
|
||||
}
|
||||
|
||||
switch ty.Kind() {
|
||||
case reflect.String:
|
||||
return reflect.ValueOf("foo").Convert(ty)
|
||||
case reflect.Bool:
|
||||
return reflect.ValueOf(true)
|
||||
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
|
||||
return reflect.ValueOf(int64(-42)).Convert(ty)
|
||||
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
|
||||
return reflect.ValueOf(uint64(42)).Convert(ty)
|
||||
case reflect.Float32, reflect.Float64:
|
||||
return reflect.ValueOf(float64(3.14)).Convert(ty)
|
||||
case reflect.Complex64, reflect.Complex128:
|
||||
return reflect.ValueOf(complex(3.14, 2.71)).Convert(ty)
|
||||
case reflect.Chan:
|
||||
return reflect.MakeChan(ty, 1)
|
||||
|
||||
// For slices, ensure that the slice is non-empty.
|
||||
case reflect.Slice:
|
||||
v := nonEmptyValue(ty.Elem())
|
||||
sl := reflect.MakeSlice(ty, 1, 1)
|
||||
sl.Index(0).Set(v)
|
||||
return sl
|
||||
|
||||
case reflect.Map:
|
||||
// Create a map with a single key-value pair, recursively creating each.
|
||||
k := nonEmptyValue(ty.Key())
|
||||
v := nonEmptyValue(ty.Elem())
|
||||
|
||||
m := reflect.MakeMap(ty)
|
||||
m.SetMapIndex(k, v)
|
||||
return m
|
||||
|
||||
default:
|
||||
panic("unhandled type " + ty.String())
|
||||
}
|
||||
}
|
||||
|
||||
typ := reflect.TypeFor[T]()
|
||||
for i, n := 0, typ.NumField(); i < n; i++ {
|
||||
sf := typ.Field(i)
|
||||
|
||||
var nonzero T
|
||||
rv := reflect.ValueOf(&nonzero).Elem()
|
||||
rv.Field(i).Set(nonEmptyValue(sf.Type))
|
||||
if nonzero.IsZero() {
|
||||
t.Errorf("IsZero = true with %v set; want false\nvalue: %#v", sf.Name, nonzero)
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user