refactor(quicdialer): separate saving from listening (#405)

With this change, we will soon be able to move the creation of
a QUIC session inside of the netxlite package.

Part of https://github.com/ooni/probe/issues/1505.
This commit is contained in:
Simone Basso 2021-06-25 16:20:08 +02:00 committed by GitHub
parent d031829a4b
commit c00cad1382
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 83 additions and 24 deletions

View File

@ -160,7 +160,16 @@ func NewQUICDialer(config Config) QUICDialer {
if config.FullResolver == nil { if config.FullResolver == nil {
config.FullResolver = NewResolver(config) config.FullResolver = NewResolver(config)
} }
var d quicdialer.ContextDialer = &quicdialer.SystemDialer{Saver: config.ReadWriteSaver} var ql quicdialer.QUICListener = &quicdialer.QUICListenerStdlib{}
if config.ReadWriteSaver != nil {
ql = &quicdialer.QUICListenerSaver{
QUICListener: ql,
Saver: config.ReadWriteSaver,
}
}
var d quicdialer.ContextDialer = &quicdialer.SystemDialer{
QUICListener: ql,
}
d = quicdialer.ErrorWrapperDialer{Dialer: d} d = quicdialer.ErrorWrapperDialer{Dialer: d}
if config.TLSSaver != nil { if config.TLSSaver != nil {
d = quicdialer.HandshakeSaver{Saver: config.TLSSaver, Dialer: d} d = quicdialer.HandshakeSaver{Saver: config.TLSSaver, Dialer: d}

View File

@ -25,7 +25,9 @@ func (r MockableResolver) LookupHost(ctx context.Context, host string) ([]string
func TestDNSDialerSuccess(t *testing.T) { func TestDNSDialerSuccess(t *testing.T) {
tlsConf := &tls.Config{NextProtos: []string{"h3"}} tlsConf := &tls.Config{NextProtos: []string{"h3"}}
dialer := quicdialer.DNSDialer{ dialer := quicdialer.DNSDialer{
Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{}} Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerStdlib{},
}}
sess, err := dialer.DialContext( sess, err := dialer.DialContext(
context.Background(), "udp", "www.google.com:443", context.Background(), "udp", "www.google.com:443",
tlsConf, &quic.Config{}) tlsConf, &quic.Config{})
@ -88,7 +90,9 @@ func TestDNSDialerLookupHostFailure(t *testing.T) {
func TestDNSDialerInvalidPort(t *testing.T) { func TestDNSDialerInvalidPort(t *testing.T) {
tlsConf := &tls.Config{NextProtos: []string{"h3"}} tlsConf := &tls.Config{NextProtos: []string{"h3"}}
dialer := quicdialer.DNSDialer{ dialer := quicdialer.DNSDialer{
Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{}} Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerStdlib{},
}}
sess, err := dialer.DialContext( sess, err := dialer.DialContext(
context.Background(), "udp", "www.google.com:0", context.Background(), "udp", "www.google.com:0",
tlsConf, &quic.Config{}) tlsConf, &quic.Config{})
@ -107,7 +111,9 @@ func TestDNSDialerInvalidPort(t *testing.T) {
func TestDNSDialerInvalidPortSyntax(t *testing.T) { func TestDNSDialerInvalidPortSyntax(t *testing.T) {
tlsConf := &tls.Config{NextProtos: []string{"h3"}} tlsConf := &tls.Config{NextProtos: []string{"h3"}}
dialer := quicdialer.DNSDialer{ dialer := quicdialer.DNSDialer{
Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{}} Resolver: new(net.Resolver), Dialer: quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerStdlib{},
}}
sess, err := dialer.DialContext( sess, err := dialer.DialContext(
context.Background(), "udp", "www.google.com:port", context.Background(), "udp", "www.google.com:port",
tlsConf, &quic.Config{}) tlsConf, &quic.Config{})

View File

@ -48,7 +48,9 @@ func TestErrorWrapperInvalidCertificate(t *testing.T) {
ServerName: servername, ServerName: servername,
} }
dlr := quicdialer.ErrorWrapperDialer{Dialer: &quicdialer.SystemDialer{}} dlr := quicdialer.ErrorWrapperDialer{Dialer: &quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerStdlib{},
}}
// use Google IP // use Google IP
sess, err := dlr.DialContext(context.Background(), "udp", sess, err := dlr.DialContext(context.Background(), "udp",
"216.58.212.164:443", tlsConf, &quic.Config{}) "216.58.212.164:443", tlsConf, &quic.Config{})
@ -69,7 +71,9 @@ func TestErrorWrapperSuccess(t *testing.T) {
NextProtos: []string{"h3"}, NextProtos: []string{"h3"},
ServerName: "www.google.com", ServerName: "www.google.com",
} }
d := quicdialer.ErrorWrapperDialer{Dialer: quicdialer.SystemDialer{}} d := quicdialer.ErrorWrapperDialer{Dialer: quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerStdlib{},
}}
sess, err := d.DialContext(ctx, "udp", "216.58.212.164:443", tlsConf, &quic.Config{}) sess, err := d.DialContext(ctx, "udp", "216.58.212.164:443", tlsConf, &quic.Config{})
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)

View File

@ -36,8 +36,10 @@ func TestHandshakeSaverSuccess(t *testing.T) {
} }
saver := &trace.Saver{} saver := &trace.Saver{}
dlr := quicdialer.HandshakeSaver{ dlr := quicdialer.HandshakeSaver{
Dialer: quicdialer.SystemDialer{}, Dialer: quicdialer.SystemDialer{
Saver: saver, QUICListener: &quicdialer.QUICListenerStdlib{},
},
Saver: saver,
} }
sess, err := dlr.DialContext(context.Background(), "udp", sess, err := dlr.DialContext(context.Background(), "udp",
"216.58.212.164:443", tlsConf, &quic.Config{}) "216.58.212.164:443", tlsConf, &quic.Config{})
@ -92,8 +94,10 @@ func TestHandshakeSaverHostNameError(t *testing.T) {
} }
saver := &trace.Saver{} saver := &trace.Saver{}
dlr := quicdialer.HandshakeSaver{ dlr := quicdialer.HandshakeSaver{
Dialer: quicdialer.SystemDialer{}, Dialer: quicdialer.SystemDialer{
Saver: saver, QUICListener: &quicdialer.QUICListenerStdlib{},
},
Saver: saver,
} }
sess, err := dlr.DialContext(context.Background(), "udp", sess, err := dlr.DialContext(context.Background(), "udp",
"216.58.212.164:443", tlsConf, &quic.Config{}) "216.58.212.164:443", tlsConf, &quic.Config{})

View File

@ -13,13 +13,47 @@ import (
"github.com/ooni/probe-cli/v3/internal/engine/netx/trace" "github.com/ooni/probe-cli/v3/internal/engine/netx/trace"
) )
// QUICListener listens for QUIC connections.
type QUICListener interface {
// Listen creates a new listening net.PacketConn.
Listen(addr *net.UDPAddr) (net.PacketConn, error)
}
// QUICListenerStdlib is a QUICListener using the standard library.
type QUICListenerStdlib struct{}
// Listen implements QUICListener.Listen.
func (qls *QUICListenerStdlib) Listen(addr *net.UDPAddr) (net.PacketConn, error) {
return net.ListenUDP("udp", addr)
}
// QUICListenerSaver is a QUICListener that also implements saving events.
type QUICListenerSaver struct {
// QUICListener is the underlying QUICListener.
QUICListener QUICListener
// Saver is the underlying Saver.
Saver *trace.Saver
}
// Listen implements QUICListener.Listen.
func (qls *QUICListenerSaver) Listen(addr *net.UDPAddr) (net.PacketConn, error) {
pconn, err := qls.QUICListener.Listen(addr)
if err != nil {
return nil, err
}
// TODO(bassosimone): refactor to remove this restriction.
udpConn, ok := pconn.(*net.UDPConn)
if !ok {
return nil, errors.New("quicdialer: cannot convert to udpConn")
}
return saverUDPConn{UDPConn: udpConn, saver: qls.Saver}, nil
}
// SystemDialer is the basic dialer for QUIC // SystemDialer is the basic dialer for QUIC
type SystemDialer struct { type SystemDialer struct {
// Saver saves read/write events on the underlying UDP // QUICListener is the underlying QUICListener to use.
// connection. (Implementation note: we need it here since QUICListener QUICListener
// this is the only part in the codebase that is able to
// observe the underlying UDP connection.)
Saver *trace.Saver
} }
// DialContext implements ContextDialer.DialContext // DialContext implements ContextDialer.DialContext
@ -35,20 +69,14 @@ func (d SystemDialer) DialContext(ctx context.Context, network string,
} }
ip := net.ParseIP(onlyhost) ip := net.ParseIP(onlyhost)
if ip == nil { if ip == nil {
// TODO(kelmenhorst): write test for this error condition.
return nil, errors.New("quicdialer: invalid IP representation") return nil, errors.New("quicdialer: invalid IP representation")
} }
udpConn, err := net.ListenUDP("udp", &net.UDPAddr{IP: net.IPv4zero, Port: 0}) pconn, err := d.QUICListener.Listen(&net.UDPAddr{IP: net.IPv4zero, Port: 0})
if err != nil { if err != nil {
return nil, err return nil, err
} }
var pconn net.PacketConn = udpConn
if d.Saver != nil {
pconn = saverUDPConn{UDPConn: udpConn, saver: d.Saver}
}
udpAddr := &net.UDPAddr{IP: ip, Port: port, Zone: ""} udpAddr := &net.UDPAddr{IP: ip, Port: port, Zone: ""}
return quic.DialEarlyContext(ctx, pconn, udpAddr, host, tlsCfg, cfg) return quic.DialEarlyContext(ctx, pconn, udpAddr, host, tlsCfg, cfg)
} }
type saverUDPConn struct { type saverUDPConn struct {

View File

@ -18,7 +18,10 @@ func TestSystemDialerInvalidIPFailure(t *testing.T) {
} }
saver := &trace.Saver{} saver := &trace.Saver{}
systemdialer := quicdialer.SystemDialer{ systemdialer := quicdialer.SystemDialer{
Saver: saver, QUICListener: &quicdialer.QUICListenerSaver{
QUICListener: &quicdialer.QUICListenerStdlib{},
Saver: saver,
},
} }
sess, err := systemdialer.DialContext(context.Background(), "udp", "a.b.c.d:0", tlsConf, &quic.Config{}) sess, err := systemdialer.DialContext(context.Background(), "udp", "a.b.c.d:0", tlsConf, &quic.Config{})
if err == nil { if err == nil {
@ -39,7 +42,12 @@ func TestSystemDialerSuccessWithReadWrite(t *testing.T) {
ServerName: "www.google.com", ServerName: "www.google.com",
} }
saver := &trace.Saver{} saver := &trace.Saver{}
systemdialer := quicdialer.SystemDialer{Saver: saver} systemdialer := quicdialer.SystemDialer{
QUICListener: &quicdialer.QUICListenerSaver{
QUICListener: &quicdialer.QUICListenerStdlib{},
Saver: saver,
},
}
_, err := systemdialer.DialContext(context.Background(), "udp", _, err := systemdialer.DialContext(context.Background(), "udp",
"216.58.212.164:443", tlsConf, &quic.Config{}) "216.58.212.164:443", tlsConf, &quic.Config{})
if err != nil { if err != nil {