Skip to content
This repository has been archived by the owner on Jun 20, 2023. It is now read-only.

Commit

Permalink
Properly report DNSLink errors (#12)
Browse files Browse the repository at this point in the history
* Properly report DNSLink errors

Only report that there is no DNSLink for a name when there are no DNSLink TXT records available for that name.  For all other errors, such as being offline, report the more general "cannot resolve name" error.

* Document that we give precedence to good results from looking up DNSLinks in TXT records from `"_dnslink."+fqdn` over results from `fqdn`.
  • Loading branch information
gammazero authored Apr 21, 2021
1 parent 3d90782 commit 22432d1
Show file tree
Hide file tree
Showing 2 changed files with 28 additions and 10 deletions.
9 changes: 0 additions & 9 deletions base.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@ package namesys

import (
"context"
"fmt"
"strings"
"time"

Expand Down Expand Up @@ -37,14 +36,6 @@ func resolve(ctx context.Context, r resolver, name string, options opts.ResolveO
}
}

if err == ErrResolveFailed {
i := len(name) - 1
for i >= 0 && name[i] != '/' {
i--
}
// Wrap error so that it can be tested if it is a ErrResolveFailed
err = fmt.Errorf("%w: %q is missing a DNSLink record (https://docs.ipfs.io/concepts/dnslink/)", ErrResolveFailed, name[i+1:])
}
return p, err
}

Expand Down
29 changes: 28 additions & 1 deletion dns.go
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import (
"errors"
"fmt"
"net"
gpath "path"
"strings"

path "github.com/ipfs/go-path"
Expand Down Expand Up @@ -88,6 +89,7 @@ func (r *DNSResolver) resolveOnceAsync(ctx context.Context, name string, options

go func() {
defer close(out)
var rootResErr, subResErr error
for {
select {
case subRes, ok := <-subChan:
Expand All @@ -98,8 +100,11 @@ func (r *DNSResolver) resolveOnceAsync(ctx context.Context, name string, options
if subRes.error == nil {
p, err := appendPath(subRes.path)
emitOnceResult(ctx, out, onceResult{value: p, err: err})
// Return without waiting for rootRes, since this result
// (for "_dnslink."+fqdn) takes precedence
return
}
subResErr = subRes.error
case rootRes, ok := <-rootChan:
if !ok {
rootChan = nil
Expand All @@ -108,11 +113,24 @@ func (r *DNSResolver) resolveOnceAsync(ctx context.Context, name string, options
if rootRes.error == nil {
p, err := appendPath(rootRes.path)
emitOnceResult(ctx, out, onceResult{value: p, err: err})
// Do not return here. Wait for subRes so that it is
// output last if good, thereby giving subRes precedence.
} else {
rootResErr = rootRes.error
}
case <-ctx.Done():
return
}
if subChan == nil && rootChan == nil {
// If here, then both lookups are done
//
// If both lookups failed due to no TXT records with a
// dnslink, then output a more specific error message
if rootResErr == ErrResolveFailed && subResErr == ErrResolveFailed {
// Wrap error so that it can be tested if it is a ErrResolveFailed
err := fmt.Errorf("%w: %q is missing a DNSLink record (https://docs.ipfs.io/concepts/dnslink/)", ErrResolveFailed, gpath.Base(name))
emitOnceResult(ctx, out, onceResult{err: err})
}
return
}
}
Expand All @@ -126,7 +144,14 @@ func workDomain(r *DNSResolver, name string, res chan lookupRes) {

txt, err := r.lookupTXT(name)
if err != nil {
// Error is != nil
if dnsErr, ok := err.(*net.DNSError); ok {
// If no TXT records found, return same error as when no text
// records contain dnslink. Otherwise, return the actual error.
if dnsErr.IsNotFound {
err = ErrResolveFailed
}
}
// Could not look up any text records for name
res <- lookupRes{"", err}
return
}
Expand All @@ -138,6 +163,8 @@ func workDomain(r *DNSResolver, name string, res chan lookupRes) {
return
}
}

// There were no TXT records with a dnslink
res <- lookupRes{"", ErrResolveFailed}
}

Expand Down

0 comments on commit 22432d1

Please sign in to comment.