feat: reintroduce the tproxy functionality (#975)
We originally removed the TProxy in https://github.com/ooni/probe/issues/2224. Nevertheless, in https://github.com/ooni/probe-cli/pull/969, we determined that something like the previous TProxy, with small changes, was required to support https://github.com/ooni/probe/issues/2340. So, this pull request reintroduces a slightly-modified TProxy functionality that better adapts to the `--remote=REMOTE` use case.
This commit is contained in:
parent
46233802ab
commit
86ffd6a0c4
38
internal/model/mocks/underlyingnetwork.go
Normal file
38
internal/model/mocks/underlyingnetwork.go
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
// UnderlyingNetwork allows mocking model.UnderlyingNetwork.
|
||||||
|
type UnderlyingNetwork struct {
|
||||||
|
MockDialContext func(ctx context.Context, timeout time.Duration, network, address string) (net.Conn, error)
|
||||||
|
|
||||||
|
MockListenUDP func(network string, addr *net.UDPAddr) (model.UDPLikeConn, error)
|
||||||
|
|
||||||
|
MockGetaddrinfoLookupANY func(ctx context.Context, domain string) ([]string, string, error)
|
||||||
|
|
||||||
|
MockGetaddrinfoResolverNetwork func() string
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ model.UnderlyingNetwork = &UnderlyingNetwork{}
|
||||||
|
|
||||||
|
func (un *UnderlyingNetwork) DialContext(ctx context.Context, timeout time.Duration, network, address string) (net.Conn, error) {
|
||||||
|
return un.MockDialContext(ctx, timeout, network, address)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (un *UnderlyingNetwork) ListenUDP(network string, addr *net.UDPAddr) (model.UDPLikeConn, error) {
|
||||||
|
return un.MockListenUDP(network, addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (un *UnderlyingNetwork) GetaddrinfoLookupANY(ctx context.Context, domain string) ([]string, string, error) {
|
||||||
|
return un.MockGetaddrinfoLookupANY(ctx, domain)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (un *UnderlyingNetwork) GetaddrinfoResolverNetwork() string {
|
||||||
|
return un.MockGetaddrinfoResolverNetwork()
|
||||||
|
}
|
79
internal/model/mocks/underlyingnetwork_test.go
Normal file
79
internal/model/mocks/underlyingnetwork_test.go
Normal file
|
@ -0,0 +1,79 @@
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"net"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestUnderlyingNetwork(t *testing.T) {
|
||||||
|
t.Run("DialContext", func(t *testing.T) {
|
||||||
|
expect := errors.New("mocked error")
|
||||||
|
un := &UnderlyingNetwork{
|
||||||
|
MockDialContext: func(ctx context.Context, timeout time.Duration, network, address string) (net.Conn, error) {
|
||||||
|
return nil, expect
|
||||||
|
},
|
||||||
|
}
|
||||||
|
ctx := context.Background()
|
||||||
|
conn, err := un.DialContext(ctx, time.Second, "tcp", "1.1.1.1:443")
|
||||||
|
if !errors.Is(err, expect) {
|
||||||
|
t.Fatal("unexpected err", err)
|
||||||
|
}
|
||||||
|
if conn != nil {
|
||||||
|
t.Fatal("expected nil conn")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("ListenUDP", func(t *testing.T) {
|
||||||
|
expect := errors.New("mocked error")
|
||||||
|
un := &UnderlyingNetwork{
|
||||||
|
MockListenUDP: func(network string, addr *net.UDPAddr) (model.UDPLikeConn, error) {
|
||||||
|
return nil, expect
|
||||||
|
},
|
||||||
|
}
|
||||||
|
pconn, err := un.ListenUDP("udp", &net.UDPAddr{})
|
||||||
|
if !errors.Is(err, expect) {
|
||||||
|
t.Fatal("unexpected err", err)
|
||||||
|
}
|
||||||
|
if pconn != nil {
|
||||||
|
t.Fatal("expected nil conn")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetaddrinfoLookupANY", func(t *testing.T) {
|
||||||
|
expect := errors.New("mocked error")
|
||||||
|
un := &UnderlyingNetwork{
|
||||||
|
MockGetaddrinfoLookupANY: func(ctx context.Context, domain string) ([]string, string, error) {
|
||||||
|
return nil, "", expect
|
||||||
|
},
|
||||||
|
}
|
||||||
|
ctx := context.Background()
|
||||||
|
addrs, cname, err := un.GetaddrinfoLookupANY(ctx, "dns.google")
|
||||||
|
if !errors.Is(err, expect) {
|
||||||
|
t.Fatal("unexpected err", err)
|
||||||
|
}
|
||||||
|
if len(addrs) != 0 {
|
||||||
|
t.Fatal("expected zero length addrs")
|
||||||
|
}
|
||||||
|
if cname != "" {
|
||||||
|
t.Fatal("expected empty name")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("GetaddrinfoResolverNetwork", func(t *testing.T) {
|
||||||
|
expect := "antani"
|
||||||
|
un := &UnderlyingNetwork{
|
||||||
|
MockGetaddrinfoResolverNetwork: func() string {
|
||||||
|
return expect
|
||||||
|
},
|
||||||
|
}
|
||||||
|
got := un.GetaddrinfoResolverNetwork()
|
||||||
|
if got != expect {
|
||||||
|
t.Fatal("unexpected resolver network")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
|
@ -480,3 +480,21 @@ type UDPLikeConn interface {
|
||||||
// which is also instrumental to setting the read buffer.
|
// which is also instrumental to setting the read buffer.
|
||||||
SyscallConn() (syscall.RawConn, error)
|
SyscallConn() (syscall.RawConn, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UnderlyingNetwork implements the underlying network APIs on
|
||||||
|
// top of which we implement network extensions.
|
||||||
|
type UnderlyingNetwork interface {
|
||||||
|
// DialContext is equivalent to net.Dialer.DialContext except that
|
||||||
|
// there is also an explicit timeout for dialing.
|
||||||
|
DialContext(ctx context.Context, timeout time.Duration, network, address string) (net.Conn, error)
|
||||||
|
|
||||||
|
// ListenUDP is equivalent to net.ListenUDP.
|
||||||
|
ListenUDP(network string, addr *net.UDPAddr) (UDPLikeConn, error)
|
||||||
|
|
||||||
|
// GetaddrinfoLookupANY is like net.Resolver.LookupHost except that it
|
||||||
|
// also returns to the caller the CNAME when it is available.
|
||||||
|
GetaddrinfoLookupANY(ctx context.Context, domain string) ([]string, string, error)
|
||||||
|
|
||||||
|
// GetaddrinfoResolverNetwork returns the resolver network.
|
||||||
|
GetaddrinfoResolverNetwork() string
|
||||||
|
}
|
||||||
|
|
|
@ -34,7 +34,7 @@ func NewDialerWithResolver(dl model.DebugLogger, r model.Resolver, w ...model.Di
|
||||||
// When possible use NewDialerWithResolver or NewDialerWithoutResolver
|
// When possible use NewDialerWithResolver or NewDialerWithoutResolver
|
||||||
// instead of using this rather low-level function.
|
// instead of using this rather low-level function.
|
||||||
//
|
//
|
||||||
// Arguments
|
// # Arguments
|
||||||
//
|
//
|
||||||
// 1. logger is used to emit debug messages (MUST NOT be nil);
|
// 1. logger is used to emit debug messages (MUST NOT be nil);
|
||||||
//
|
//
|
||||||
|
@ -47,7 +47,7 @@ func NewDialerWithResolver(dl model.DebugLogger, r model.Resolver, w ...model.Di
|
||||||
// modify the behavior of the returned dialer (see below). Please note
|
// modify the behavior of the returned dialer (see below). Please note
|
||||||
// that this function will just ignore any nil wrapper.
|
// that this function will just ignore any nil wrapper.
|
||||||
//
|
//
|
||||||
// Return value
|
// # Return value
|
||||||
//
|
//
|
||||||
// The returned dialer is an opaque type consisting of the composition of
|
// The returned dialer is an opaque type consisting of the composition of
|
||||||
// several simple dialers. The following pseudo code illustrates the general
|
// several simple dialers. The following pseudo code illustrates the general
|
||||||
|
@ -68,7 +68,6 @@ func NewDialerWithResolver(dl model.DebugLogger, r model.Resolver, w ...model.Di
|
||||||
// }
|
// }
|
||||||
// return nil, errors[0]
|
// return nil, errors[0]
|
||||||
//
|
//
|
||||||
//
|
|
||||||
// The following table describes the structure of the returned dialer:
|
// The following table describes the structure of the returned dialer:
|
||||||
//
|
//
|
||||||
// +-------+-----------------+------------------------------------------+
|
// +-------+-----------------+------------------------------------------+
|
||||||
|
@ -98,7 +97,7 @@ func NewDialerWithResolver(dl model.DebugLogger, r model.Resolver, w ...model.Di
|
||||||
// The list of wrappers allows to insert modified dialers in the correct
|
// The list of wrappers allows to insert modified dialers in the correct
|
||||||
// place for observing and saving I/O events (connect, read, etc.).
|
// place for observing and saving I/O events (connect, read, etc.).
|
||||||
//
|
//
|
||||||
// Remarks
|
// # Remarks
|
||||||
//
|
//
|
||||||
// When the resolver is &NullResolver{} any attempt to perform DNS resolutions
|
// When the resolver is &NullResolver{} any attempt to perform DNS resolutions
|
||||||
// in the dialer at index N+2 will fail with ErrNoResolver.
|
// in the dialer at index N+2 will fail with ErrNoResolver.
|
||||||
|
@ -155,16 +154,16 @@ var _ model.Dialer = &DialerSystem{}
|
||||||
|
|
||||||
const dialerDefaultTimeout = 15 * time.Second
|
const dialerDefaultTimeout = 15 * time.Second
|
||||||
|
|
||||||
func (d *DialerSystem) newUnderlyingDialer() model.SimpleDialer {
|
func (d *DialerSystem) configuredTimeout() time.Duration {
|
||||||
t := d.timeout
|
t := d.timeout
|
||||||
if t <= 0 {
|
if t <= 0 {
|
||||||
t = dialerDefaultTimeout
|
t = dialerDefaultTimeout
|
||||||
}
|
}
|
||||||
return &net.Dialer{Timeout: t}
|
return t
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DialerSystem) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
func (d *DialerSystem) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
|
||||||
return d.newUnderlyingDialer().DialContext(ctx, network, address)
|
return TProxy.DialContext(ctx, d.configuredTimeout(), network, address)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DialerSystem) CloseIdleConnections() {
|
func (d *DialerSystem) CloseIdleConnections() {
|
||||||
|
|
|
@ -83,8 +83,8 @@ func TestNewDialer(t *testing.T) {
|
||||||
func TestDialerSystem(t *testing.T) {
|
func TestDialerSystem(t *testing.T) {
|
||||||
t.Run("has a default timeout", func(t *testing.T) {
|
t.Run("has a default timeout", func(t *testing.T) {
|
||||||
d := &DialerSystem{}
|
d := &DialerSystem{}
|
||||||
ud := d.newUnderlyingDialer()
|
timeout := d.configuredTimeout()
|
||||||
if ud.(*net.Dialer).Timeout != dialerDefaultTimeout {
|
if timeout != dialerDefaultTimeout {
|
||||||
t.Fatal("unexpected default timeout")
|
t.Fatal("unexpected default timeout")
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -92,8 +92,8 @@ func TestDialerSystem(t *testing.T) {
|
||||||
t.Run("we can change the timeout for testing", func(t *testing.T) {
|
t.Run("we can change the timeout for testing", func(t *testing.T) {
|
||||||
const smaller = 1 * time.Second
|
const smaller = 1 * time.Second
|
||||||
d := &DialerSystem{timeout: smaller}
|
d := &DialerSystem{timeout: smaller}
|
||||||
ud := d.newUnderlyingDialer()
|
timeout := d.configuredTimeout()
|
||||||
if ud.(*net.Dialer).Timeout != smaller {
|
if timeout != smaller {
|
||||||
t.Fatal("unexpected timeout")
|
t.Fatal("unexpected timeout")
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
|
@ -104,7 +104,7 @@ func (txp *dnsOverGetaddrinfoTransport) lookupfn() func(ctx context.Context, dom
|
||||||
if txp.testableLookupANY != nil {
|
if txp.testableLookupANY != nil {
|
||||||
return txp.testableLookupANY
|
return txp.testableLookupANY
|
||||||
}
|
}
|
||||||
return getaddrinfoLookupANY
|
return TProxy.GetaddrinfoLookupANY
|
||||||
}
|
}
|
||||||
|
|
||||||
func (txp *dnsOverGetaddrinfoTransport) RequiresPadding() bool {
|
func (txp *dnsOverGetaddrinfoTransport) RequiresPadding() bool {
|
||||||
|
@ -112,7 +112,7 @@ func (txp *dnsOverGetaddrinfoTransport) RequiresPadding() bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (txp *dnsOverGetaddrinfoTransport) Network() string {
|
func (txp *dnsOverGetaddrinfoTransport) Network() string {
|
||||||
return getaddrinfoResolverNetwork()
|
return TProxy.GetaddrinfoResolverNetwork()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (txp *dnsOverGetaddrinfoTransport) Address() string {
|
func (txp *dnsOverGetaddrinfoTransport) Address() string {
|
||||||
|
|
|
@ -8,13 +8,13 @@
|
||||||
// You should consider checking the tutorial explaining how to use this package
|
// You should consider checking the tutorial explaining how to use this package
|
||||||
// for network measurements: https://github.com/ooni/probe-cli/tree/master/internal/tutorial/netxlite.
|
// for network measurements: https://github.com/ooni/probe-cli/tree/master/internal/tutorial/netxlite.
|
||||||
//
|
//
|
||||||
// Naming and history
|
// # Naming and history
|
||||||
//
|
//
|
||||||
// Previous versions of this package were called netx. Compared to such
|
// Previous versions of this package were called netx. Compared to such
|
||||||
// versions this package is lightweight because it does not contain code
|
// versions this package is lightweight because it does not contain code
|
||||||
// to perform the measurements, hence its name.
|
// to perform the measurements, hence its name.
|
||||||
//
|
//
|
||||||
// Design
|
// # Design
|
||||||
//
|
//
|
||||||
// We want to potentially be able to observe each low-level operation
|
// We want to potentially be able to observe each low-level operation
|
||||||
// separately, even though this is not done by this package. This is
|
// separately, even though this is not done by this package. This is
|
||||||
|
@ -41,7 +41,19 @@
|
||||||
// See also the design document at docs/design/dd-003-step-by-step.md,
|
// See also the design document at docs/design/dd-003-step-by-step.md,
|
||||||
// which provides an overview of netxlite's main concerns.
|
// which provides an overview of netxlite's main concerns.
|
||||||
//
|
//
|
||||||
// Operations
|
// To implement integration testing, we support hijacking the core network
|
||||||
|
// primitives used by this package, that is:
|
||||||
|
//
|
||||||
|
// 1. connecting a new TCP/UDP connection;
|
||||||
|
//
|
||||||
|
// 2. creating listening UDP sockets;
|
||||||
|
//
|
||||||
|
// 3. resolving domain names with getaddrinfo.
|
||||||
|
//
|
||||||
|
// By overriding the TProxy variable, you can control these operations and route
|
||||||
|
// traffic to, e.g., a wireguard peer where you implement censorship.
|
||||||
|
//
|
||||||
|
// # Operations
|
||||||
//
|
//
|
||||||
// This package implements the following operations:
|
// This package implements the following operations:
|
||||||
//
|
//
|
||||||
|
@ -62,7 +74,7 @@
|
||||||
// Operations 1, 2, 3, and 4 are used when we perform measurements,
|
// Operations 1, 2, 3, and 4 are used when we perform measurements,
|
||||||
// while 5 and 6 are mostly used when speaking with our backend.
|
// while 5 and 6 are mostly used when speaking with our backend.
|
||||||
//
|
//
|
||||||
// Getaddrinfo usage
|
// # Getaddrinfo usage
|
||||||
//
|
//
|
||||||
// When compiled with CGO_ENABLED=1, this package will link with libc
|
// When compiled with CGO_ENABLED=1, this package will link with libc
|
||||||
// and call getaddrinfo directly. While this design choice means we will
|
// and call getaddrinfo directly. While this design choice means we will
|
||||||
|
|
|
@ -29,7 +29,7 @@ var _ model.QUICListener = &quicListenerStdlib{}
|
||||||
|
|
||||||
// Listen implements QUICListener.Listen.
|
// Listen implements QUICListener.Listen.
|
||||||
func (qls *quicListenerStdlib) Listen(addr *net.UDPAddr) (model.UDPLikeConn, error) {
|
func (qls *quicListenerStdlib) Listen(addr *net.UDPAddr) (model.UDPLikeConn, error) {
|
||||||
return net.ListenUDP("udp", addr)
|
return TProxy.ListenUDP("udp", addr)
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewQUICDialerWithResolver is the WrapDialer equivalent for QUIC where
|
// NewQUICDialerWithResolver is the WrapDialer equivalent for QUIC where
|
||||||
|
|
39
internal/netxlite/tproxy.go
Normal file
39
internal/netxlite/tproxy.go
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TProxy refers to the UnderlyingNetwork implementation. By overriding this
|
||||||
|
// variable you can force netxlite to use alternative network primitives.
|
||||||
|
var TProxy model.UnderlyingNetwork = &DefaultTProxy{}
|
||||||
|
|
||||||
|
// defaultTProxy is the default UnderlyingNetwork implementation.
|
||||||
|
type DefaultTProxy struct{}
|
||||||
|
|
||||||
|
// DialContext implements UnderlyingNetwork.
|
||||||
|
func (tp *DefaultTProxy) DialContext(ctx context.Context, timeout time.Duration, network, address string) (net.Conn, error) {
|
||||||
|
d := &net.Dialer{
|
||||||
|
Timeout: timeout,
|
||||||
|
}
|
||||||
|
return d.DialContext(ctx, network, address)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ListenUDP implements UnderlyingNetwork.
|
||||||
|
func (tp *DefaultTProxy) ListenUDP(network string, addr *net.UDPAddr) (model.UDPLikeConn, error) {
|
||||||
|
return net.ListenUDP(network, addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetaddrinfoLookupANY implements UnderlyingNetwork.
|
||||||
|
func (tp *DefaultTProxy) GetaddrinfoLookupANY(ctx context.Context, domain string) ([]string, string, error) {
|
||||||
|
return getaddrinfoLookupANY(ctx, domain)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetaddrinfoResolverNetwork implements UnderlyingNetwork.
|
||||||
|
func (tp *DefaultTProxy) GetaddrinfoResolverNetwork() string {
|
||||||
|
return getaddrinfoResolverNetwork()
|
||||||
|
}
|
22
internal/netxlite/tproxy_test.go
Normal file
22
internal/netxlite/tproxy_test.go
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
package netxlite
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestDefaultTProxy(t *testing.T) {
|
||||||
|
t.Run("DialContext honours the timeout", func(t *testing.T) {
|
||||||
|
tp := &DefaultTProxy{}
|
||||||
|
ctx := context.Background()
|
||||||
|
conn, err := tp.DialContext(ctx, 100*time.Microsecond, "tcp", "1.1.1.1:443")
|
||||||
|
if err == nil || !strings.HasSuffix(err.Error(), "i/o timeout") {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if conn != nil {
|
||||||
|
t.Fatal("expected nil conn")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user