package quicdialer import ( "context" "crypto/tls" "net" "github.com/lucas-clemente/quic-go" "github.com/ooni/probe-cli/v3/internal/netxlite" ) // DNSDialer is a dialer that uses the configured Resolver to resolve a // domain name to IP addresses type DNSDialer struct { Dialer ContextDialer Resolver Resolver } // DialContext implements ContextDialer.DialContext func (d DNSDialer) DialContext( ctx context.Context, network, host string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error) { onlyhost, onlyport, err := net.SplitHostPort(host) if err != nil { return nil, err } // TODO(kelmenhorst): Should this be somewhere else? // failure if tlsCfg is nil but that should not happen if tlsCfg.ServerName == "" { tlsCfg.ServerName = onlyhost } var addrs []string addrs, err = d.LookupHost(ctx, onlyhost) if err != nil { return nil, err } var errorslist []error for _, addr := range addrs { target := net.JoinHostPort(addr, onlyport) sess, err := d.Dialer.DialContext( ctx, network, target, tlsCfg, cfg) if err == nil { return sess, nil } errorslist = append(errorslist, err) } // TODO(bassosimone): maybe ReduceErrors could be in netx/internal. return nil, netxlite.ReduceErrors(errorslist) } // LookupHost implements Resolver.LookupHost func (d DNSDialer) LookupHost(ctx context.Context, hostname string) ([]string, error) { if net.ParseIP(hostname) != nil { return []string{hostname}, nil } return d.Resolver.LookupHost(ctx, hostname) }