refactor: move base http3 transport into netxlite (#412)
This diff is part of https://github.com/ooni/probe/issues/1505. You will notice that I have not adapted all the (great) tests we had previously. They should live at another layer, and namely the one that deals with performing measurements. When I'm refactoring such a layer I'll ensure those tests that I have not adapted here are reintroduced into the tree.
This commit is contained in:
parent
527e1a0707
commit
4dc2907472
|
@ -1,43 +1,10 @@
|
||||||
package httptransport
|
package httptransport
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
"crypto/tls"
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go"
|
|
||||||
"github.com/lucas-clemente/quic-go/http3"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/quicdialer"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// QUICWrapperDialer is a QUICDialer that wraps a ContextDialer
|
|
||||||
// This is necessary because the http3 RoundTripper does not support a DialContext method.
|
|
||||||
type QUICWrapperDialer struct {
|
|
||||||
Dialer quicdialer.ContextDialer
|
|
||||||
}
|
|
||||||
|
|
||||||
// Dial implements QUICDialer.Dial
|
|
||||||
func (d QUICWrapperDialer) Dial(network, host string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error) {
|
|
||||||
return d.Dialer.DialContext(context.Background(), network, host, tlsCfg, cfg)
|
|
||||||
}
|
|
||||||
|
|
||||||
// HTTP3Transport is a httptransport.RoundTripper using the http3 protocol.
|
|
||||||
type HTTP3Transport struct {
|
|
||||||
http3.RoundTripper
|
|
||||||
}
|
|
||||||
|
|
||||||
// CloseIdleConnections closes all the connections opened by this transport.
|
|
||||||
func (t *HTTP3Transport) CloseIdleConnections() {
|
|
||||||
t.RoundTripper.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewHTTP3Transport creates a new HTTP3Transport instance.
|
// NewHTTP3Transport creates a new HTTP3Transport instance.
|
||||||
func NewHTTP3Transport(config Config) RoundTripper {
|
func NewHTTP3Transport(config Config) RoundTripper {
|
||||||
txp := &HTTP3Transport{}
|
return netxlite.NewHTTP3Transport(config.QUICDialer, config.TLSConfig)
|
||||||
txp.QuicConfig = &quic.Config{}
|
|
||||||
txp.TLSClientConfig = config.TLSConfig
|
|
||||||
txp.Dial = config.QUICDialer.Dial
|
|
||||||
return txp
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ RoundTripper = &http.Transport{}
|
|
||||||
|
|
|
@ -1,166 +1,12 @@
|
||||||
package httptransport_test
|
package httptransport_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"crypto/tls"
|
|
||||||
"crypto/x509"
|
|
||||||
"errors"
|
|
||||||
"net"
|
|
||||||
"net/http"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/dialer"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/httptransport"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/httptransport"
|
||||||
)
|
)
|
||||||
|
|
||||||
type MockQUICDialer struct{}
|
func TestNewHTTP3Transport(t *testing.T) {
|
||||||
|
// mainly to cover a line which otherwise won't be directly covered
|
||||||
func (d MockQUICDialer) Dial(network, host string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error) {
|
httptransport.NewHTTP3Transport(httptransport.Config{})
|
||||||
return quic.DialAddrEarly(host, tlsCfg, cfg)
|
|
||||||
}
|
|
||||||
|
|
||||||
type MockSNIQUICDialer struct {
|
|
||||||
namech chan string
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d MockSNIQUICDialer) Dial(network, host string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error) {
|
|
||||||
d.namech <- tlsCfg.ServerName
|
|
||||||
return quic.DialAddrEarly(host, tlsCfg, cfg)
|
|
||||||
}
|
|
||||||
|
|
||||||
type MockCertQUICDialer struct {
|
|
||||||
certch chan *x509.CertPool
|
|
||||||
}
|
|
||||||
|
|
||||||
func (d MockCertQUICDialer) Dial(network, host string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error) {
|
|
||||||
d.certch <- tlsCfg.RootCAs
|
|
||||||
return quic.DialAddrEarly(host, tlsCfg, cfg)
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestHTTP3TransportSNI(t *testing.T) {
|
|
||||||
namech := make(chan string, 1)
|
|
||||||
sni := "sni.org"
|
|
||||||
txp := httptransport.NewHTTP3Transport(httptransport.Config{
|
|
||||||
Dialer: dialer.New(&dialer.Config{}, &net.Resolver{}),
|
|
||||||
QUICDialer: MockSNIQUICDialer{namech: namech},
|
|
||||||
TLSConfig: &tls.Config{ServerName: sni}})
|
|
||||||
req, err := http.NewRequest("GET", "https://www.google.com", nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
resp, err := txp.RoundTrip(req)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("expected error here")
|
|
||||||
}
|
|
||||||
if resp != nil {
|
|
||||||
t.Fatal("expected nil resp here")
|
|
||||||
}
|
|
||||||
if !strings.Contains(err.Error(), "certificate is valid for www.google.com, not "+sni) {
|
|
||||||
t.Fatal("unexpected error type", err)
|
|
||||||
}
|
|
||||||
servername := <-namech
|
|
||||||
if servername != sni {
|
|
||||||
t.Fatal("unexpected server name", servername)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestHTTP3TransportSNINoVerify(t *testing.T) {
|
|
||||||
namech := make(chan string, 1)
|
|
||||||
sni := "sni.org"
|
|
||||||
txp := httptransport.NewHTTP3Transport(httptransport.Config{
|
|
||||||
Dialer: dialer.New(&dialer.Config{}, &net.Resolver{}),
|
|
||||||
QUICDialer: MockSNIQUICDialer{namech: namech},
|
|
||||||
TLSConfig: &tls.Config{ServerName: sni, InsecureSkipVerify: true}})
|
|
||||||
req, err := http.NewRequest("GET", "https://www.google.com", nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
resp, err := txp.RoundTrip(req)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("unexpected error: %+v", err)
|
|
||||||
}
|
|
||||||
if resp == nil {
|
|
||||||
t.Fatal("unexpected nil resp")
|
|
||||||
}
|
|
||||||
servername := <-namech
|
|
||||||
if servername != sni {
|
|
||||||
t.Fatal("unexpected server name", servername)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestHTTP3TransportCABundle(t *testing.T) {
|
|
||||||
certch := make(chan *x509.CertPool, 1)
|
|
||||||
certpool := x509.NewCertPool()
|
|
||||||
txp := httptransport.NewHTTP3Transport(httptransport.Config{
|
|
||||||
Dialer: dialer.New(&dialer.Config{}, &net.Resolver{}),
|
|
||||||
QUICDialer: MockCertQUICDialer{certch: certch},
|
|
||||||
TLSConfig: &tls.Config{RootCAs: certpool}})
|
|
||||||
req, err := http.NewRequest("GET", "https://www.google.com", nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
resp, err := txp.RoundTrip(req)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("expected error here")
|
|
||||||
}
|
|
||||||
if resp != nil {
|
|
||||||
t.Fatal("expected nil resp here")
|
|
||||||
}
|
|
||||||
// since the certificate pool is empty, the unknown authority error should be thrown
|
|
||||||
if !strings.Contains(err.Error(), "certificate signed by unknown authority") {
|
|
||||||
t.Fatal("unexpected error type")
|
|
||||||
}
|
|
||||||
certs := <-certch
|
|
||||||
if certs != certpool {
|
|
||||||
t.Fatal("not the certpool we expected")
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUnitHTTP3TransportSuccess(t *testing.T) {
|
|
||||||
txp := httptransport.NewHTTP3Transport(httptransport.Config{
|
|
||||||
Dialer: dialer.New(&dialer.Config{}, &net.Resolver{}),
|
|
||||||
QUICDialer: MockQUICDialer{}})
|
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", "https://www.google.com", nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
resp, err := txp.RoundTrip(req)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
if resp == nil {
|
|
||||||
t.Fatal("unexpected nil response here")
|
|
||||||
}
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
t.Fatal("HTTP statuscode should be 200 OK", resp.StatusCode)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestUnitHTTP3TransportFailure(t *testing.T) {
|
|
||||||
txp := httptransport.NewHTTP3Transport(httptransport.Config{
|
|
||||||
Dialer: dialer.New(&dialer.Config{}, &net.Resolver{}),
|
|
||||||
QUICDialer: MockQUICDialer{}})
|
|
||||||
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
|
||||||
cancel() // so that the request immediately fails
|
|
||||||
req, err := http.NewRequestWithContext(ctx, "GET", "https://www.google.com", nil)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
resp, err := txp.RoundTrip(req)
|
|
||||||
if err == nil {
|
|
||||||
t.Fatal("expected error here")
|
|
||||||
}
|
|
||||||
// context.Canceled error occurs if the test host supports QUIC
|
|
||||||
// timeout error ("Handshake did not complete in time") occurs if the test host does not support QUIC
|
|
||||||
if !(errors.Is(err, context.Canceled) || strings.HasSuffix(err.Error(), "Handshake did not complete in time")) {
|
|
||||||
t.Fatal("not the error we expected", err)
|
|
||||||
}
|
|
||||||
if resp != nil {
|
|
||||||
t.Fatal("expected nil response here")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,7 +30,7 @@ type TLSDialer interface {
|
||||||
|
|
||||||
// QUICDialer is the definition of dialer for QUIC assumed by this package.
|
// QUICDialer is the definition of dialer for QUIC assumed by this package.
|
||||||
type QUICDialer interface {
|
type QUICDialer interface {
|
||||||
Dial(network, addr string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error)
|
DialContext(ctx context.Context, network, addr string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RoundTripper is the definition of http.RoundTripper used by this package.
|
// RoundTripper is the definition of http.RoundTripper used by this package.
|
||||||
|
|
|
@ -54,7 +54,7 @@ type Dialer interface {
|
||||||
|
|
||||||
// QUICDialer is the definition of a dialer for QUIC assumed by this package.
|
// QUICDialer is the definition of a dialer for QUIC assumed by this package.
|
||||||
type QUICDialer interface {
|
type QUICDialer interface {
|
||||||
Dial(network, addr string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error)
|
DialContext(ctx context.Context, network, addr string, tlsCfg *tls.Config, cfg *quic.Config) (quic.EarlySession, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// TLSDialer is the definition of a TLS dialer assumed by this package.
|
// TLSDialer is the definition of a TLS dialer assumed by this package.
|
||||||
|
@ -175,8 +175,7 @@ func NewQUICDialer(config Config) QUICDialer {
|
||||||
d = quicdialer.HandshakeSaver{Saver: config.TLSSaver, Dialer: d}
|
d = quicdialer.HandshakeSaver{Saver: config.TLSSaver, Dialer: d}
|
||||||
}
|
}
|
||||||
d = &netxlite.QUICDialerResolver{Resolver: config.FullResolver, Dialer: d}
|
d = &netxlite.QUICDialerResolver{Resolver: config.FullResolver, Dialer: d}
|
||||||
var dialer QUICDialer = &httptransport.QUICWrapperDialer{Dialer: d}
|
return d
|
||||||
return dialer
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewTLSDialer creates a new TLSDialer from the specified config
|
// NewTLSDialer creates a new TLSDialer from the specified config
|
||||||
|
|
54
internal/netxlite/http3.go
Normal file
54
internal/netxlite/http3.go
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/lucas-clemente/quic-go"
|
||||||
|
"github.com/lucas-clemente/quic-go/http3"
|
||||||
|
)
|
||||||
|
|
||||||
|
// http3Dialer adapts a QUICContextDialer to work with
|
||||||
|
// an http3.RoundTripper. This is necessary because the
|
||||||
|
// http3.RoundTripper does not support DialContext.
|
||||||
|
type http3Dialer struct {
|
||||||
|
Dialer QUICContextDialer
|
||||||
|
}
|
||||||
|
|
||||||
|
// dial is like QUICContextDialer.DialContext but without context.
|
||||||
|
func (d *http3Dialer) dial(network, address string, tlsConfig *tls.Config,
|
||||||
|
quicConfig *quic.Config) (quic.EarlySession, error) {
|
||||||
|
return d.Dialer.DialContext(
|
||||||
|
context.Background(), network, address, tlsConfig, quicConfig)
|
||||||
|
}
|
||||||
|
|
||||||
|
// http3Transport is an HTTPTransport using the http3 protocol.
|
||||||
|
type http3Transport struct {
|
||||||
|
child *http3.RoundTripper
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ HTTPTransport = &http3Transport{}
|
||||||
|
|
||||||
|
// RoundTrip implements HTTPTransport.RoundTrip.
|
||||||
|
func (txp *http3Transport) RoundTrip(req *http.Request) (*http.Response, error) {
|
||||||
|
return txp.child.RoundTrip(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
// CloseIdleConnections implements HTTPTransport.CloseIdleConnections.
|
||||||
|
func (txp *http3Transport) CloseIdleConnections() {
|
||||||
|
txp.child.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewHTTP3Transport creates a new HTTPTransport using http3. The
|
||||||
|
// dialer argument MUST NOT be nil. If the tlsConfig argument is nil,
|
||||||
|
// then the code will use the default TLS configuration.
|
||||||
|
func NewHTTP3Transport(
|
||||||
|
dialer QUICContextDialer, tlsConfig *tls.Config) HTTPTransport {
|
||||||
|
return &http3Transport{
|
||||||
|
child: &http3.RoundTripper{
|
||||||
|
Dial: (&http3Dialer{dialer}).dial,
|
||||||
|
TLSClientConfig: tlsConfig,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
25
internal/netxlite/http3_test.go
Normal file
25
internal/netxlite/http3_test.go
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/tls"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestHTTP3TransportWorks(t *testing.T) {
|
||||||
|
d := &QUICDialerResolver{
|
||||||
|
Dialer: &QUICDialerQUICGo{
|
||||||
|
QUICListener: &QUICListenerStdlib{},
|
||||||
|
},
|
||||||
|
Resolver: &net.Resolver{},
|
||||||
|
}
|
||||||
|
txp := NewHTTP3Transport(d, &tls.Config{})
|
||||||
|
client := &http.Client{Transport: txp}
|
||||||
|
resp, err := client.Get("https://www.google.com/robots.txt")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
resp.Body.Close()
|
||||||
|
txp.CloseIdleConnections()
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user