refactor(netxlite): expose useful HTTPTransport/DNSTransport factories (#813)

These factories will soon be useful to finish with
https://github.com/ooni/probe/issues/2135.
This commit is contained in:
Simone Basso 2022-06-09 00:30:18 +02:00 committed by GitHub
parent 1a706e47bc
commit 1685ef75b5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 224 additions and 30 deletions

View File

@ -29,8 +29,7 @@ func init() {
// puzzling https://github.com/ooni/probe/issues/1409 issue. // puzzling https://github.com/ooni/probe/issues/1409 issue.
const resolverURL = "https://8.8.8.8/dns-query" const resolverURL = "https://8.8.8.8/dns-query"
resolver = netxlite.NewParallelDNSOverHTTPSResolver(log.Log, resolverURL) resolver = netxlite.NewParallelDNSOverHTTPSResolver(log.Log, resolverURL)
txp := netxlite.NewHTTPTransportWithResolver(log.Log, resolver) httpClient = netxlite.NewHTTPClientWithResolver(log.Log, resolver)
httpClient = netxlite.NewHTTPClient(txp)
} }
func main() { func main() {

View File

@ -32,8 +32,7 @@ func init() {
// default resolver configured by the box. Also, use an encrypted transport thus // default resolver configured by the box. Also, use an encrypted transport thus
// we're less vulnerable to any policy implemented by the box's provider. // we're less vulnerable to any policy implemented by the box's provider.
resolver = netxlite.NewParallelDNSOverHTTPSResolver(log.Log, "https://8.8.8.8/dns-query") resolver = netxlite.NewParallelDNSOverHTTPSResolver(log.Log, "https://8.8.8.8/dns-query")
txp := netxlite.NewHTTPTransportWithResolver(log.Log, resolver) httpClient = netxlite.NewHTTPClientWithResolver(log.Log, resolver)
httpClient = netxlite.NewHTTPClient(txp)
} }
func shutdown(srv *http.Server) { func shutdown(srv *http.Server) {

View File

@ -197,11 +197,9 @@ func NewSession(ctx context.Context, config SessionConfig) (*Session, error) {
Logger: sess.logger, Logger: sess.logger,
ProxyURL: proxyURL, ProxyURL: proxyURL,
} }
dialer := netxlite.NewDialerWithResolver(sess.logger, sess.resolver) txp := netxlite.NewHTTPTransportWithLoggerResolverAndOptionalProxyURL(
dialer = netxlite.MaybeWrapWithProxyDialer(dialer, proxyURL) sess.logger, sess.resolver, sess.proxyURL,
handshaker := netxlite.NewTLSHandshakerStdlib(sess.logger) )
tlsDialer := netxlite.NewTLSDialer(dialer, handshaker)
txp := netxlite.NewHTTPTransport(sess.logger, dialer, tlsDialer)
txp = bytecounter.WrapHTTPTransport(txp, sess.byteCounter) txp = bytecounter.WrapHTTPTransport(txp, sess.byteCounter)
sess.httpDefaultTransport = txp sess.httpDefaultTransport = txp
return sess, nil return sess, nil

View File

@ -43,6 +43,18 @@ func NewUnwrappedDNSOverHTTPSTransport(client model.HTTPClient, URL string) *DNS
return NewUnwrappedDNSOverHTTPSTransportWithHostOverride(client, URL, "") return NewUnwrappedDNSOverHTTPSTransportWithHostOverride(client, URL, "")
} }
// NewDNSOverHTTPSTransport is like NewUnwrappedDNSOverHTTPSTransport but
// returns an already wrapped DNSTransport.
func NewDNSOverHTTPSTransport(client model.HTTPClient, URL string) model.DNSTransport {
return WrapDNSTransport(NewUnwrappedDNSOverHTTPSTransport(client, URL))
}
// NewDNSOverHTTPSTransportWithHTTPTransport is like NewDNSOverHTTPSTransport
// but takes in input an HTTPTransport rather than an HTTPClient.
func NewDNSOverHTTPSTransportWithHTTPTransport(txp model.HTTPTransport, URL string) model.DNSTransport {
return WrapDNSTransport(NewUnwrappedDNSOverHTTPSTransport(NewHTTPClient(txp), URL))
}
// NewUnwrappedDNSOverHTTPSTransportWithHostOverride creates a new DNSOverHTTPSTransport // NewUnwrappedDNSOverHTTPSTransportWithHostOverride creates a new DNSOverHTTPSTransport
// with the given Host header override. This instance has not been wrapped yet. // with the given Host header override. This instance has not been wrapped yet.
func NewUnwrappedDNSOverHTTPSTransportWithHostOverride( func NewUnwrappedDNSOverHTTPSTransportWithHostOverride(

View File

@ -13,6 +13,36 @@ import (
"github.com/ooni/probe-cli/v3/internal/model/mocks" "github.com/ooni/probe-cli/v3/internal/model/mocks"
) )
func TestNewDNSOverHTTPSTransport(t *testing.T) {
const URL = "https://1.1.1.1/dns-query"
clnt := NewHTTPClientStdlib(model.DiscardLogger)
txp := NewDNSOverHTTPSTransport(clnt, URL)
ew := txp.(*dnsTransportErrWrapper)
https := ew.DNSTransport.(*DNSOverHTTPSTransport)
if https.Client != clnt {
t.Fatal("invalid client")
}
if https.URL != URL {
t.Fatal("invalid URL")
}
}
func TestNewDNSOverHTTPSTransportWithHTTPTransport(t *testing.T) {
const URL = "https://1.1.1.1/dns-query"
httpTxp := NewHTTPTransportStdlib(model.DiscardLogger)
txp := NewDNSOverHTTPSTransportWithHTTPTransport(httpTxp, URL)
ew := txp.(*dnsTransportErrWrapper)
https := ew.DNSTransport.(*DNSOverHTTPSTransport)
ewClient := https.Client.(*httpClientErrWrapper)
clnt := ewClient.HTTPClient.(*http.Client)
if clnt.Transport != httpTxp {
t.Fatal("invalid transport")
}
if https.URL != URL {
t.Fatal("invalid URL")
}
}
func TestDNSOverHTTPSTransport(t *testing.T) { func TestDNSOverHTTPSTransport(t *testing.T) {
t.Run("RoundTrip", func(t *testing.T) { t.Run("RoundTrip", func(t *testing.T) {
t.Run("query serialization failure", func(t *testing.T) { t.Run("query serialization failure", func(t *testing.T) {

View File

@ -9,6 +9,7 @@ import (
"errors" "errors"
"net" "net"
"net/http" "net/http"
"net/url"
"time" "time"
oohttp "github.com/ooni/oohttp" oohttp "github.com/ooni/oohttp"
@ -105,6 +106,25 @@ func (txp *httpTransportConnectionsCloser) CloseIdleConnections() {
txp.TLSDialer.CloseIdleConnections() txp.TLSDialer.CloseIdleConnections()
} }
// NewHTTPTransportWithLoggerResolverAndOptionalProxyURL creates an HTTPTransport using
// the given logger and resolver and an optional proxy URL.
//
// Arguments:
//
// - logger is the MANDATORY logger;
//
// - resolver is the MANDATORY resolver;
//
// - purl is the OPTIONAL proxy URL.
func NewHTTPTransportWithLoggerResolverAndOptionalProxyURL(
logger model.DebugLogger, resolver model.Resolver, purl *url.URL) model.HTTPTransport {
dialer := NewDialerWithResolver(logger, resolver)
dialer = MaybeWrapWithProxyDialer(dialer, purl)
handshaker := NewTLSHandshakerStdlib(logger)
tlsDialer := NewTLSDialer(dialer, handshaker)
return NewHTTPTransport(logger, dialer, tlsDialer)
}
// NewHTTPTransportWithResolver creates a new HTTP transport using // NewHTTPTransportWithResolver creates a new HTTP transport using
// the stdlib for everything but the given resolver. // the stdlib for everything but the given resolver.
func NewHTTPTransportWithResolver(logger model.DebugLogger, reso model.Resolver) model.HTTPTransport { func NewHTTPTransportWithResolver(logger model.DebugLogger, reso model.Resolver) model.HTTPTransport {
@ -335,6 +355,12 @@ func NewHTTPClientStdlib(logger model.DebugLogger) model.HTTPClient {
return NewHTTPClient(txp) return NewHTTPClient(txp)
} }
// NewHTTPClientWithResolver creates a new HTTPTransport using the
// given resolver and then from that builds an HTTPClient.
func NewHTTPClientWithResolver(logger model.Logger, reso model.Resolver) model.HTTPClient {
return NewHTTPClient(NewHTTPTransportWithResolver(logger, reso))
}
// NewHTTPClient creates a new, wrapped HTTPClient using the given transport. // NewHTTPClient creates a new, wrapped HTTPClient using the given transport.
func NewHTTPClient(txp model.HTTPTransport) model.HTTPClient { func NewHTTPClient(txp model.HTTPTransport) model.HTTPClient {
return WrapHTTPClient(&http.Client{Transport: txp}) return WrapHTTPClient(&http.Client{Transport: txp})

View File

@ -61,3 +61,19 @@ func NewHTTP3Transport(
dialer: dialer, dialer: dialer,
}) })
} }
// NewHTTP3TransportStdlib creates a new HTTPTransport using http3 that
// uses standard functionality for everything but the logger.
func NewHTTP3TransportStdlib(logger model.DebugLogger) model.HTTPTransport {
ql := NewQUICListener()
reso := NewStdlibResolver(logger)
qd := NewQUICDialerWithResolver(ql, logger, reso)
return NewHTTP3Transport(logger, qd, nil)
}
// NewHTTPTransportWithResolver creates a new HTTPTransport using http3
// that uses the given logger and the given resolver.
func NewHTTP3TransportWithResolver(logger model.Logger, reso model.Resolver) model.HTTPTransport {
qd := NewQUICDialerWithResolver(NewQUICListener(), logger, reso)
return NewHTTP3Transport(logger, qd, nil)
}

View File

@ -6,8 +6,8 @@ import (
"net/http" "net/http"
"testing" "testing"
"github.com/apex/log"
"github.com/lucas-clemente/quic-go/http3" "github.com/lucas-clemente/quic-go/http3"
"github.com/ooni/probe-cli/v3/internal/model"
"github.com/ooni/probe-cli/v3/internal/model/mocks" "github.com/ooni/probe-cli/v3/internal/model/mocks"
nlmocks "github.com/ooni/probe-cli/v3/internal/netxlite/mocks" nlmocks "github.com/ooni/probe-cli/v3/internal/netxlite/mocks"
) )
@ -63,20 +63,46 @@ func TestHTTP3Transport(t *testing.T) {
}) })
} }
func TestNewHTTP3Transport(t *testing.T) { // verifyTypeChainForHTTP3 helps to verify type chains for HTTP3.
t.Run("creates the correct type chain", func(t *testing.T) { //
qd := &mocks.QUICDialer{} // Arguments:
config := &tls.Config{} //
txp := NewHTTP3Transport(log.Log, qd, config) // - t is the MANDATORY testing ref;
//
// - txp is the MANDATORY HTTP transport to verify;
//
// - underlyingLogger is the MANDATORY logger we expect to find;
//
// - qd is the OPTIONAL QUIC dialer: if not nil, we expect to
// see this value as the QUIC dialer, otherwise we will check the
// type chain of the real dialer;
//
// - config is the MANDATORY TLS config: we'll always check
// whether the TLSClientConfig is equal to this value: passing
// nil here means we expect to see nil in the object;
//
// - reso is the OPTIONAL resolver: if present and the qd is
// nil, we'll unwrap the QUIC dialer and check whether we have
// this resolver as the underlying resolver.
func verifyTypeChainForHTTP3(t *testing.T, txp model.HTTPTransport,
underlyingLogger model.DebugLogger, qd model.QUICDialer,
config *tls.Config, reso model.Resolver) {
logger := txp.(*httpTransportLogger) logger := txp.(*httpTransportLogger)
if logger.Logger != log.Log { if logger.Logger != underlyingLogger {
t.Fatal("invalid logger") t.Fatal("invalid logger")
} }
ew := logger.HTTPTransport.(*httpTransportErrWrapper) ew := logger.HTTPTransport.(*httpTransportErrWrapper)
h3txp := ew.HTTPTransport.(*http3Transport) h3txp := ew.HTTPTransport.(*http3Transport)
if h3txp.dialer != qd { if qd != nil && h3txp.dialer != qd {
t.Fatal("invalid dialer") t.Fatal("invalid dialer")
} }
if qd == nil {
qdlog := h3txp.dialer.(*quicDialerLogger)
qdr := qdlog.Dialer.(*quicDialerResolver)
if reso != nil && qdr.Resolver != reso {
t.Fatal("invalid resolver")
}
}
h3 := h3txp.child.(*http3.RoundTripper) h3 := h3txp.child.(*http3.RoundTripper)
if h3.Dial == nil { if h3.Dial == nil {
t.Fatal("invalid Dial") t.Fatal("invalid Dial")
@ -87,5 +113,28 @@ func TestNewHTTP3Transport(t *testing.T) {
if h3.TLSClientConfig != config { if h3.TLSClientConfig != config {
t.Fatal("invalid TLSClientConfig") t.Fatal("invalid TLSClientConfig")
} }
}
func TestNewHTTP3Transport(t *testing.T) {
t.Run("creates the correct type chain", func(t *testing.T) {
qd := &mocks.QUICDialer{}
config := &tls.Config{}
txp := NewHTTP3Transport(model.DiscardLogger, qd, config)
verifyTypeChainForHTTP3(t, txp, model.DiscardLogger, qd, config, nil)
})
}
func TestNewHTTP3TransportStdlib(t *testing.T) {
t.Run("creates the correct type chain", func(t *testing.T) {
txp := NewHTTP3TransportStdlib(model.DiscardLogger)
verifyTypeChainForHTTP3(t, txp, model.DiscardLogger, nil, nil, nil)
})
}
func TestNewHTTP3TransportWithResolver(t *testing.T) {
t.Run("creates the correct type chain", func(t *testing.T) {
reso := &mocks.Resolver{}
txp := NewHTTP3TransportWithResolver(model.DiscardLogger, reso)
verifyTypeChainForHTTP3(t, txp, model.DiscardLogger, nil, nil, reso)
}) })
} }

View File

@ -6,6 +6,7 @@ import (
"io" "io"
"net" "net"
"net/http" "net/http"
"net/url"
"strings" "strings"
"testing" "testing"
"time" "time"
@ -16,6 +17,48 @@ import (
"github.com/ooni/probe-cli/v3/internal/model/mocks" "github.com/ooni/probe-cli/v3/internal/model/mocks"
) )
func TestNewHTTPTransportWithLoggerResolverAndOptionalProxyURL(t *testing.T) {
t.Run("without proxy URL", func(t *testing.T) {
logger := &mocks.Logger{}
resolver := &mocks.Resolver{}
txp := NewHTTPTransportWithLoggerResolverAndOptionalProxyURL(logger, resolver, nil)
txpLogger := txp.(*httpTransportLogger)
if txpLogger.Logger != logger {
t.Fatal("unexpected logger")
}
txpErrWrapper := txpLogger.HTTPTransport.(*httpTransportErrWrapper)
txpCc := txpErrWrapper.HTTPTransport.(*httpTransportConnectionsCloser)
dialer := txpCc.Dialer
dialerWithReadTimeout := dialer.(*httpDialerWithReadTimeout)
dialerLog := dialerWithReadTimeout.Dialer.(*dialerLogger)
dialerReso := dialerLog.Dialer.(*dialerResolver)
if dialerReso.Resolver != resolver {
t.Fatal("invalid resolver")
}
})
t.Run("with proxy URL", func(t *testing.T) {
URL := &url.URL{}
logger := &mocks.Logger{}
resolver := &mocks.Resolver{}
txp := NewHTTPTransportWithLoggerResolverAndOptionalProxyURL(logger, resolver, URL)
txpLogger := txp.(*httpTransportLogger)
if txpLogger.Logger != logger {
t.Fatal("unexpected logger")
}
txpErrWrapper := txpLogger.HTTPTransport.(*httpTransportErrWrapper)
txpCc := txpErrWrapper.HTTPTransport.(*httpTransportConnectionsCloser)
dialer := txpCc.Dialer
dialerWithReadTimeout := dialer.(*httpDialerWithReadTimeout)
dialerProxy := dialerWithReadTimeout.Dialer.(*proxyDialer)
dialerLog := dialerProxy.Dialer.(*dialerLogger)
dialerReso := dialerLog.Dialer.(*dialerResolver)
if dialerReso.Resolver != resolver {
t.Fatal("invalid resolver")
}
})
}
func TestNewHTTPTransportWithResolver(t *testing.T) { func TestNewHTTPTransportWithResolver(t *testing.T) {
expected := errors.New("mocked error") expected := errors.New("mocked error")
reso := &mocks.Resolver{ reso := &mocks.Resolver{
@ -553,6 +596,28 @@ func TestNewHTTPClientStdlib(t *testing.T) {
} }
} }
func TestNewHTTPClientWithResolver(t *testing.T) {
reso := &mocks.Resolver{}
clnt := NewHTTPClientWithResolver(model.DiscardLogger, reso)
ewc, ok := clnt.(*httpClientErrWrapper)
if !ok {
t.Fatal("expected *httpClientErrWrapper")
}
httpClnt, ok := ewc.HTTPClient.(*http.Client)
if !ok {
t.Fatal("expected *http.Client")
}
txp := httpClnt.Transport.(*httpTransportLogger)
txpEwrap := txp.HTTPTransport.(*httpTransportErrWrapper)
txpCc := txpEwrap.HTTPTransport.(*httpTransportConnectionsCloser)
dialer := txpCc.Dialer.(*httpDialerWithReadTimeout)
dialerLogger := dialer.Dialer.(*dialerLogger)
dialerReso := dialerLogger.Dialer.(*dialerResolver)
if dialerReso.Resolver != reso {
t.Fatal("invalid resolver")
}
}
func TestWrapHTTPClient(t *testing.T) { func TestWrapHTTPClient(t *testing.T) {
origClient := &http.Client{} origClient := &http.Client{}
wrapped := WrapHTTPClient(origClient) wrapped := WrapHTTPClient(origClient)