b9a844ecee
* feat: run ~always netxlite integration tests This diff ensures that we check on windows, linux, macos that our fundamental networking library (netxlite) works. We combine unit and integration tests. This work is part of https://github.com/ooni/probe/issues/1733, where I want to have more strong guarantees about the foundations. * fix(filtering/tls_test.go): make portable on Windows The trick here is to use the wrapped error so to normalize the different errors messages we see on Windows. * fix(netxlite/quic_test.go): make portable on windows Rather than using the zero port, use the `x` port which fails when the stdlib is parsing the address. The zero port seems to work on Windows while it does not on Unix. * fix(serialresolver_test.go): make error more timeout than before This seems enough to convince Go on Windows about this error being really a timeout timeouty timeouted thingie.
276 lines
7.2 KiB
Go
276 lines
7.2 KiB
Go
package netxlite
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"errors"
|
|
"net"
|
|
"testing"
|
|
|
|
"github.com/miekg/dns"
|
|
"github.com/ooni/probe-cli/v3/internal/atomicx"
|
|
"github.com/ooni/probe-cli/v3/internal/netxlite/mocks"
|
|
)
|
|
|
|
// errorWithTimeout is an error that golang will always consider
|
|
// to be a timeout because it has a Timeout() bool method
|
|
type errorWithTimeout struct {
|
|
error
|
|
}
|
|
|
|
// Timeout returns whether this error is a timeout.
|
|
func (err *errorWithTimeout) Timeout() bool {
|
|
return true
|
|
}
|
|
|
|
// Unwrap allows to unwrap the error.
|
|
func (err *errorWithTimeout) Unwrap() error {
|
|
return err.error
|
|
}
|
|
|
|
func TestSerialResolver(t *testing.T) {
|
|
t.Run("transport okay", func(t *testing.T) {
|
|
txp := NewDNSOverTLS((&tls.Dialer{}).DialContext, "8.8.8.8:853")
|
|
r := NewSerialResolver(txp)
|
|
rtx := r.Transport()
|
|
if rtx.Network() != "dot" || rtx.Address() != "8.8.8.8:853" {
|
|
t.Fatal("not the transport we expected")
|
|
}
|
|
if r.Network() != rtx.Network() {
|
|
t.Fatal("invalid network seen from the resolver")
|
|
}
|
|
if r.Address() != rtx.Address() {
|
|
t.Fatal("invalid address seen from the resolver")
|
|
}
|
|
})
|
|
|
|
t.Run("LookupHost", func(t *testing.T) {
|
|
t.Run("Encode error", func(t *testing.T) {
|
|
mocked := errors.New("mocked error")
|
|
txp := NewDNSOverTLS((&tls.Dialer{}).DialContext, "8.8.8.8:853")
|
|
r := SerialResolver{
|
|
Encoder: &mocks.DNSEncoder{
|
|
MockEncode: func(domain string, qtype uint16, padding bool) ([]byte, error) {
|
|
return nil, mocked
|
|
},
|
|
},
|
|
Txp: txp,
|
|
}
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if !errors.Is(err, mocked) {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if addrs != nil {
|
|
t.Fatal("expected nil address here")
|
|
}
|
|
})
|
|
|
|
t.Run("RoundTrip error", func(t *testing.T) {
|
|
mocked := errors.New("mocked error")
|
|
txp := &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return nil, mocked
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return true
|
|
},
|
|
}
|
|
r := NewSerialResolver(txp)
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if !errors.Is(err, mocked) {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if addrs != nil {
|
|
t.Fatal("expected nil address here")
|
|
}
|
|
})
|
|
|
|
t.Run("empty reply", func(t *testing.T) {
|
|
txp := &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return dnsGenLookupHostReplySuccess(t, dns.TypeA), nil
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return true
|
|
},
|
|
}
|
|
r := NewSerialResolver(txp)
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if !errors.Is(err, ErrOODNSNoAnswer) {
|
|
t.Fatal("not the error we expected", err)
|
|
}
|
|
if addrs != nil {
|
|
t.Fatal("expected nil address here")
|
|
}
|
|
})
|
|
|
|
t.Run("with A reply", func(t *testing.T) {
|
|
txp := &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return dnsGenLookupHostReplySuccess(t, dns.TypeA, "8.8.8.8"), nil
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return true
|
|
},
|
|
}
|
|
r := NewSerialResolver(txp)
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(addrs) != 1 || addrs[0] != "8.8.8.8" {
|
|
t.Fatal("not the result we expected")
|
|
}
|
|
})
|
|
|
|
t.Run("with AAAA reply", func(t *testing.T) {
|
|
txp := &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return dnsGenLookupHostReplySuccess(t, dns.TypeAAAA, "::1"), nil
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return true
|
|
},
|
|
}
|
|
r := NewSerialResolver(txp)
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if len(addrs) != 1 || addrs[0] != "::1" {
|
|
t.Fatal("not the result we expected")
|
|
}
|
|
})
|
|
|
|
t.Run("with timeout", func(t *testing.T) {
|
|
txp := &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return nil, &net.OpError{
|
|
Err: &errorWithTimeout{ETIMEDOUT},
|
|
Op: "dial",
|
|
}
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return true
|
|
},
|
|
}
|
|
r := NewSerialResolver(txp)
|
|
addrs, err := r.LookupHost(context.Background(), "www.gogle.com")
|
|
if !errors.Is(err, ETIMEDOUT) {
|
|
t.Fatal("not the error we expected")
|
|
}
|
|
if addrs != nil {
|
|
t.Fatal("expected nil address here")
|
|
}
|
|
if r.NumTimeouts.Load() <= 0 {
|
|
t.Fatal("we didn't actually take the timeouts")
|
|
}
|
|
})
|
|
})
|
|
|
|
t.Run("CloseIdleConnections", func(t *testing.T) {
|
|
var called bool
|
|
r := &SerialResolver{
|
|
Txp: &mocks.DNSTransport{
|
|
MockCloseIdleConnections: func() {
|
|
called = true
|
|
},
|
|
},
|
|
}
|
|
r.CloseIdleConnections()
|
|
if !called {
|
|
t.Fatal("not called")
|
|
}
|
|
})
|
|
|
|
t.Run("LookupHTTPS", func(t *testing.T) {
|
|
t.Run("for encoding error", func(t *testing.T) {
|
|
expected := errors.New("mocked error")
|
|
r := &SerialResolver{
|
|
Encoder: &mocks.DNSEncoder{
|
|
MockEncode: func(domain string, qtype uint16, padding bool) ([]byte, error) {
|
|
return nil, expected
|
|
},
|
|
},
|
|
Decoder: nil,
|
|
NumTimeouts: &atomicx.Int64{},
|
|
Txp: &mocks.DNSTransport{
|
|
MockRequiresPadding: func() bool {
|
|
return false
|
|
},
|
|
},
|
|
}
|
|
ctx := context.Background()
|
|
https, err := r.LookupHTTPS(ctx, "example.com")
|
|
if !errors.Is(err, expected) {
|
|
t.Fatal("unexpected err", err)
|
|
}
|
|
if https != nil {
|
|
t.Fatal("unexpected result")
|
|
}
|
|
})
|
|
|
|
t.Run("for round-trip error", func(t *testing.T) {
|
|
expected := errors.New("mocked error")
|
|
r := &SerialResolver{
|
|
Encoder: &mocks.DNSEncoder{
|
|
MockEncode: func(domain string, qtype uint16, padding bool) ([]byte, error) {
|
|
return make([]byte, 64), nil
|
|
},
|
|
},
|
|
Decoder: nil,
|
|
NumTimeouts: &atomicx.Int64{},
|
|
Txp: &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return nil, expected
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return false
|
|
},
|
|
},
|
|
}
|
|
ctx := context.Background()
|
|
https, err := r.LookupHTTPS(ctx, "example.com")
|
|
if !errors.Is(err, expected) {
|
|
t.Fatal("unexpected err", err)
|
|
}
|
|
if https != nil {
|
|
t.Fatal("unexpected result")
|
|
}
|
|
})
|
|
|
|
t.Run("for decode error", func(t *testing.T) {
|
|
expected := errors.New("mocked error")
|
|
r := &SerialResolver{
|
|
Encoder: &mocks.DNSEncoder{
|
|
MockEncode: func(domain string, qtype uint16, padding bool) ([]byte, error) {
|
|
return make([]byte, 64), nil
|
|
},
|
|
},
|
|
Decoder: &mocks.DNSDecoder{
|
|
MockDecodeHTTPS: func(reply []byte) (*mocks.HTTPSSvc, error) {
|
|
return nil, expected
|
|
},
|
|
},
|
|
NumTimeouts: &atomicx.Int64{},
|
|
Txp: &mocks.DNSTransport{
|
|
MockRoundTrip: func(ctx context.Context, query []byte) (reply []byte, err error) {
|
|
return make([]byte, 128), nil
|
|
},
|
|
MockRequiresPadding: func() bool {
|
|
return false
|
|
},
|
|
},
|
|
}
|
|
ctx := context.Background()
|
|
https, err := r.LookupHTTPS(ctx, "example.com")
|
|
if !errors.Is(err, expected) {
|
|
t.Fatal("unexpected err", err)
|
|
}
|
|
if https != nil {
|
|
t.Fatal("unexpected result")
|
|
}
|
|
})
|
|
})
|
|
}
|