Commit 4d6a69f2 authored by Mikio Hara's avatar Mikio Hara

net: force LookupAddr results to be rooted DNS paths even in the case of local source

The builtin name resolver using various resolution techniques is a bit
complicated and we sometimes fotget to take care of all the go and cgo
code paths and exchanging information to local and remote sources. This
change makes LookupAddr return absolute domain names even in the case of
local source.

Updates #12189.
Fixes #12240.

Change-Id: Icdd3375bcddc7f5d4d3b24f134d93815073736fc
Reviewed-on: https://go-review.googlesource.com/17216Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent a0233fdb
...@@ -222,12 +222,7 @@ func cgoLookupPTR(addr string) ([]string, error, bool) { ...@@ -222,12 +222,7 @@ func cgoLookupPTR(addr string) ([]string, error, bool) {
break break
} }
} }
// Add trailing dot to match pure Go reverse resolver return []string{absDomainName(b)}, nil, true
// and all other lookup routines. See golang.org/issue/12189.
if len(b) > 0 && b[len(b)-1] != '.' {
b = append(b, '.')
}
return []string{string(b)}, nil, true
} }
func cgoSockaddr(ip IP) (*C.struct_sockaddr, C.socklen_t) { func cgoSockaddr(ip IP) (*C.struct_sockaddr, C.socklen_t) {
......
...@@ -161,6 +161,17 @@ func isDomainName(s string) bool { ...@@ -161,6 +161,17 @@ func isDomainName(s string) bool {
return ok return ok
} }
// absDomainName returns an absoulte domain name which ends with a
// trailing dot to match pure Go reverse resolver and all other lookup
// routines.
// See golang.org/issue/12189.
func absDomainName(b []byte) string {
if len(b) > 0 && b[len(b)-1] != '.' {
b = append(b, '.')
}
return string(b)
}
// An SRV represents a single DNS SRV record. // An SRV represents a single DNS SRV record.
type SRV struct { type SRV struct {
Target string Target string
......
...@@ -70,10 +70,10 @@ func readHosts() { ...@@ -70,10 +70,10 @@ func readHosts() {
continue continue
} }
for i := 1; i < len(f); i++ { for i := 1; i < len(f); i++ {
name := f[i] name := absDomainName([]byte(f[i]))
h := []byte(f[i]) h := []byte(f[i])
lowerASCIIBytes(h) lowerASCIIBytes(h)
key := string(h) key := absDomainName(h)
hs[key] = append(hs[key], addr) hs[key] = append(hs[key], addr)
is[addr] = append(is[addr], name) is[addr] = append(is[addr], name)
} }
...@@ -97,7 +97,7 @@ func lookupStaticHost(host string) []string { ...@@ -97,7 +97,7 @@ func lookupStaticHost(host string) []string {
// or linear scan the byName map if it's small enough? // or linear scan the byName map if it's small enough?
lowerHost := []byte(host) lowerHost := []byte(host)
lowerASCIIBytes(lowerHost) lowerASCIIBytes(lowerHost)
if ips, ok := hosts.byName[string(lowerHost)]; ok { if ips, ok := hosts.byName[absDomainName(lowerHost)]; ok {
return ips return ips
} }
} }
......
...@@ -64,7 +64,7 @@ func TestLookupStaticHost(t *testing.T) { ...@@ -64,7 +64,7 @@ func TestLookupStaticHost(t *testing.T) {
for _, tt := range lookupStaticHostTests { for _, tt := range lookupStaticHostTests {
testHookHostsPath = tt.name testHookHostsPath = tt.name
for _, ent := range tt.ents { for _, ent := range tt.ents {
ins := []string{ent.in, strings.ToLower(ent.in), strings.ToUpper(ent.in)} ins := []string{ent.in, absDomainName([]byte(ent.in)), strings.ToLower(ent.in), strings.ToUpper(ent.in)}
for _, in := range ins { for _, in := range ins {
addrs := lookupStaticHost(in) addrs := lookupStaticHost(in)
if !reflect.DeepEqual(addrs, ent.out) { if !reflect.DeepEqual(addrs, ent.out) {
...@@ -130,6 +130,9 @@ func TestLookupStaticAddr(t *testing.T) { ...@@ -130,6 +130,9 @@ func TestLookupStaticAddr(t *testing.T) {
testHookHostsPath = tt.name testHookHostsPath = tt.name
for _, ent := range tt.ents { for _, ent := range tt.ents {
hosts := lookupStaticAddr(ent.in) hosts := lookupStaticAddr(ent.in)
for i := range ent.out {
ent.out[i] = absDomainName([]byte(ent.out[i]))
}
if !reflect.DeepEqual(hosts, ent.out) { if !reflect.DeepEqual(hosts, ent.out) {
t.Errorf("%s, lookupStaticAddr(%s) = %v; want %v", tt.name, ent.in, hosts, ent.out) t.Errorf("%s, lookupStaticAddr(%s) = %v; want %v", tt.name, ent.in, hosts, ent.out)
} }
......
...@@ -395,17 +395,42 @@ func TestLookupIPDeadline(t *testing.T) { ...@@ -395,17 +395,42 @@ func TestLookupIPDeadline(t *testing.T) {
t.Logf("%v succeeded, %v failed (%v timeout, %v temporary, %v other, %v unknown)", qstats.succeeded, qstats.failed, qstats.timeout, qstats.temporary, qstats.other, qstats.unknown) t.Logf("%v succeeded, %v failed (%v timeout, %v temporary, %v other, %v unknown)", qstats.succeeded, qstats.failed, qstats.timeout, qstats.temporary, qstats.other, qstats.unknown)
} }
func TestLookupDots(t *testing.T) { func TestLookupDotsWithLocalSoruce(t *testing.T) {
if !supportsIPv4 {
t.Skip("IPv4 is required")
}
for i, fn := range []func() func(){forceGoDNS, forceCgoDNS} {
fixup := fn()
if fixup == nil {
continue
}
names, err := LookupAddr("127.0.0.1")
fixup()
if err != nil {
t.Errorf("#%d: %v", i, err)
continue
}
for _, name := range names {
if !strings.HasSuffix(name, ".") {
t.Errorf("#%d: got %s; want name ending with trailing dot", i, name)
}
}
}
}
func TestLookupDotsWithRemoteSource(t *testing.T) {
if testing.Short() || !*testExternal { if testing.Short() || !*testExternal {
t.Skipf("skipping external network test") t.Skipf("skipping external network test")
} }
fixup := forceGoDNS() if fixup := forceGoDNS(); fixup != nil {
defer fixup()
testDots(t, "go") testDots(t, "go")
fixup()
if forceCgoDNS() { }
if fixup := forceCgoDNS(); fixup != nil {
testDots(t, "cgo") testDots(t, "cgo")
fixup()
} }
} }
......
...@@ -7,5 +7,5 @@ ...@@ -7,5 +7,5 @@
package net package net
// See unix_test.go for what these (don't) do. // See unix_test.go for what these (don't) do.
func forceGoDNS() func() { return func() {} } func forceGoDNS() func() { return nil }
func forceCgoDNS() bool { return false } func forceCgoDNS() func() { return nil }
...@@ -421,11 +421,17 @@ func forceGoDNS() func() { ...@@ -421,11 +421,17 @@ func forceGoDNS() func() {
} }
// forceCgoDNS forces the resolver configuration to use the cgo resolver // forceCgoDNS forces the resolver configuration to use the cgo resolver
// and returns true to indicate that it did so. // and returns a fixup function to restore the old settings.
// (On non-Unix systems forceCgoDNS returns false.) // (On non-Unix systems forceCgoDNS returns nil.)
func forceCgoDNS() bool { func forceCgoDNS() func() {
c := systemConf() c := systemConf()
oldGo := c.netGo
oldCgo := c.netCgo
fixup := func() {
c.netGo = oldGo
c.netCgo = oldCgo
}
c.netGo = false c.netGo = false
c.netCgo = true c.netCgo = true
return true return fixup
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment