2021-02-02 12:05:47 +01:00
|
|
|
package dialer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"net"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/errorx"
|
|
|
|
)
|
|
|
|
|
2021-06-09 09:42:31 +02:00
|
|
|
// dnsDialer is a dialer that uses the configured Resolver to resolver a
|
2021-02-02 12:05:47 +01:00
|
|
|
// domain name to IP addresses, and the configured Dialer to connect.
|
2021-06-09 09:42:31 +02:00
|
|
|
type dnsDialer struct {
|
2021-02-02 12:05:47 +01:00
|
|
|
Dialer
|
|
|
|
Resolver Resolver
|
|
|
|
}
|
|
|
|
|
|
|
|
// DialContext implements Dialer.DialContext.
|
2021-06-09 09:42:31 +02:00
|
|
|
func (d *dnsDialer) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
2021-02-02 12:05:47 +01:00
|
|
|
onlyhost, onlyport, err := net.SplitHostPort(address)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var addrs []string
|
2021-06-09 09:42:31 +02:00
|
|
|
addrs, err = d.lookupHost(ctx, onlyhost)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
var errorslist []error
|
|
|
|
for _, addr := range addrs {
|
|
|
|
target := net.JoinHostPort(addr, onlyport)
|
|
|
|
conn, err := d.Dialer.DialContext(ctx, network, target)
|
|
|
|
if err == nil {
|
|
|
|
return conn, nil
|
|
|
|
}
|
|
|
|
errorslist = append(errorslist, err)
|
|
|
|
}
|
|
|
|
return nil, ReduceErrors(errorslist)
|
|
|
|
}
|
|
|
|
|
2021-06-09 09:42:31 +02:00
|
|
|
// ReduceErrors finds a known error in a list of errors since it's probably most relevant.
|
2021-02-02 12:05:47 +01:00
|
|
|
func ReduceErrors(errorslist []error) error {
|
|
|
|
if len(errorslist) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// If we have a known error, let's consider this the real error
|
|
|
|
// since it's probably most relevant. Otherwise let's return the
|
|
|
|
// first considering that (1) local resolvers likely will give
|
|
|
|
// us IPv4 first and (2) also our resolver does that. So, in case
|
|
|
|
// the user has no IPv6 connectivity, an IPv6 error is going to
|
|
|
|
// appear later in the list of errors.
|
|
|
|
for _, err := range errorslist {
|
|
|
|
var wrapper *errorx.ErrWrapper
|
|
|
|
if errors.As(err, &wrapper) && !strings.HasPrefix(
|
|
|
|
err.Error(), "unknown_failure",
|
|
|
|
) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// TODO(bassosimone): handle this case in a better way
|
|
|
|
return errorslist[0]
|
|
|
|
}
|
|
|
|
|
2021-06-09 09:42:31 +02:00
|
|
|
// lookupHost performs a domain name resolution.
|
|
|
|
func (d *dnsDialer) lookupHost(ctx context.Context, hostname string) ([]string, error) {
|
2021-02-02 12:05:47 +01:00
|
|
|
if net.ParseIP(hostname) != nil {
|
|
|
|
return []string{hostname}, nil
|
|
|
|
}
|
|
|
|
return d.Resolver.LookupHost(ctx, hostname)
|
|
|
|
}
|