refactor: start pivoting netx (#396)
What do I mean by pivoting? Netx is currently organized by row: ``` | dialer | quicdialer | resolver | ... saving | | | | ... errorwrapping | | | | ... logging | | | | ... mocking/sys | | | | ... ``` Every row needs to implement saving, errorwrapping, logging, mocking (or adapting to the system or to some underlying library). This causes cross package dependencies and, in turn, complexity. For example, we need the `trace` package for supporting saving. And `dialer`, `quickdialer`, et al. need to depend on such a package. The same goes for errorwrapping. This arrangement further complicates testing. For example, I am currently working on https://github.com/ooni/probe/issues/1505 and I realize it need to repeat integration tests in multiple places. Let's say instead we pivot the above matrix as follows: ``` | saving | errorwrapping | logging | ... dialer | | | | ... quicdialer | | | | ... logging | | | | ... mocking/sys | | | | ... ... ``` In this way, now every row contains everything related to a specific action to perform. We can now share code without relying on extra support packages. What's more, we can write tests and, judding from the way in which things are made, it seems we only need integration testing in `errorwrapping` because it's where data quality matters whereas, in all other cases, unit testing is fine. I am going, therefore, to proceed with these changes and "pivot" `netx`. Hopefully, it won't be too painful.
This commit is contained in:
parent
c74c94d616
commit
8a0beee808
|
@ -11,8 +11,8 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/cmd/oohelperd/internal"
|
"github.com/ooni/probe-cli/v3/internal/cmd/oohelperd/internal"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/iox"
|
"github.com/ooni/probe-cli/v3/internal/iox"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
const simplerequest = `{
|
const simplerequest = `{
|
||||||
|
@ -56,7 +56,7 @@ func TestWorkingAsIntended(t *testing.T) {
|
||||||
Client: http.DefaultClient,
|
Client: http.DefaultClient,
|
||||||
Dialer: new(net.Dialer),
|
Dialer: new(net.Dialer),
|
||||||
MaxAcceptableBody: 1 << 24,
|
MaxAcceptableBody: 1 << 24,
|
||||||
Resolver: resolver.SystemResolver{},
|
Resolver: netxlite.ResolverSystem{},
|
||||||
}
|
}
|
||||||
srv := httptest.NewServer(handler)
|
srv := httptest.NewServer(handler)
|
||||||
defer srv.Close()
|
defer srv.Close()
|
||||||
|
|
|
@ -10,6 +10,7 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/model"
|
"github.com/ooni/probe-cli/v3/internal/engine/model"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/dialer"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/dialer"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
type dialManager struct {
|
type dialManager struct {
|
||||||
|
@ -33,8 +34,8 @@ func newDialManager(ndt7URL string, logger model.Logger, userAgent string) dialM
|
||||||
}
|
}
|
||||||
|
|
||||||
func (mgr dialManager) dialWithTestName(ctx context.Context, testName string) (*websocket.Conn, error) {
|
func (mgr dialManager) dialWithTestName(ctx context.Context, testName string) (*websocket.Conn, error) {
|
||||||
var reso resolver.Resolver = resolver.SystemResolver{}
|
var reso resolver.Resolver = netxlite.ResolverSystem{}
|
||||||
reso = resolver.LoggingResolver{Resolver: reso, Logger: mgr.logger}
|
reso = netxlite.ResolverLogger{Resolver: reso, Logger: mgr.logger}
|
||||||
dlr := dialer.New(&dialer.Config{
|
dlr := dialer.New(&dialer.Config{
|
||||||
ContextByteCounting: true,
|
ContextByteCounting: true,
|
||||||
Logger: mgr.logger,
|
Logger: mgr.logger,
|
||||||
|
|
|
@ -12,6 +12,7 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/handlers"
|
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/handlers"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/modelx"
|
"github.com/ooni/probe-cli/v3/internal/engine/legacy/netx/modelx"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -158,7 +159,7 @@ func resolverWrapTransport(txp resolver.RoundTripper) resolver.EmitterResolver {
|
||||||
}
|
}
|
||||||
|
|
||||||
func newResolverSystem() resolver.EmitterResolver {
|
func newResolverSystem() resolver.EmitterResolver {
|
||||||
return resolverWrapResolver(resolver.SystemResolver{})
|
return resolverWrapResolver(netxlite.ResolverSystem{})
|
||||||
}
|
}
|
||||||
|
|
||||||
func newResolverUDP(dialer resolver.Dialer, address string) resolver.EmitterResolver {
|
func newResolverUDP(dialer resolver.Dialer, address string) resolver.EmitterResolver {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import (
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Dialer establishes network connections.
|
// Dialer establishes network connections.
|
||||||
|
@ -64,10 +65,10 @@ type Config struct {
|
||||||
|
|
||||||
// New creates a new Dialer from the specified config and resolver.
|
// New creates a new Dialer from the specified config and resolver.
|
||||||
func New(config *Config, resolver Resolver) Dialer {
|
func New(config *Config, resolver Resolver) Dialer {
|
||||||
var d Dialer = systemDialer
|
var d Dialer = netxlite.DefaultDialer
|
||||||
d = &errorWrapperDialer{Dialer: d}
|
d = &errorWrapperDialer{Dialer: d}
|
||||||
if config.Logger != nil {
|
if config.Logger != nil {
|
||||||
d = &loggingDialer{Dialer: d, Logger: config.Logger}
|
d = &netxlite.DialerLogger{Dialer: d, Logger: config.Logger}
|
||||||
}
|
}
|
||||||
if config.DialSaver != nil {
|
if config.DialSaver != nil {
|
||||||
d = &saverDialer{Dialer: d, Saver: config.DialSaver}
|
d = &saverDialer{Dialer: d, Saver: config.DialSaver}
|
||||||
|
@ -75,7 +76,7 @@ func New(config *Config, resolver Resolver) Dialer {
|
||||||
if config.ReadWriteSaver != nil {
|
if config.ReadWriteSaver != nil {
|
||||||
d = &saverConnDialer{Dialer: d, Saver: config.ReadWriteSaver}
|
d = &saverConnDialer{Dialer: d, Saver: config.ReadWriteSaver}
|
||||||
}
|
}
|
||||||
d = &dnsDialer{Resolver: resolver, Dialer: d}
|
d = &netxlite.DialerResolver{Resolver: resolver, Dialer: d}
|
||||||
d = &proxyDialer{ProxyURL: config.ProxyURL, Dialer: d}
|
d = &proxyDialer{ProxyURL: config.ProxyURL, Dialer: d}
|
||||||
if config.ContextByteCounting {
|
if config.ContextByteCounting {
|
||||||
d = &byteCounterDialer{Dialer: d}
|
d = &byteCounterDialer{Dialer: d}
|
||||||
|
|
|
@ -7,6 +7,7 @@ import (
|
||||||
|
|
||||||
"github.com/apex/log"
|
"github.com/apex/log"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNewCreatesTheExpectedChain(t *testing.T) {
|
func TestNewCreatesTheExpectedChain(t *testing.T) {
|
||||||
|
@ -30,7 +31,7 @@ func TestNewCreatesTheExpectedChain(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not a proxyDialer")
|
t.Fatal("not a proxyDialer")
|
||||||
}
|
}
|
||||||
dnsd, ok := pd.Dialer.(*dnsDialer)
|
dnsd, ok := pd.Dialer.(*netxlite.DialerResolver)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not a dnsDialer")
|
t.Fatal("not a dnsDialer")
|
||||||
}
|
}
|
||||||
|
@ -42,7 +43,7 @@ func TestNewCreatesTheExpectedChain(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not a saverDialer")
|
t.Fatal("not a saverDialer")
|
||||||
}
|
}
|
||||||
ld, ok := sd.Dialer.(*loggingDialer)
|
ld, ok := sd.Dialer.(*netxlite.DialerLogger)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not a loggingDialer")
|
t.Fatal("not a loggingDialer")
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,71 +0,0 @@
|
||||||
package dialer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
"net"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/errorx"
|
|
||||||
)
|
|
||||||
|
|
||||||
// dnsDialer is a dialer that uses the configured Resolver to resolver a
|
|
||||||
// domain name to IP addresses, and the configured Dialer to connect.
|
|
||||||
type dnsDialer struct {
|
|
||||||
Dialer
|
|
||||||
Resolver Resolver
|
|
||||||
}
|
|
||||||
|
|
||||||
// DialContext implements Dialer.DialContext.
|
|
||||||
func (d *dnsDialer) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
|
||||||
onlyhost, onlyport, err := net.SplitHostPort(address)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
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)
|
|
||||||
conn, err := d.Dialer.DialContext(ctx, network, target)
|
|
||||||
if err == nil {
|
|
||||||
return conn, nil
|
|
||||||
}
|
|
||||||
errorslist = append(errorslist, err)
|
|
||||||
}
|
|
||||||
return nil, ReduceErrors(errorslist)
|
|
||||||
}
|
|
||||||
|
|
||||||
// ReduceErrors finds a known error in a list of errors since it's probably most relevant.
|
|
||||||
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]
|
|
||||||
}
|
|
||||||
|
|
||||||
// lookupHost performs a domain name resolution.
|
|
||||||
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)
|
|
||||||
}
|
|
|
@ -1,23 +0,0 @@
|
||||||
package dialer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"net"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// loggingDialer is a Dialer with logging
|
|
||||||
type loggingDialer struct {
|
|
||||||
Dialer
|
|
||||||
Logger Logger
|
|
||||||
}
|
|
||||||
|
|
||||||
// DialContext implements Dialer.DialContext
|
|
||||||
func (d *loggingDialer) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
|
||||||
d.Logger.Debugf("dial %s/%s...", address, network)
|
|
||||||
start := time.Now()
|
|
||||||
conn, err := d.Dialer.DialContext(ctx, network, address)
|
|
||||||
stop := time.Now()
|
|
||||||
d.Logger.Debugf("dial %s/%s... %+v in %s", address, network, err, stop.Sub(start))
|
|
||||||
return conn, err
|
|
||||||
}
|
|
|
@ -1,30 +0,0 @@
|
||||||
package dialer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
"io"
|
|
||||||
"net"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/apex/log"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/mockablex"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestLoggingDialerFailure(t *testing.T) {
|
|
||||||
d := &loggingDialer{
|
|
||||||
Dialer: mockablex.Dialer{
|
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
|
||||||
return nil, io.EOF
|
|
||||||
},
|
|
||||||
},
|
|
||||||
Logger: log.Log,
|
|
||||||
}
|
|
||||||
conn, err := d.DialContext(context.Background(), "tcp", "www.google.com:443")
|
|
||||||
if !errors.Is(err, io.EOF) {
|
|
||||||
t.Fatal("not the error we expected")
|
|
||||||
}
|
|
||||||
if conn != nil {
|
|
||||||
t.Fatal("expected nil conn here")
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,12 +0,0 @@
|
||||||
package dialer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// systemDialer is the underlying net.Dialer.
|
|
||||||
var systemDialer = &net.Dialer{
|
|
||||||
Timeout: 15 * time.Second,
|
|
||||||
KeepAlive: 15 * time.Second,
|
|
||||||
}
|
|
|
@ -1,28 +0,0 @@
|
||||||
package dialer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/ooni/psiphon/oopsi/golang.org/x/net/context"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestSystemDialerWorks(t *testing.T) {
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
cancel() // fail immediately
|
|
||||||
conn, err := systemDialer.DialContext(ctx, "tcp", "8.8.8.8:853")
|
|
||||||
if err == nil || !strings.HasSuffix(err.Error(), "operation was canceled") {
|
|
||||||
t.Fatal("not the error we expected", err)
|
|
||||||
}
|
|
||||||
if conn != nil {
|
|
||||||
t.Fatal("expected nil conn here")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSystemDialerHasTimeout(t *testing.T) {
|
|
||||||
expected := 15 * time.Second
|
|
||||||
if systemDialer.Timeout != expected {
|
|
||||||
t.Fatal("unexpected timeout value")
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -39,6 +39,7 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsdialer"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsdialer"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsx"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsx"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Logger is the logger assumed by this package
|
// Logger is the logger assumed by this package
|
||||||
|
@ -114,7 +115,7 @@ var defaultCertPool *x509.CertPool = tlsx.NewDefaultCertPool()
|
||||||
// NewResolver creates a new resolver from the specified config
|
// NewResolver creates a new resolver from the specified config
|
||||||
func NewResolver(config Config) Resolver {
|
func NewResolver(config Config) Resolver {
|
||||||
if config.BaseResolver == nil {
|
if config.BaseResolver == nil {
|
||||||
config.BaseResolver = resolver.SystemResolver{}
|
config.BaseResolver = netxlite.ResolverSystem{}
|
||||||
}
|
}
|
||||||
var r Resolver = config.BaseResolver
|
var r Resolver = config.BaseResolver
|
||||||
r = resolver.AddressResolver{Resolver: r}
|
r = resolver.AddressResolver{Resolver: r}
|
||||||
|
@ -133,7 +134,7 @@ func NewResolver(config Config) Resolver {
|
||||||
}
|
}
|
||||||
r = resolver.ErrorWrapperResolver{Resolver: r}
|
r = resolver.ErrorWrapperResolver{Resolver: r}
|
||||||
if config.Logger != nil {
|
if config.Logger != nil {
|
||||||
r = resolver.LoggingResolver{Logger: config.Logger, Resolver: r}
|
r = netxlite.ResolverLogger{Logger: config.Logger, Resolver: r}
|
||||||
}
|
}
|
||||||
if config.ResolveSaver != nil {
|
if config.ResolveSaver != nil {
|
||||||
r = resolver.SaverResolver{Resolver: r, Saver: config.ResolveSaver}
|
r = resolver.SaverResolver{Resolver: r, Saver: config.ResolveSaver}
|
||||||
|
@ -317,7 +318,7 @@ func NewDNSClientWithOverrides(config Config, URL, hostOverride, SNIOverride,
|
||||||
}
|
}
|
||||||
switch resolverURL.Scheme {
|
switch resolverURL.Scheme {
|
||||||
case "system":
|
case "system":
|
||||||
c.Resolver = resolver.SystemResolver{}
|
c.Resolver = netxlite.ResolverSystem{}
|
||||||
return c, nil
|
return c, nil
|
||||||
case "https":
|
case "https":
|
||||||
config.TLSConfig.NextProtos = []string{"h2", "http/1.1"}
|
config.TLSConfig.NextProtos = []string{"h2", "http/1.1"}
|
||||||
|
|
|
@ -15,6 +15,7 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsdialer"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsdialer"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsx"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/tlsx"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestNewResolverVanilla(t *testing.T) {
|
func TestNewResolverVanilla(t *testing.T) {
|
||||||
|
@ -31,7 +32,7 @@ func TestNewResolverVanilla(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -81,7 +82,7 @@ func TestNewResolverWithBogonFilter(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -95,7 +96,7 @@ func TestNewResolverWithLogging(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
lr, ok := ir.Resolver.(resolver.LoggingResolver)
|
lr, ok := ir.Resolver.(netxlite.ResolverLogger)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -110,7 +111,7 @@ func TestNewResolverWithLogging(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -140,7 +141,7 @@ func TestNewResolverWithSaver(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -169,7 +170,7 @@ func TestNewResolverWithReadWriteCache(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -203,7 +204,7 @@ func TestNewResolverWithPrefilledReadonlyCache(t *testing.T) {
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
_, ok = ar.Resolver.(resolver.SystemResolver)
|
_, ok = ar.Resolver.(netxlite.ResolverSystem)
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
|
@ -597,7 +598,7 @@ func TestNewDNSClientSystemResolver(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
if _, ok := dnsclient.Resolver.(resolver.SystemResolver); !ok {
|
if _, ok := dnsclient.Resolver.(netxlite.ResolverSystem); !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
dnsclient.CloseIdleConnections()
|
dnsclient.CloseIdleConnections()
|
||||||
|
@ -609,7 +610,7 @@ func TestNewDNSClientEmpty(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
if _, ok := dnsclient.Resolver.(resolver.SystemResolver); !ok {
|
if _, ok := dnsclient.Resolver.(netxlite.ResolverSystem); !ok {
|
||||||
t.Fatal("not the resolver we expected")
|
t.Fatal("not the resolver we expected")
|
||||||
}
|
}
|
||||||
dnsclient.CloseIdleConnections()
|
dnsclient.CloseIdleConnections()
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go"
|
"github.com/lucas-clemente/quic-go"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/dialer"
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
// DNSDialer is a dialer that uses the configured Resolver to resolve a
|
// DNSDialer is a dialer that uses the configured Resolver to resolve a
|
||||||
|
@ -45,7 +45,7 @@ func (d DNSDialer) DialContext(
|
||||||
errorslist = append(errorslist, err)
|
errorslist = append(errorslist, err)
|
||||||
}
|
}
|
||||||
// TODO(bassosimone): maybe ReduceErrors could be in netx/internal.
|
// TODO(bassosimone): maybe ReduceErrors could be in netx/internal.
|
||||||
return nil, dialer.ReduceErrors(errorslist)
|
return nil, netxlite.ReduceErrors(errorslist)
|
||||||
}
|
}
|
||||||
|
|
||||||
// LookupHost implements Resolver.LookupHost
|
// LookupHost implements Resolver.LookupHost
|
||||||
|
|
|
@ -5,12 +5,13 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestChainLookupHost(t *testing.T) {
|
func TestChainLookupHost(t *testing.T) {
|
||||||
r := resolver.ChainResolver{
|
r := resolver.ChainResolver{
|
||||||
Primary: resolver.NewFakeResolverThatFails(),
|
Primary: resolver.NewFakeResolverThatFails(),
|
||||||
Secondary: resolver.SystemResolver{},
|
Secondary: netxlite.ResolverSystem{},
|
||||||
}
|
}
|
||||||
if r.Address() != "" {
|
if r.Address() != "" {
|
||||||
t.Fatal("invalid address")
|
t.Fatal("invalid address")
|
||||||
|
|
|
@ -8,6 +8,7 @@ import (
|
||||||
|
|
||||||
"github.com/apex/log"
|
"github.com/apex/log"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -18,7 +19,7 @@ func testresolverquick(t *testing.T, reso resolver.Resolver) {
|
||||||
if testing.Short() {
|
if testing.Short() {
|
||||||
t.Skip("skip test in short mode")
|
t.Skip("skip test in short mode")
|
||||||
}
|
}
|
||||||
reso = resolver.LoggingResolver{Logger: log.Log, Resolver: reso}
|
reso = netxlite.ResolverLogger{Logger: log.Log, Resolver: reso}
|
||||||
addrs, err := reso.LookupHost(context.Background(), "dns.google.com")
|
addrs, err := reso.LookupHost(context.Background(), "dns.google.com")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -44,7 +45,7 @@ func testresolverquickidna(t *testing.T, reso resolver.Resolver) {
|
||||||
t.Skip("skip test in short mode")
|
t.Skip("skip test in short mode")
|
||||||
}
|
}
|
||||||
reso = resolver.IDNAResolver{
|
reso = resolver.IDNAResolver{
|
||||||
resolver.LoggingResolver{Logger: log.Log, Resolver: reso},
|
netxlite.ResolverLogger{Logger: log.Log, Resolver: reso},
|
||||||
}
|
}
|
||||||
addrs, err := reso.LookupHost(context.Background(), "яндекс.рф")
|
addrs, err := reso.LookupHost(context.Background(), "яндекс.рф")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -56,7 +57,7 @@ func testresolverquickidna(t *testing.T, reso resolver.Resolver) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestNewResolverSystem(t *testing.T) {
|
func TestNewResolverSystem(t *testing.T) {
|
||||||
reso := resolver.SystemResolver{}
|
reso := netxlite.ResolverSystem{}
|
||||||
testresolverquick(t, reso)
|
testresolverquick(t, reso)
|
||||||
testresolverquickidna(t, reso)
|
testresolverquickidna(t, reso)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,30 +0,0 @@
|
||||||
package resolver
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Logger is the logger assumed by this package
|
|
||||||
type Logger interface {
|
|
||||||
Debugf(format string, v ...interface{})
|
|
||||||
Debug(message string)
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoggingResolver is a resolver that emits events
|
|
||||||
type LoggingResolver struct {
|
|
||||||
Resolver
|
|
||||||
Logger Logger
|
|
||||||
}
|
|
||||||
|
|
||||||
// LookupHost returns the IP addresses of a host
|
|
||||||
func (r LoggingResolver) LookupHost(ctx context.Context, hostname string) ([]string, error) {
|
|
||||||
r.Logger.Debugf("resolve %s...", hostname)
|
|
||||||
start := time.Now()
|
|
||||||
addrs, err := r.Resolver.LookupHost(ctx, hostname)
|
|
||||||
stop := time.Now()
|
|
||||||
r.Logger.Debugf("resolve %s... (%+v, %+v) in %s", hostname, addrs, err, stop.Sub(start))
|
|
||||||
return addrs, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var _ Resolver = LoggingResolver{}
|
|
|
@ -1,23 +0,0 @@
|
||||||
package resolver_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/apex/log"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestLoggingResolver(t *testing.T) {
|
|
||||||
r := resolver.LoggingResolver{
|
|
||||||
Logger: log.Log,
|
|
||||||
Resolver: resolver.NewFakeResolverThatFails(),
|
|
||||||
}
|
|
||||||
addrs, err := r.LookupHost(context.Background(), "www.google.com")
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("expected an error here")
|
|
||||||
}
|
|
||||||
if addrs != nil {
|
|
||||||
t.Fatal("expected nil addr here")
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,29 +0,0 @@
|
||||||
package resolver
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"net"
|
|
||||||
)
|
|
||||||
|
|
||||||
// SystemResolver is the system resolver.
|
|
||||||
type SystemResolver struct{}
|
|
||||||
|
|
||||||
// LookupHost implements Resolver.LookupHost.
|
|
||||||
func (r SystemResolver) LookupHost(ctx context.Context, hostname string) ([]string, error) {
|
|
||||||
return net.DefaultResolver.LookupHost(ctx, hostname)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Network implements Resolver.Network.
|
|
||||||
func (r SystemResolver) Network() string {
|
|
||||||
return "system"
|
|
||||||
}
|
|
||||||
|
|
||||||
// Address implements Resolver.Address.
|
|
||||||
func (r SystemResolver) Address() string {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
// Default is the resolver we use by default.
|
|
||||||
var Default = SystemResolver{}
|
|
||||||
|
|
||||||
var _ Resolver = SystemResolver{}
|
|
|
@ -1,25 +0,0 @@
|
||||||
package resolver_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/resolver"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestSystemResolverLookupHost(t *testing.T) {
|
|
||||||
r := resolver.SystemResolver{}
|
|
||||||
if r.Network() != "system" {
|
|
||||||
t.Fatal("invalid Network")
|
|
||||||
}
|
|
||||||
if r.Address() != "" {
|
|
||||||
t.Fatal("invalid Address")
|
|
||||||
}
|
|
||||||
addrs, err := r.LookupHost(context.Background(), "dns.google.com")
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if addrs == nil {
|
|
||||||
t.Fatal("expected non-nil result here")
|
|
||||||
}
|
|
||||||
}
|
|
86
internal/netxlite/dialer.go
Normal file
86
internal/netxlite/dialer.go
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Dialer establishes network connections.
|
||||||
|
type Dialer interface {
|
||||||
|
// DialContext behaves like net.Dialer.DialContext.
|
||||||
|
DialContext(ctx context.Context, network, address string) (net.Conn, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultDialer is the Dialer we use by default.
|
||||||
|
var DefaultDialer = &net.Dialer{
|
||||||
|
Timeout: 15 * time.Second,
|
||||||
|
KeepAlive: 15 * time.Second,
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ Dialer = DefaultDialer
|
||||||
|
|
||||||
|
// DialerResolver is a dialer that uses the configured Resolver to resolver a
|
||||||
|
// domain name to IP addresses, and the configured Dialer to connect.
|
||||||
|
type DialerResolver struct {
|
||||||
|
// Dialer is the underlying Dialer.
|
||||||
|
Dialer Dialer
|
||||||
|
|
||||||
|
// Resolver is the underlying Resolver.
|
||||||
|
Resolver Resolver
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ Dialer = &DialerResolver{}
|
||||||
|
|
||||||
|
// DialContext implements Dialer.DialContext.
|
||||||
|
func (d *DialerResolver) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
||||||
|
onlyhost, onlyport, err := net.SplitHostPort(address)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
var addrs []string
|
||||||
|
addrs, err = d.lookupHost(ctx, onlyhost)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// TODO(bassosimone): here we should be using multierror rather
|
||||||
|
// than just calling ReduceErrors. We are not ready to do that
|
||||||
|
// yet, though. To do that, we need first to modify nettests so
|
||||||
|
// that we actually avoid dialing when measuring.
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
// lookupHost performs a domain name resolution.
|
||||||
|
func (d *DialerResolver) lookupHost(ctx context.Context, hostname string) ([]string, error) {
|
||||||
|
if net.ParseIP(hostname) != nil {
|
||||||
|
return []string{hostname}, nil
|
||||||
|
}
|
||||||
|
return d.Resolver.LookupHost(ctx, hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DialerLogger is a Dialer with logging
|
||||||
|
type DialerLogger struct {
|
||||||
|
Dialer
|
||||||
|
Logger Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ Dialer = &DialerLogger{}
|
||||||
|
|
||||||
|
// DialContext implements Dialer.DialContext
|
||||||
|
func (d *DialerLogger) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
||||||
|
d.Logger.Debugf("dial %s/%s...", address, network)
|
||||||
|
start := time.Now()
|
||||||
|
conn, err := d.Dialer.DialContext(ctx, network, address)
|
||||||
|
stop := time.Now()
|
||||||
|
d.Logger.Debugf("dial %s/%s... %+v in %s", address, network, err, stop.Sub(start))
|
||||||
|
return conn, err
|
||||||
|
}
|
|
@ -1,18 +1,20 @@
|
||||||
package dialer
|
package netxlite
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/errorx"
|
"github.com/apex/log"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/mockablex"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/mockablex"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestDNSDialerNoPort(t *testing.T) {
|
func TestDialerResolverNoPort(t *testing.T) {
|
||||||
dialer := &dnsDialer{Dialer: new(net.Dialer), Resolver: new(net.Resolver)}
|
dialer := &DialerResolver{Dialer: new(net.Dialer), Resolver: DefaultResolver}
|
||||||
conn, err := dialer.DialContext(context.Background(), "tcp", "antani.ooni.nu")
|
conn, err := dialer.DialContext(context.Background(), "tcp", "antani.ooni.nu")
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("expected an error here")
|
t.Fatal("expected an error here")
|
||||||
|
@ -22,8 +24,8 @@ func TestDNSDialerNoPort(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDNSDialerLookupHostAddress(t *testing.T) {
|
func TestDialerResolverLookupHostAddress(t *testing.T) {
|
||||||
dialer := &dnsDialer{Dialer: new(net.Dialer), Resolver: MockableResolver{
|
dialer := &DialerResolver{Dialer: new(net.Dialer), Resolver: MockableResolver{
|
||||||
Err: errors.New("mocked error"),
|
Err: errors.New("mocked error"),
|
||||||
}}
|
}}
|
||||||
addrs, err := dialer.lookupHost(context.Background(), "1.1.1.1")
|
addrs, err := dialer.lookupHost(context.Background(), "1.1.1.1")
|
||||||
|
@ -35,9 +37,9 @@ func TestDNSDialerLookupHostAddress(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDNSDialerLookupHostFailure(t *testing.T) {
|
func TestDialerResolverLookupHostFailure(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
dialer := &dnsDialer{Dialer: new(net.Dialer), Resolver: MockableResolver{
|
dialer := &DialerResolver{Dialer: new(net.Dialer), Resolver: MockableResolver{
|
||||||
Err: expected,
|
Err: expected,
|
||||||
}}
|
}}
|
||||||
conn, err := dialer.DialContext(context.Background(), "tcp", "dns.google.com:853")
|
conn, err := dialer.DialContext(context.Background(), "tcp", "dns.google.com:853")
|
||||||
|
@ -58,12 +60,20 @@ func (r MockableResolver) LookupHost(ctx context.Context, host string) ([]string
|
||||||
return r.Addresses, r.Err
|
return r.Addresses, r.Err
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDNSDialerDialForSingleIPFails(t *testing.T) {
|
func (r MockableResolver) Network() string {
|
||||||
dialer := &dnsDialer{Dialer: mockablex.Dialer{
|
return "mockable"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r MockableResolver) Address() string {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDialerResolverDialForSingleIPFails(t *testing.T) {
|
||||||
|
dialer := &DialerResolver{Dialer: mockablex.Dialer{
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
return nil, io.EOF
|
return nil, io.EOF
|
||||||
},
|
},
|
||||||
}, Resolver: new(net.Resolver)}
|
}, Resolver: DefaultResolver}
|
||||||
conn, err := dialer.DialContext(context.Background(), "tcp", "1.1.1.1:853")
|
conn, err := dialer.DialContext(context.Background(), "tcp", "1.1.1.1:853")
|
||||||
if !errors.Is(err, io.EOF) {
|
if !errors.Is(err, io.EOF) {
|
||||||
t.Fatal("not the error we expected")
|
t.Fatal("not the error we expected")
|
||||||
|
@ -73,8 +83,8 @@ func TestDNSDialerDialForSingleIPFails(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDNSDialerDialForManyIPFails(t *testing.T) {
|
func TestDialerResolverDialForManyIPFails(t *testing.T) {
|
||||||
dialer := &dnsDialer{
|
dialer := &DialerResolver{
|
||||||
Dialer: mockablex.Dialer{
|
Dialer: mockablex.Dialer{
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
return nil, io.EOF
|
return nil, io.EOF
|
||||||
|
@ -91,8 +101,8 @@ func TestDNSDialerDialForManyIPFails(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDNSDialerDialForManyIPSuccess(t *testing.T) {
|
func TestDialerResolverDialForManyIPSuccess(t *testing.T) {
|
||||||
dialer := &dnsDialer{Dialer: mockablex.Dialer{
|
dialer := &DialerResolver{Dialer: mockablex.Dialer{
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
return &mockablex.Conn{
|
return &mockablex.Conn{
|
||||||
MockClose: func() error {
|
MockClose: func() error {
|
||||||
|
@ -113,43 +123,39 @@ func TestDNSDialerDialForManyIPSuccess(t *testing.T) {
|
||||||
conn.Close()
|
conn.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestReduceErrors(t *testing.T) {
|
func TestDialerLoggerFailure(t *testing.T) {
|
||||||
t.Run("no errors", func(t *testing.T) {
|
d := &DialerLogger{
|
||||||
result := ReduceErrors(nil)
|
Dialer: mockablex.Dialer{
|
||||||
if result != nil {
|
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
t.Fatal("wrong result")
|
return nil, io.EOF
|
||||||
}
|
},
|
||||||
})
|
},
|
||||||
|
Logger: log.Log,
|
||||||
t.Run("single error", func(t *testing.T) {
|
}
|
||||||
err := errors.New("mocked error")
|
conn, err := d.DialContext(context.Background(), "tcp", "www.google.com:443")
|
||||||
result := ReduceErrors([]error{err})
|
if !errors.Is(err, io.EOF) {
|
||||||
if result != err {
|
t.Fatal("not the error we expected")
|
||||||
t.Fatal("wrong result")
|
}
|
||||||
}
|
if conn != nil {
|
||||||
})
|
t.Fatal("expected nil conn here")
|
||||||
|
}
|
||||||
t.Run("multiple errors", func(t *testing.T) {
|
}
|
||||||
err1 := errors.New("mocked error #1")
|
|
||||||
err2 := errors.New("mocked error #2")
|
func TestDefaultDialerWorks(t *testing.T) {
|
||||||
result := ReduceErrors([]error{err1, err2})
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
if result.Error() != "mocked error #1" {
|
cancel() // fail immediately
|
||||||
t.Fatal("wrong result")
|
conn, err := DefaultDialer.DialContext(ctx, "tcp", "8.8.8.8:853")
|
||||||
}
|
if err == nil || !strings.HasSuffix(err.Error(), "operation was canceled") {
|
||||||
})
|
t.Fatal("not the error we expected", err)
|
||||||
|
}
|
||||||
t.Run("multiple errors with meaningful ones", func(t *testing.T) {
|
if conn != nil {
|
||||||
err1 := errors.New("mocked error #1")
|
t.Fatal("expected nil conn here")
|
||||||
err2 := &errorx.ErrWrapper{
|
}
|
||||||
Failure: "unknown_failure: antani",
|
}
|
||||||
}
|
|
||||||
err3 := &errorx.ErrWrapper{
|
func TestDefaultDialerHasTimeout(t *testing.T) {
|
||||||
Failure: errorx.FailureConnectionRefused,
|
expected := 15 * time.Second
|
||||||
}
|
if DefaultDialer.Timeout != expected {
|
||||||
err4 := errors.New("mocked error #3")
|
t.Fatal("unexpected timeout value")
|
||||||
result := ReduceErrors([]error{err1, err2, err3, err4})
|
}
|
||||||
if result.Error() != errorx.FailureConnectionRefused {
|
|
||||||
t.Fatal("wrong result")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
46
internal/netxlite/doc.go
Normal file
46
internal/netxlite/doc.go
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
// Package netxlite contains network extensions.
|
||||||
|
//
|
||||||
|
// This package is the basic networking building block that you
|
||||||
|
// should be using every time you need networking.
|
||||||
|
//
|
||||||
|
// Naming and history
|
||||||
|
//
|
||||||
|
// Previous versions of this package were called netx. Compared to such
|
||||||
|
// versions this package is lightweight because it does not contain code
|
||||||
|
// to perform the measurements, hence its name.
|
||||||
|
//
|
||||||
|
// Design
|
||||||
|
//
|
||||||
|
// We want to potentially be able to observe each low-level operation
|
||||||
|
// separately, even though this is not done by this package. This is
|
||||||
|
// the use case where we are performing measurements.
|
||||||
|
//
|
||||||
|
// We also want to be able to use this package in a more casual way
|
||||||
|
// without having to compose each operation separately. This, instead, is
|
||||||
|
// the use case where we're communicating with the OONI backend.
|
||||||
|
//
|
||||||
|
// We want to optionally provide detailed logging of every operation,
|
||||||
|
// thus users can use `-v` to obtain OONI logs.
|
||||||
|
//
|
||||||
|
// We also want to mock any underlying dependency for testing.
|
||||||
|
//
|
||||||
|
// Operations
|
||||||
|
//
|
||||||
|
// This package implements the following operations:
|
||||||
|
//
|
||||||
|
// 1. establishing a TCP connection;
|
||||||
|
//
|
||||||
|
// 2. performing a domain name resolution;
|
||||||
|
//
|
||||||
|
// 3. performing the TLS handshake;
|
||||||
|
//
|
||||||
|
// 4. performing the QUIC handshake;
|
||||||
|
//
|
||||||
|
// 5. dialing with TCP, TLS, and QUIC (where in this context dialing
|
||||||
|
// means combining domain name resolution and "connecting");
|
||||||
|
//
|
||||||
|
// 6. performing HTTP, HTTP2, and HTTP3 round trips.
|
||||||
|
//
|
||||||
|
// Operations 1, 2, 3, and 4 are used when we perform measurements,
|
||||||
|
// while 5 and 6 are mostly used when speaking with our backend.
|
||||||
|
package netxlite
|
39
internal/netxlite/legacy.go
Normal file
39
internal/netxlite/legacy.go
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/errorx"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ReduceErrors finds a known error in a list of errors since
|
||||||
|
// it's probably most relevant.
|
||||||
|
//
|
||||||
|
// Deprecation warning
|
||||||
|
//
|
||||||
|
// Albeit still used, this function is now DEPRECATED.
|
||||||
|
//
|
||||||
|
// In perspective, we would like to transition to a scenario where
|
||||||
|
// full dialing is NOT used for measurements and we return a multierror here.
|
||||||
|
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]
|
||||||
|
}
|
49
internal/netxlite/legacy_test.go
Normal file
49
internal/netxlite/legacy_test.go
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/errorx"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestReduceErrors(t *testing.T) {
|
||||||
|
t.Run("no errors", func(t *testing.T) {
|
||||||
|
result := ReduceErrors(nil)
|
||||||
|
if result != nil {
|
||||||
|
t.Fatal("wrong result")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("single error", func(t *testing.T) {
|
||||||
|
err := errors.New("mocked error")
|
||||||
|
result := ReduceErrors([]error{err})
|
||||||
|
if result != err {
|
||||||
|
t.Fatal("wrong result")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("multiple errors", func(t *testing.T) {
|
||||||
|
err1 := errors.New("mocked error #1")
|
||||||
|
err2 := errors.New("mocked error #2")
|
||||||
|
result := ReduceErrors([]error{err1, err2})
|
||||||
|
if result.Error() != "mocked error #1" {
|
||||||
|
t.Fatal("wrong result")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("multiple errors with meaningful ones", func(t *testing.T) {
|
||||||
|
err1 := errors.New("mocked error #1")
|
||||||
|
err2 := &errorx.ErrWrapper{
|
||||||
|
Failure: "unknown_failure: antani",
|
||||||
|
}
|
||||||
|
err3 := &errorx.ErrWrapper{
|
||||||
|
Failure: errorx.FailureConnectionRefused,
|
||||||
|
}
|
||||||
|
err4 := errors.New("mocked error #3")
|
||||||
|
result := ReduceErrors([]error{err1, err2, err3, err4})
|
||||||
|
if result.Error() != errorx.FailureConnectionRefused {
|
||||||
|
t.Fatal("wrong result")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
7
internal/netxlite/logger.go
Normal file
7
internal/netxlite/logger.go
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
// Logger is the interface we expect from a logger.
|
||||||
|
type Logger interface {
|
||||||
|
// Debugf formats and emits a debug message.
|
||||||
|
Debugf(format string, v ...interface{})
|
||||||
|
}
|
78
internal/netxlite/resolver.go
Normal file
78
internal/netxlite/resolver.go
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Resolver performs domain name resolutions.
|
||||||
|
type Resolver interface {
|
||||||
|
// LookupHost behaves like net.Resolver.LookupHost.
|
||||||
|
LookupHost(ctx context.Context, hostname string) (addrs []string, err error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResolverSystem is the system resolver.
|
||||||
|
type ResolverSystem struct{}
|
||||||
|
|
||||||
|
var _ Resolver = ResolverSystem{}
|
||||||
|
|
||||||
|
// LookupHost implements Resolver.LookupHost.
|
||||||
|
func (r ResolverSystem) LookupHost(ctx context.Context, hostname string) ([]string, error) {
|
||||||
|
return net.DefaultResolver.LookupHost(ctx, hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Network implements Resolver.Network.
|
||||||
|
func (r ResolverSystem) Network() string {
|
||||||
|
return "system"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Address implements Resolver.Address.
|
||||||
|
func (r ResolverSystem) Address() string {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// DefaultResolver is the resolver we use by default.
|
||||||
|
var DefaultResolver = ResolverSystem{}
|
||||||
|
|
||||||
|
// ResolverLogger is a resolver that emits events
|
||||||
|
type ResolverLogger struct {
|
||||||
|
Resolver
|
||||||
|
Logger Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ Resolver = ResolverLogger{}
|
||||||
|
|
||||||
|
// LookupHost returns the IP addresses of a host
|
||||||
|
func (r ResolverLogger) LookupHost(ctx context.Context, hostname string) ([]string, error) {
|
||||||
|
r.Logger.Debugf("resolve %s...", hostname)
|
||||||
|
start := time.Now()
|
||||||
|
addrs, err := r.Resolver.LookupHost(ctx, hostname)
|
||||||
|
stop := time.Now()
|
||||||
|
r.Logger.Debugf("resolve %s... (%+v, %+v) in %s", hostname, addrs, err, stop.Sub(start))
|
||||||
|
return addrs, err
|
||||||
|
}
|
||||||
|
|
||||||
|
type resolverNetworker interface {
|
||||||
|
Network() string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Network implements Resolver.Network.
|
||||||
|
func (r ResolverLogger) Network() string {
|
||||||
|
if rn, ok := r.Resolver.(resolverNetworker); ok {
|
||||||
|
return rn.Network()
|
||||||
|
}
|
||||||
|
return "logger"
|
||||||
|
}
|
||||||
|
|
||||||
|
type resolverAddresser interface {
|
||||||
|
Address() string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Address implements Resolver.Address.
|
||||||
|
func (r ResolverLogger) Address() string {
|
||||||
|
if ra, ok := r.Resolver.(resolverAddresser); ok {
|
||||||
|
return ra.Address()
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
56
internal/netxlite/resolver_test.go
Normal file
56
internal/netxlite/resolver_test.go
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/apex/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestResolverSystemLookupHost(t *testing.T) {
|
||||||
|
r := ResolverSystem{}
|
||||||
|
if r.Network() != "system" {
|
||||||
|
t.Fatal("invalid Network")
|
||||||
|
}
|
||||||
|
if r.Address() != "" {
|
||||||
|
t.Fatal("invalid Address")
|
||||||
|
}
|
||||||
|
addrs, err := r.LookupHost(context.Background(), "dns.google.com")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if addrs == nil {
|
||||||
|
t.Fatal("expected non-nil result here")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestResolverLoggerWithFailure(t *testing.T) {
|
||||||
|
r := ResolverLogger{
|
||||||
|
Logger: log.Log,
|
||||||
|
Resolver: DefaultResolver,
|
||||||
|
}
|
||||||
|
if r.Network() != "system" {
|
||||||
|
t.Fatal("invalid Network")
|
||||||
|
}
|
||||||
|
if r.Address() != "" {
|
||||||
|
t.Fatal("invalid Address")
|
||||||
|
}
|
||||||
|
addrs, err := r.LookupHost(context.Background(), "nonexistent.antani")
|
||||||
|
if err == nil {
|
||||||
|
t.Fatal("expected an error here")
|
||||||
|
}
|
||||||
|
if addrs != nil {
|
||||||
|
t.Fatal("expected nil addr here")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestResolverLoggerDefaultNetworkAddress(t *testing.T) {
|
||||||
|
r := &ResolverLogger{Logger: log.Log, Resolver: &net.Resolver{}}
|
||||||
|
if r.Network() != "logger" {
|
||||||
|
t.Fatal("invalid Network")
|
||||||
|
}
|
||||||
|
if r.Address() != "" {
|
||||||
|
t.Fatal("invalid Address")
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user