feat(torsf): collect tor logs, select rendezvous method, count bytes (#683)
This diff contains significant improvements over the previous implementation of the torsf experiment. We add support for configuring different rendezvous methods after the convo at https://github.com/ooni/probe/issues/2004. In doing that, I've tried to use a terminology that is consistent with the names being actually used by tor developers. In terms of what to do next, this diff basically instruments torsf to always rendezvous using domain fronting. Yet, it's also possible to change the rendezvous method from the command line, when using miniooni, which allows to experiment a bit more. In the same vein, by default we use a persistent tor datadir, but it's also possible to use a temporary datadir using the cmdline. Here's how a generic invocation of `torsf` looks like: ```bash ./miniooni -O DisablePersistentDatadir=true \ -O RendezvousMethod=amp \ -O DisableProgress=true \ torsf ``` (The default is `DisablePersistentDatadir=false` and `RendezvousMethod=domain_fronting`.) With this implementation, we can start measuring whether snowflake and tor together can boostrap, which seems the most important thing to focus on at the beginning. Understanding why the bootstrap most often does not converge with a temporary datadir on Android devices remains instead an open problem for now. (I'll also update the relevant issues or create new issues after commit this.) We also address some methodology improvements that were proposed in https://github.com/ooni/probe/issues/1686. Namely: 1. we record the tor version; 2. we include the bootstrap percentage by reading the logs; 3. we set the anomaly key correctly; 4. we measure the bytes send and received (by `tor` not by `snowflake`, since doing it for snowflake seems more complex at this stage). What remains to be done is the possibility of including Snowflake events into the measurement, which is not possible until the new improvements at common/event in snowflake.git are included into a tagged version of snowflake itself. (I'll make sure to mention this aspect to @cohosh in https://github.com/ooni/probe/issues/2004.)
This commit is contained in:
parent
4e5f9bd254
commit
85664f1e31
|
@ -24,3 +24,16 @@ func (c *Conn) Write(p []byte) (int, error) {
|
||||||
c.Counter.CountBytesSent(count)
|
c.Counter.CountBytesSent(count)
|
||||||
return count, err
|
return count, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Wrap returns a new conn that uses the given counter.
|
||||||
|
func Wrap(conn net.Conn, counter *Counter) net.Conn {
|
||||||
|
return &Conn{Conn: conn, Counter: counter}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaybeWrap is like wrap if counter is not nil, otherwise it's a no-op.
|
||||||
|
func MaybeWrap(conn net.Conn, counter *Counter) net.Conn {
|
||||||
|
if counter == nil {
|
||||||
|
return conn
|
||||||
|
}
|
||||||
|
return Wrap(conn, counter)
|
||||||
|
}
|
||||||
|
|
|
@ -64,3 +64,34 @@ func TestConnWorksOnFailure(t *testing.T) {
|
||||||
t.Fatal("unexpected number of bytes sent")
|
t.Fatal("unexpected number of bytes sent")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestWrap(t *testing.T) {
|
||||||
|
conn := &mocks.Conn{}
|
||||||
|
counter := New()
|
||||||
|
nconn := Wrap(conn, counter)
|
||||||
|
_, good := nconn.(*Conn)
|
||||||
|
if !good {
|
||||||
|
t.Fatal("did not wrap")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMaybeWrap(t *testing.T) {
|
||||||
|
t.Run("with nil counter", func(t *testing.T) {
|
||||||
|
conn := &mocks.Conn{}
|
||||||
|
nconn := MaybeWrap(conn, nil)
|
||||||
|
_, good := nconn.(*mocks.Conn)
|
||||||
|
if !good {
|
||||||
|
t.Fatal("did not wrap")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("with legit counter", func(t *testing.T) {
|
||||||
|
conn := &mocks.Conn{}
|
||||||
|
counter := New()
|
||||||
|
nconn := MaybeWrap(conn, counter)
|
||||||
|
_, good := nconn.(*Conn)
|
||||||
|
if !good {
|
||||||
|
t.Fatal("did not wrap")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
40
internal/bytecounter/context.go
Normal file
40
internal/bytecounter/context.go
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
package bytecounter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
)
|
||||||
|
|
||||||
|
type byteCounterSessionKey struct{}
|
||||||
|
|
||||||
|
// ContextSessionByteCounter retrieves the session byte counter from the context
|
||||||
|
func ContextSessionByteCounter(ctx context.Context) *Counter {
|
||||||
|
counter, _ := ctx.Value(byteCounterSessionKey{}).(*Counter)
|
||||||
|
return counter
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithSessionByteCounter assigns the session byte counter to the context.
|
||||||
|
func WithSessionByteCounter(ctx context.Context, counter *Counter) context.Context {
|
||||||
|
return context.WithValue(ctx, byteCounterSessionKey{}, counter)
|
||||||
|
}
|
||||||
|
|
||||||
|
type byteCounterExperimentKey struct{}
|
||||||
|
|
||||||
|
// ContextExperimentByteCounter retrieves the experiment byte counter from the context
|
||||||
|
func ContextExperimentByteCounter(ctx context.Context) *Counter {
|
||||||
|
counter, _ := ctx.Value(byteCounterExperimentKey{}).(*Counter)
|
||||||
|
return counter
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithExperimentByteCounter assigns the experiment byte counter to the context.
|
||||||
|
func WithExperimentByteCounter(ctx context.Context, counter *Counter) context.Context {
|
||||||
|
return context.WithValue(ctx, byteCounterExperimentKey{}, counter)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaybeWrapWithContextByteCounters wraps a conn with the byte counters
|
||||||
|
// that have previosuly been configured into a context.
|
||||||
|
func MaybeWrapWithContextByteCounters(ctx context.Context, conn net.Conn) net.Conn {
|
||||||
|
conn = MaybeWrap(conn, ContextExperimentByteCounter(ctx))
|
||||||
|
conn = MaybeWrap(conn, ContextSessionByteCounter(ctx))
|
||||||
|
return conn
|
||||||
|
}
|
61
internal/bytecounter/context_test.go
Normal file
61
internal/bytecounter/context_test.go
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
package bytecounter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"net"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/model/mocks"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSessionByteCounter(t *testing.T) {
|
||||||
|
counter := New()
|
||||||
|
ctx := context.Background()
|
||||||
|
ctx = WithSessionByteCounter(ctx, counter)
|
||||||
|
outer := ContextSessionByteCounter(ctx)
|
||||||
|
if outer != counter {
|
||||||
|
t.Fatal("unexpected result")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestExperimentByteCounter(t *testing.T) {
|
||||||
|
counter := New()
|
||||||
|
ctx := context.Background()
|
||||||
|
ctx = WithExperimentByteCounter(ctx, counter)
|
||||||
|
outer := ContextExperimentByteCounter(ctx)
|
||||||
|
if outer != counter {
|
||||||
|
t.Fatal("unexpected result")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMaybeWrapWithContextByteCounters(t *testing.T) {
|
||||||
|
var conn net.Conn = &mocks.Conn{
|
||||||
|
MockRead: func(b []byte) (int, error) {
|
||||||
|
return len(b), nil
|
||||||
|
},
|
||||||
|
MockWrite: func(b []byte) (int, error) {
|
||||||
|
return len(b), nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
sessCounter := New()
|
||||||
|
expCounter := New()
|
||||||
|
ctx := context.Background()
|
||||||
|
ctx = WithSessionByteCounter(ctx, sessCounter)
|
||||||
|
ctx = WithExperimentByteCounter(ctx, expCounter)
|
||||||
|
conn = MaybeWrapWithContextByteCounters(ctx, conn)
|
||||||
|
buf := make([]byte, 128)
|
||||||
|
conn.Read(buf)
|
||||||
|
conn.Write(buf)
|
||||||
|
if sessCounter.Received.Load() != 128 {
|
||||||
|
t.Fatal("invalid value")
|
||||||
|
}
|
||||||
|
if sessCounter.Sent.Load() != 128 {
|
||||||
|
t.Fatal("invalid value")
|
||||||
|
}
|
||||||
|
if expCounter.Received.Load() != 128 {
|
||||||
|
t.Fatal("invalid value")
|
||||||
|
}
|
||||||
|
if expCounter.Sent.Load() != 128 {
|
||||||
|
t.Fatal("invalid value")
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,43 +0,0 @@
|
||||||
// Command ptxclient implements a pluggable transports CLI client. This command
|
|
||||||
// is only meant for testing and is not production ready.
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"os/signal"
|
|
||||||
|
|
||||||
"github.com/apex/log"
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/ptx"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
mode := flag.String("m", "snowflake", "one of snowflake and obfs4")
|
|
||||||
verbose := flag.Bool("v", false, "enable verbose mode")
|
|
||||||
flag.Parse()
|
|
||||||
if *verbose {
|
|
||||||
log.SetLevel(log.DebugLevel)
|
|
||||||
}
|
|
||||||
var dialer ptx.PTDialer
|
|
||||||
switch *mode {
|
|
||||||
case "snowflake":
|
|
||||||
dialer = &ptx.SnowflakeDialer{}
|
|
||||||
case "obfs4":
|
|
||||||
dialer = ptx.DefaultTestingOBFS4Bridge()
|
|
||||||
default:
|
|
||||||
fmt.Fprintf(os.Stderr, "unknown pluggable transport: %s\n", *mode)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
listener := &ptx.Listener{
|
|
||||||
PTDialer: dialer,
|
|
||||||
Logger: log.Log,
|
|
||||||
}
|
|
||||||
if err := listener.Start(); err != nil {
|
|
||||||
log.WithError(err).Fatal("listener.Start failed")
|
|
||||||
}
|
|
||||||
sigch := make(chan os.Signal, 1)
|
|
||||||
signal.Notify(sigch, os.Interrupt)
|
|
||||||
<-sigch
|
|
||||||
listener.Stop()
|
|
||||||
}
|
|
|
@ -11,7 +11,6 @@ import (
|
||||||
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/bytecounter"
|
"github.com/ooni/probe-cli/v3/internal/bytecounter"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/geolocate"
|
"github.com/ooni/probe-cli/v3/internal/engine/geolocate"
|
||||||
"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"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/probeservices"
|
"github.com/ooni/probe-cli/v3/internal/engine/probeservices"
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
|
@ -161,8 +160,8 @@ func (e *Experiment) MeasureAsync(
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
ctx = dialer.WithSessionByteCounter(ctx, e.session.byteCounter)
|
ctx = bytecounter.WithSessionByteCounter(ctx, e.session.byteCounter)
|
||||||
ctx = dialer.WithExperimentByteCounter(ctx, e.byteCounter)
|
ctx = bytecounter.WithExperimentByteCounter(ctx, e.byteCounter)
|
||||||
var async model.ExperimentMeasurerAsync
|
var async model.ExperimentMeasurerAsync
|
||||||
if v, okay := e.measurer.(model.ExperimentMeasurerAsync); okay {
|
if v, okay := e.measurer.(model.ExperimentMeasurerAsync); okay {
|
||||||
async = v
|
async = v
|
||||||
|
|
1
internal/engine/experiment/torsf/.gitignore
vendored
Normal file
1
internal/engine/experiment/torsf/.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
/torsf
|
31
internal/engine/experiment/torsf/testdata/tor.log
vendored
Normal file
31
internal/engine/experiment/torsf/testdata/tor.log
vendored
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
Feb 04 15:04:29.000 [notice] Tor 0.4.6.9 opening new log file.
|
||||||
|
Feb 04 15:04:29.360 [notice] We compiled with OpenSSL 101010cf: OpenSSL 1.1.1l FIPS 24 Aug 2021 and we are running with OpenSSL 101010cf: 1.1.1l. These two versions should be binary compatible.
|
||||||
|
Feb 04 15:04:29.363 [notice] Tor 0.4.6.9 running on Linux with Libevent 2.1.12-stable, OpenSSL 1.1.1l, Zlib 1.2.11, Liblzma 5.2.5, Libzstd 1.5.2 and Glibc 2.34 as libc.
|
||||||
|
Feb 04 15:04:29.363 [notice] Tor can't help you if you use it wrong! Learn how to be safe at https://www.torproject.org/download/download#warning
|
||||||
|
Feb 04 15:04:29.363 [warn] Tor was compiled with zstd 1.5.1, but is running with zstd 1.5.2. For safety, we'll avoid using advanced zstd functionality.
|
||||||
|
Feb 04 15:04:29.363 [notice] Read configuration file "/home/sbs/.miniooni/tunnel/torsf/tor/torrc-2981077975".
|
||||||
|
Feb 04 15:04:29.366 [notice] Opening Control listener on 127.0.0.1:0
|
||||||
|
Feb 04 15:04:29.367 [notice] Control listener listening on port 41423.
|
||||||
|
Feb 04 15:04:29.367 [notice] Opened Control listener connection (ready) on 127.0.0.1:41423
|
||||||
|
Feb 04 15:04:29.367 [notice] DisableNetwork is set. Tor will not make or accept non-control network connections. Shutting down all existing connections.
|
||||||
|
Feb 04 15:04:29.000 [notice] Parsing GEOIP IPv4 file /usr/share/tor/geoip.
|
||||||
|
Feb 04 15:04:29.000 [notice] Parsing GEOIP IPv6 file /usr/share/tor/geoip6.
|
||||||
|
Feb 04 15:04:29.000 [notice] Bootstrapped 0% (starting): Starting
|
||||||
|
Feb 04 15:04:29.000 [notice] Starting with guard context "bridges"
|
||||||
|
Feb 04 15:04:29.000 [notice] new bridge descriptor 'flakey4' (cached): $2B280B23E1107BB62ABFC40DDCC8824814F80A72~flakey4 [1zOHpg+FxqQfi/6jDLtCpHHqBTH8gjYmCKXkus1D5Ko] at 192.0.2.3
|
||||||
|
Feb 04 15:04:29.000 [notice] Delaying directory fetches: DisableNetwork is set.
|
||||||
|
Feb 04 15:04:29.000 [notice] New control connection opened from 127.0.0.1.
|
||||||
|
Feb 04 15:04:29.000 [notice] Opening Socks listener on 127.0.0.1:0
|
||||||
|
Feb 04 15:04:29.000 [notice] Socks listener listening on port 42089.
|
||||||
|
Feb 04 15:04:29.000 [notice] Opened Socks listener connection (ready) on 127.0.0.1:42089
|
||||||
|
Feb 04 15:04:29.000 [notice] Tor 0.4.6.9 opening log file.
|
||||||
|
Feb 04 15:04:29.000 [notice] Bootstrapped 1% (conn_pt): Connecting to pluggable transport
|
||||||
|
Feb 04 15:04:30.000 [notice] Bootstrapped 2% (conn_done_pt): Connected to pluggable transport
|
||||||
|
Feb 04 15:04:30.000 [notice] Bootstrapped 10% (conn_done): Connected to a relay
|
||||||
|
Feb 04 15:06:20.000 [notice] Bootstrapped 14% (handshake): Handshaking with a relay
|
||||||
|
Feb 04 15:06:24.000 [notice] Bootstrapped 15% (handshake_done): Handshake with a relay done
|
||||||
|
Feb 04 15:06:24.000 [notice] Bootstrapped 75% (enough_dirinfo): Loaded enough directory info to build circuits
|
||||||
|
Feb 04 15:06:24.000 [notice] Bootstrapped 95% (circuit_create): Establishing a Tor circuit
|
||||||
|
Feb 04 15:06:26.000 [notice] new bridge descriptor 'flakey4' (fresh): $2B280B23E1107BB62ABFC40DDCC8824814F80A72~flakey4 [1zOHpg+FxqQfi/6jDLtCpHHqBTH8gjYmCKXkus1D5Ko] at 192.0.2.3
|
||||||
|
Feb 04 15:06:39.000 [notice] Bootstrapped 100% (done): Done
|
||||||
|
Feb 04 15:06:39.000 [notice] Catching signal TERM, exiting cleanly.
|
|
@ -5,11 +5,17 @@
|
||||||
package torsf
|
package torsf
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"regexp"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/apex/log"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/bytecounter"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/netx/archival"
|
"github.com/ooni/probe-cli/v3/internal/engine/netx/archival"
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
"github.com/ooni/probe-cli/v3/internal/ptx"
|
"github.com/ooni/probe-cli/v3/internal/ptx"
|
||||||
|
@ -17,11 +23,18 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
// testVersion is the tor experiment version.
|
// testVersion is the tor experiment version.
|
||||||
const testVersion = "0.1.1"
|
const testVersion = "0.2.0"
|
||||||
|
|
||||||
// Config contains the experiment config.
|
// Config contains the experiment config.
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
// DisablePersistentDatadir disables using a persistent datadir.
|
||||||
|
DisablePersistentDatadir bool `ooni:"Disable using a persistent tor datadir"`
|
||||||
|
|
||||||
|
// DisableProgress disables printing progress messages.
|
||||||
DisableProgress bool `ooni:"Disable printing progress messages"`
|
DisableProgress bool `ooni:"Disable printing progress messages"`
|
||||||
|
|
||||||
|
// RendezvousMethod allows to choose the method with which to rendezvous.
|
||||||
|
RendezvousMethod string `ooni:"Choose the method with which to rendezvous. Must be one of amp and domain_fronting. Leaving this field empty means we should use the default."`
|
||||||
}
|
}
|
||||||
|
|
||||||
// TestKeys contains the experiment's result.
|
// TestKeys contains the experiment's result.
|
||||||
|
@ -31,6 +44,18 @@ type TestKeys struct {
|
||||||
|
|
||||||
// Failure contains the failure string or nil.
|
// Failure contains the failure string or nil.
|
||||||
Failure *string `json:"failure"`
|
Failure *string `json:"failure"`
|
||||||
|
|
||||||
|
// PersistentDatadir indicates whether we're using a persistent tor datadir.
|
||||||
|
PersistentDatadir bool `json:"persistent_datadir"`
|
||||||
|
|
||||||
|
// RendezvousMethod contains the method used to perform the rendezvous.
|
||||||
|
RendezvousMethod string `json:"rendezvous_method"`
|
||||||
|
|
||||||
|
// TorLogs contains the bootstrap logs.
|
||||||
|
TorLogs []string `json:"tor_logs"`
|
||||||
|
|
||||||
|
// TorVersion contains the version of tor (if it's possible to obtain it).
|
||||||
|
TorVersion string `json:"tor_version"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Measurer performs the measurement.
|
// Measurer performs the measurement.
|
||||||
|
@ -44,7 +69,8 @@ type Measurer struct {
|
||||||
|
|
||||||
// mockStartTunnel is an optional function that allows us to override the
|
// mockStartTunnel is an optional function that allows us to override the
|
||||||
// default tunnel.Start function used to start a tunnel.
|
// default tunnel.Start function used to start a tunnel.
|
||||||
mockStartTunnel func(ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, error)
|
mockStartTunnel func(
|
||||||
|
ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, tunnel.DebugInfo, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ExperimentName implements model.ExperimentMeasurer.ExperimentName.
|
// ExperimentName implements model.ExperimentMeasurer.ExperimentName.
|
||||||
|
@ -73,22 +99,27 @@ func (m *Measurer) Run(
|
||||||
ctx context.Context, sess model.ExperimentSession,
|
ctx context.Context, sess model.ExperimentSession,
|
||||||
measurement *model.Measurement, callbacks model.ExperimentCallbacks,
|
measurement *model.Measurement, callbacks model.ExperimentCallbacks,
|
||||||
) error {
|
) error {
|
||||||
|
ptl, sfdialer, err := m.setup(ctx, sess.Logger())
|
||||||
|
if err != nil {
|
||||||
|
// we cannot setup the experiment
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer ptl.Stop()
|
||||||
m.registerExtensions(measurement)
|
m.registerExtensions(measurement)
|
||||||
testkeys := &TestKeys{}
|
|
||||||
measurement.TestKeys = testkeys
|
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
const maxRuntime = 600 * time.Second
|
const maxRuntime = 600 * time.Second
|
||||||
ctx, cancel := context.WithTimeout(ctx, maxRuntime)
|
ctx, cancel := context.WithTimeout(ctx, maxRuntime)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
errch := make(chan error)
|
tkch := make(chan *TestKeys)
|
||||||
ticker := time.NewTicker(2 * time.Second)
|
ticker := time.NewTicker(2 * time.Second)
|
||||||
defer ticker.Stop()
|
defer ticker.Stop()
|
||||||
go m.run(ctx, sess, testkeys, errch)
|
go m.bootstrap(ctx, sess, tkch, ptl, sfdialer)
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case err := <-errch:
|
case tk := <-tkch:
|
||||||
|
measurement.TestKeys = tk
|
||||||
callbacks.OnProgress(1.0, "torsf experiment is finished")
|
callbacks.OnProgress(1.0, "torsf experiment is finished")
|
||||||
return err
|
return nil
|
||||||
case <-ticker.C:
|
case <-ticker.C:
|
||||||
if !m.config.DisableProgress {
|
if !m.config.DisableProgress {
|
||||||
elapsedTime := time.Since(start)
|
elapsedTime := time.Since(start)
|
||||||
|
@ -101,28 +132,50 @@ func (m *Measurer) Run(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// run runs the bootstrap. This function ONLY returns an error when
|
// setup prepares for running the torsf experiment. Returns a valid ptx listener
|
||||||
// there has been a fundamental error starting the test. This behavior
|
// and snowflake dialer on success. Returns an error on failure. On success,
|
||||||
// follows the expectations for the ExperimentMeasurer.Run method.
|
// remember to Stop the ptx listener when you're done.
|
||||||
func (m *Measurer) run(ctx context.Context,
|
func (m *Measurer) setup(ctx context.Context,
|
||||||
sess model.ExperimentSession, testkeys *TestKeys, errch chan<- error) {
|
logger model.Logger) (*ptx.Listener, *ptx.SnowflakeDialer, error) {
|
||||||
sfdialer := &ptx.SnowflakeDialer{}
|
rm, err := ptx.NewSnowflakeRendezvousMethod(m.config.RendezvousMethod)
|
||||||
|
if err != nil {
|
||||||
|
// cannot run the experiment with unknown rendezvous method
|
||||||
|
return nil, nil, err
|
||||||
|
}
|
||||||
|
sfdialer := ptx.NewSnowflakeDialerWithRendezvousMethod(rm)
|
||||||
ptl := &ptx.Listener{
|
ptl := &ptx.Listener{
|
||||||
|
ExperimentByteCounter: bytecounter.ContextExperimentByteCounter(ctx),
|
||||||
|
Logger: logger,
|
||||||
PTDialer: sfdialer,
|
PTDialer: sfdialer,
|
||||||
Logger: sess.Logger(),
|
SessionByteCounter: bytecounter.ContextSessionByteCounter(ctx),
|
||||||
}
|
}
|
||||||
if err := m.startListener(ptl.Start); err != nil {
|
if err := m.startListener(ptl.Start); err != nil {
|
||||||
testkeys.Failure = archival.NewFailure(err)
|
|
||||||
// This error condition mostly means "I could not open a local
|
// This error condition mostly means "I could not open a local
|
||||||
// listening port", which strikes as fundamental failure.
|
// listening port", which strikes as fundamental failure.
|
||||||
errch <- err
|
return nil, nil, err
|
||||||
return
|
|
||||||
}
|
}
|
||||||
defer ptl.Stop()
|
logger.Infof("torsf: rendezvous method: '%s'", m.config.RendezvousMethod)
|
||||||
tun, err := m.startTunnel()(ctx, &tunnel.Config{
|
return ptl, sfdialer, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// bootstrap runs the bootstrap.
|
||||||
|
func (m *Measurer) bootstrap(ctx context.Context, sess model.ExperimentSession,
|
||||||
|
out chan<- *TestKeys, ptl *ptx.Listener, sfdialer *ptx.SnowflakeDialer) {
|
||||||
|
tk := &TestKeys{
|
||||||
|
BootstrapTime: 0,
|
||||||
|
Failure: nil,
|
||||||
|
PersistentDatadir: !m.config.DisablePersistentDatadir,
|
||||||
|
RendezvousMethod: sfdialer.RendezvousMethod.Name(),
|
||||||
|
}
|
||||||
|
sess.Logger().Infof(
|
||||||
|
"torsf: disable persistent datadir: %+v", m.config.DisablePersistentDatadir)
|
||||||
|
defer func() {
|
||||||
|
out <- tk
|
||||||
|
}()
|
||||||
|
tun, debugInfo, err := m.startTunnel()(ctx, &tunnel.Config{
|
||||||
Name: "tor",
|
Name: "tor",
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: path.Join(sess.TempDir(), "torsf"),
|
TunnelDir: path.Join(m.baseTunnelDir(sess), "torsf"),
|
||||||
Logger: sess.Logger(),
|
Logger: sess.Logger(),
|
||||||
TorArgs: []string{
|
TorArgs: []string{
|
||||||
"UseBridges", "1",
|
"UseBridges", "1",
|
||||||
|
@ -130,18 +183,61 @@ func (m *Measurer) run(ctx context.Context,
|
||||||
"Bridge", sfdialer.AsBridgeArgument(),
|
"Bridge", sfdialer.AsBridgeArgument(),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
tk.TorVersion = debugInfo.Version
|
||||||
|
m.readTorLogs(sess.Logger(), tk, debugInfo.LogFilePath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// Note: archival.NewFailure scrubs IP addresses
|
// Note: archival.NewFailure scrubs IP addresses
|
||||||
testkeys.Failure = archival.NewFailure(err)
|
tk.Failure = archival.NewFailure(err)
|
||||||
// This error condition means we could not bootstrap with snowflake
|
|
||||||
// for $reasons, so the experiment didn't fail, rather it did record
|
|
||||||
// that something prevented snowflake from running.
|
|
||||||
errch <- nil
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
defer tun.Stop()
|
defer tun.Stop()
|
||||||
testkeys.BootstrapTime = tun.BootstrapTime().Seconds()
|
tk.BootstrapTime = tun.BootstrapTime().Seconds()
|
||||||
errch <- nil
|
}
|
||||||
|
|
||||||
|
// torProgressRegexp helps to extract progress info from logs.
|
||||||
|
//
|
||||||
|
// See https://regex101.com/r/3YfIed/1.
|
||||||
|
var torProgressRegexp = regexp.MustCompile(
|
||||||
|
`^[A-Za-z0-9.: ]+ \[notice\] Bootstrapped [0-9]+% \([a-zA-z]+\): [A-Za-z0-9 ]+$`)
|
||||||
|
|
||||||
|
// readTorLogs attempts to read and include the tor logs into
|
||||||
|
// the test keys if this operation is possible.
|
||||||
|
//
|
||||||
|
// This function aims to _only_ include notice information about
|
||||||
|
// bootstrap according to the torProgressRegexp regexp.
|
||||||
|
//
|
||||||
|
// Tor is know to be good software that does not break its output
|
||||||
|
// unnecessarily and that does not include PII into its logs unless
|
||||||
|
// explicitly asked to. This fact gives me confidence that we can
|
||||||
|
// safely include this subset of the logs into the results.
|
||||||
|
//
|
||||||
|
// On this note, I think it's safe to include timestamps from the
|
||||||
|
// logs into the output, since we have a timestamp for the whole
|
||||||
|
// experiment already, so we don't leak much more by also including
|
||||||
|
// the Tor proper timestamps into the results.
|
||||||
|
func (m *Measurer) readTorLogs(logger model.Logger, tk *TestKeys, logFilePath string) {
|
||||||
|
if logFilePath == "" {
|
||||||
|
log.Warn("the tunnel claims there is no log file")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
data, err := os.ReadFile(logFilePath)
|
||||||
|
if err != nil {
|
||||||
|
log.Warnf("could not read tor logs: %s", err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, bline := range bytes.Split(data, []byte("\n")) {
|
||||||
|
if torProgressRegexp.Match(bline) {
|
||||||
|
tk.TorLogs = append(tk.TorLogs, string(bline))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// baseTunnelDir returns the base directory to use for tunnelling
|
||||||
|
func (m *Measurer) baseTunnelDir(sess model.ExperimentSession) string {
|
||||||
|
if m.config.DisablePersistentDatadir {
|
||||||
|
return sess.TempDir()
|
||||||
|
}
|
||||||
|
return sess.TunnelDir()
|
||||||
}
|
}
|
||||||
|
|
||||||
// startListener either calls f or mockStartListener depending
|
// startListener either calls f or mockStartListener depending
|
||||||
|
@ -155,7 +251,7 @@ func (m *Measurer) startListener(f func() error) error {
|
||||||
|
|
||||||
// startTunnel returns the proper function to start a tunnel.
|
// startTunnel returns the proper function to start a tunnel.
|
||||||
func (m *Measurer) startTunnel() func(
|
func (m *Measurer) startTunnel() func(
|
||||||
ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, error) {
|
ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, tunnel.DebugInfo, error) {
|
||||||
if m.mockStartTunnel != nil {
|
if m.mockStartTunnel != nil {
|
||||||
return m.mockStartTunnel
|
return m.mockStartTunnel
|
||||||
}
|
}
|
||||||
|
@ -175,7 +271,22 @@ type SummaryKeys struct {
|
||||||
IsAnomaly bool `json:"-"`
|
IsAnomaly bool `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
// errInvalidTestKeysType indicates the test keys type is invalid.
|
||||||
|
errInvalidTestKeysType = errors.New("torsf: invalid test keys type")
|
||||||
|
|
||||||
|
//errNilTestKeys indicates that the test keys are nil.
|
||||||
|
errNilTestKeys = errors.New("torsf: nil test keys")
|
||||||
|
)
|
||||||
|
|
||||||
// GetSummaryKeys implements model.ExperimentMeasurer.GetSummaryKeys.
|
// GetSummaryKeys implements model.ExperimentMeasurer.GetSummaryKeys.
|
||||||
func (m *Measurer) GetSummaryKeys(measurement *model.Measurement) (interface{}, error) {
|
func (m *Measurer) GetSummaryKeys(measurement *model.Measurement) (interface{}, error) {
|
||||||
return SummaryKeys{IsAnomaly: false}, nil
|
testkeys, good := measurement.TestKeys.(*TestKeys)
|
||||||
|
if !good {
|
||||||
|
return nil, errInvalidTestKeysType
|
||||||
|
}
|
||||||
|
if testkeys == nil {
|
||||||
|
return nil, errNilTestKeys
|
||||||
|
}
|
||||||
|
return SummaryKeys{IsAnomaly: testkeys.Failure != nil}, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,14 +3,16 @@ package torsf
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"net/url"
|
"path/filepath"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/apex/log"
|
"github.com/ooni/probe-cli/v3/internal/atomicx"
|
||||||
"github.com/ooni/probe-cli/v3/internal/engine/mockable"
|
"github.com/ooni/probe-cli/v3/internal/engine/mockable"
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/ptx"
|
||||||
"github.com/ooni/probe-cli/v3/internal/tunnel"
|
"github.com/ooni/probe-cli/v3/internal/tunnel"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/tunnel/mocks"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestExperimentNameAndVersion(t *testing.T) {
|
func TestExperimentNameAndVersion(t *testing.T) {
|
||||||
|
@ -18,85 +20,34 @@ func TestExperimentNameAndVersion(t *testing.T) {
|
||||||
if m.ExperimentName() != "torsf" {
|
if m.ExperimentName() != "torsf" {
|
||||||
t.Fatal("invalid experiment name")
|
t.Fatal("invalid experiment name")
|
||||||
}
|
}
|
||||||
if m.ExperimentVersion() != "0.1.1" {
|
if m.ExperimentVersion() != "0.2.0" {
|
||||||
t.Fatal("invalid experiment version")
|
t.Fatal("invalid experiment version")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// mockedTunnel is a mocked tunnel.
|
func TestFailureWithInvalidRendezvousMethod(t *testing.T) {
|
||||||
type mockedTunnel struct {
|
|
||||||
bootstrapTime time.Duration
|
|
||||||
proxyURL *url.URL
|
|
||||||
}
|
|
||||||
|
|
||||||
// BootstrapTime implements Tunnel.BootstrapTime.
|
|
||||||
func (mt *mockedTunnel) BootstrapTime() time.Duration {
|
|
||||||
return mt.bootstrapTime
|
|
||||||
}
|
|
||||||
|
|
||||||
// SOCKS5ProxyURL implements Tunnel.SOCKS5ProxyURL.
|
|
||||||
func (mt *mockedTunnel) SOCKS5ProxyURL() *url.URL {
|
|
||||||
return mt.proxyURL
|
|
||||||
}
|
|
||||||
|
|
||||||
// Stop implements Tunnel.Stop.
|
|
||||||
func (mt *mockedTunnel) Stop() {
|
|
||||||
// nothing
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestSuccessWithMockedTunnelStart(t *testing.T) {
|
|
||||||
bootstrapTime := 400 * time.Millisecond
|
|
||||||
m := &Measurer{
|
m := &Measurer{
|
||||||
config: Config{},
|
config: Config{
|
||||||
mockStartTunnel: func(ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, error) {
|
DisablePersistentDatadir: false,
|
||||||
// run for some time so we also exercise printing progress.
|
DisableProgress: false,
|
||||||
time.Sleep(bootstrapTime)
|
RendezvousMethod: "antani",
|
||||||
return &mockedTunnel{
|
|
||||||
bootstrapTime: time.Duration(bootstrapTime),
|
|
||||||
}, nil
|
|
||||||
},
|
},
|
||||||
|
mockStartTunnel: nil,
|
||||||
}
|
}
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
measurement := &model.Measurement{}
|
measurement := &model.Measurement{}
|
||||||
sess := &mockable.Session{}
|
sess := &mockable.Session{
|
||||||
|
MockableLogger: model.DiscardLogger,
|
||||||
|
}
|
||||||
callbacks := &model.PrinterCallbacks{
|
callbacks := &model.PrinterCallbacks{
|
||||||
Logger: log.Log,
|
Logger: model.DiscardLogger,
|
||||||
}
|
}
|
||||||
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
err := m.Run(ctx, sess, measurement, callbacks)
|
||||||
t.Fatal(err)
|
if !errors.Is(err, ptx.ErrSnowflakeNoSuchRendezvousMethod) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
}
|
}
|
||||||
tk := measurement.TestKeys.(*TestKeys)
|
if measurement.TestKeys != nil {
|
||||||
if tk.BootstrapTime != bootstrapTime.Seconds() {
|
t.Fatal("expected nil test keys")
|
||||||
t.Fatal("unexpected bootstrap time")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestFailureToStartTunnel(t *testing.T) {
|
|
||||||
expected := errors.New("mocked error")
|
|
||||||
m := &Measurer{
|
|
||||||
config: Config{},
|
|
||||||
mockStartTunnel: func(ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, error) {
|
|
||||||
return nil, expected
|
|
||||||
},
|
|
||||||
}
|
|
||||||
ctx := context.Background()
|
|
||||||
measurement := &model.Measurement{}
|
|
||||||
sess := &mockable.Session{}
|
|
||||||
callbacks := &model.PrinterCallbacks{
|
|
||||||
Logger: log.Log,
|
|
||||||
}
|
|
||||||
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
|
||||||
t.Fatal(err)
|
|
||||||
}
|
|
||||||
tk := measurement.TestKeys.(*TestKeys)
|
|
||||||
if tk.BootstrapTime != 0 {
|
|
||||||
t.Fatal("unexpected bootstrap time")
|
|
||||||
}
|
|
||||||
if tk.Failure == nil {
|
|
||||||
t.Fatal("unexpectedly nil failure string")
|
|
||||||
}
|
|
||||||
if *tk.Failure != "unknown_failure: mocked error" {
|
|
||||||
t.Fatal("unexpected failure string", *tk.Failure)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,11 +63,130 @@ func TestFailureToStartPTXListener(t *testing.T) {
|
||||||
measurement := &model.Measurement{}
|
measurement := &model.Measurement{}
|
||||||
sess := &mockable.Session{}
|
sess := &mockable.Session{}
|
||||||
callbacks := &model.PrinterCallbacks{
|
callbacks := &model.PrinterCallbacks{
|
||||||
Logger: log.Log,
|
Logger: model.DiscardLogger,
|
||||||
}
|
}
|
||||||
if err := m.Run(ctx, sess, measurement, callbacks); !errors.Is(err, expected) {
|
if err := m.Run(ctx, sess, measurement, callbacks); !errors.Is(err, expected) {
|
||||||
t.Fatal("not the error we expected", err)
|
t.Fatal("not the error we expected", err)
|
||||||
}
|
}
|
||||||
|
if tk := measurement.TestKeys; tk != nil {
|
||||||
|
t.Fatal("expected nil bootstrap time here")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSuccessWithMockedTunnelStart(t *testing.T) {
|
||||||
|
bootstrapTime := 3 * time.Second
|
||||||
|
called := &atomicx.Int64{}
|
||||||
|
m := &Measurer{
|
||||||
|
config: Config{},
|
||||||
|
mockStartTunnel: func(
|
||||||
|
ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, tunnel.DebugInfo, error) {
|
||||||
|
// run for some time so we also exercise printing progress.
|
||||||
|
time.Sleep(bootstrapTime)
|
||||||
|
return &mocks.Tunnel{
|
||||||
|
MockBootstrapTime: func() time.Duration {
|
||||||
|
return bootstrapTime
|
||||||
|
},
|
||||||
|
MockStop: func() {
|
||||||
|
called.Add(1)
|
||||||
|
},
|
||||||
|
}, tunnel.DebugInfo{
|
||||||
|
Name: "tor",
|
||||||
|
LogFilePath: filepath.Join("testdata", "tor.log"),
|
||||||
|
}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
ctx := context.Background()
|
||||||
|
measurement := &model.Measurement{}
|
||||||
|
sess := &mockable.Session{
|
||||||
|
MockableLogger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
callbacks := &model.PrinterCallbacks{
|
||||||
|
Logger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if called.Load() != 1 {
|
||||||
|
t.Fatal("stop was not called")
|
||||||
|
}
|
||||||
|
tk := measurement.TestKeys.(*TestKeys)
|
||||||
|
if tk.BootstrapTime != bootstrapTime.Seconds() {
|
||||||
|
t.Fatal("unexpected bootstrap time")
|
||||||
|
}
|
||||||
|
if tk.Failure != nil {
|
||||||
|
t.Fatal("unexpected failure")
|
||||||
|
}
|
||||||
|
if !tk.PersistentDatadir {
|
||||||
|
t.Fatal("unexpected persistent data dir")
|
||||||
|
}
|
||||||
|
if tk.RendezvousMethod != "domain_fronting" {
|
||||||
|
t.Fatal("unexpected rendezvous method")
|
||||||
|
}
|
||||||
|
if count := len(tk.TorLogs); count != 9 {
|
||||||
|
t.Fatal("unexpected length of tor logs", count)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestWithCancelledContext(t *testing.T) {
|
||||||
|
// This test calls the real tunnel.Start function so we cover
|
||||||
|
// it but fails immediately because of the cancelled ctx.
|
||||||
|
m := &Measurer{
|
||||||
|
config: Config{},
|
||||||
|
}
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
cancel() // fail immediately
|
||||||
|
measurement := &model.Measurement{}
|
||||||
|
sess := &mockable.Session{
|
||||||
|
MockableLogger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
callbacks := &model.PrinterCallbacks{
|
||||||
|
Logger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
tk := measurement.TestKeys.(*TestKeys)
|
||||||
|
if tk.BootstrapTime != 0 {
|
||||||
|
t.Fatal("unexpected bootstrap time")
|
||||||
|
}
|
||||||
|
if *tk.Failure != "interrupted" {
|
||||||
|
t.Fatal("unexpected failure")
|
||||||
|
}
|
||||||
|
if !tk.PersistentDatadir {
|
||||||
|
t.Fatal("unexpected persistent data dir")
|
||||||
|
}
|
||||||
|
if tk.RendezvousMethod != "domain_fronting" {
|
||||||
|
t.Fatal("unexpected rendezvous method")
|
||||||
|
}
|
||||||
|
if len(tk.TorLogs) != 0 {
|
||||||
|
t.Fatal("unexpected length of tor logs")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFailureToStartTunnel(t *testing.T) {
|
||||||
|
expected := errors.New("mocked error")
|
||||||
|
m := &Measurer{
|
||||||
|
config: Config{},
|
||||||
|
mockStartTunnel: func(
|
||||||
|
ctx context.Context, config *tunnel.Config) (tunnel.Tunnel, tunnel.DebugInfo, error) {
|
||||||
|
return nil,
|
||||||
|
tunnel.DebugInfo{
|
||||||
|
Name: "tor",
|
||||||
|
LogFilePath: filepath.Join("testdata", "tor.log"),
|
||||||
|
}, expected
|
||||||
|
},
|
||||||
|
}
|
||||||
|
ctx := context.Background()
|
||||||
|
measurement := &model.Measurement{}
|
||||||
|
sess := &mockable.Session{
|
||||||
|
MockableLogger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
callbacks := &model.PrinterCallbacks{
|
||||||
|
Logger: model.DiscardLogger,
|
||||||
|
}
|
||||||
|
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
tk := measurement.TestKeys.(*TestKeys)
|
tk := measurement.TestKeys.(*TestKeys)
|
||||||
if tk.BootstrapTime != 0 {
|
if tk.BootstrapTime != 0 {
|
||||||
t.Fatal("unexpected bootstrap time")
|
t.Fatal("unexpected bootstrap time")
|
||||||
|
@ -127,34 +197,124 @@ func TestFailureToStartPTXListener(t *testing.T) {
|
||||||
if *tk.Failure != "unknown_failure: mocked error" {
|
if *tk.Failure != "unknown_failure: mocked error" {
|
||||||
t.Fatal("unexpected failure string", *tk.Failure)
|
t.Fatal("unexpected failure string", *tk.Failure)
|
||||||
}
|
}
|
||||||
|
if !tk.PersistentDatadir {
|
||||||
|
t.Fatal("unexpected persistent datadir")
|
||||||
|
}
|
||||||
|
if tk.RendezvousMethod != "domain_fronting" {
|
||||||
|
t.Fatal("unexpected rendezvous method")
|
||||||
|
}
|
||||||
|
if count := len(tk.TorLogs); count != 9 {
|
||||||
|
t.Fatal("unexpected length of tor logs", count)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestStartWithCancelledContext(t *testing.T) {
|
func TestBaseTunnelDir(t *testing.T) {
|
||||||
m := &Measurer{config: Config{}}
|
t.Run("without persistent data dir", func(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
m := &Measurer{
|
||||||
cancel() // fail immediately
|
config: Config{
|
||||||
measurement := &model.Measurement{}
|
DisablePersistentDatadir: true,
|
||||||
sess := &mockable.Session{}
|
},
|
||||||
callbacks := &model.PrinterCallbacks{
|
|
||||||
Logger: log.Log,
|
|
||||||
}
|
}
|
||||||
if err := m.Run(ctx, sess, measurement, callbacks); err != nil {
|
sess := &mockable.Session{
|
||||||
t.Fatal(err)
|
MockableTunnelDir: "a",
|
||||||
|
MockableTempDir: "b",
|
||||||
}
|
}
|
||||||
tk := measurement.TestKeys.(*TestKeys)
|
dir := m.baseTunnelDir(sess)
|
||||||
if tk.BootstrapTime != 0 {
|
if dir != "b" {
|
||||||
t.Fatal("unexpected bootstrap time")
|
t.Fatal("unexpected base tunnel dir", dir)
|
||||||
}
|
}
|
||||||
if tk.Failure == nil {
|
})
|
||||||
t.Fatal("unexpected nil failure")
|
|
||||||
|
t.Run("with persistent data dir", func(t *testing.T) {
|
||||||
|
m := &Measurer{
|
||||||
|
config: Config{
|
||||||
|
DisablePersistentDatadir: false,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
if *tk.Failure != "interrupted" {
|
sess := &mockable.Session{
|
||||||
t.Fatal("unexpected failure string", *tk.Failure)
|
MockableTunnelDir: "a",
|
||||||
|
MockableTempDir: "b",
|
||||||
}
|
}
|
||||||
|
dir := m.baseTunnelDir(sess)
|
||||||
|
if dir != "a" {
|
||||||
|
t.Fatal("unexpected base tunnel dir", dir)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestReadTorLogs(t *testing.T) {
|
||||||
|
t.Run("with empty file path", func(t *testing.T) {
|
||||||
|
m := &Measurer{}
|
||||||
|
logger := model.DiscardLogger
|
||||||
|
tk := &TestKeys{}
|
||||||
|
m.readTorLogs(logger, tk, "")
|
||||||
|
if len(tk.TorLogs) != 0 {
|
||||||
|
t.Fatal("expected no tor logs")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("with nonexistent file path", func(t *testing.T) {
|
||||||
|
m := &Measurer{}
|
||||||
|
logger := model.DiscardLogger
|
||||||
|
tk := &TestKeys{}
|
||||||
|
m.readTorLogs(logger, tk, filepath.Join("testdata", "nonexistent"))
|
||||||
|
if len(tk.TorLogs) != 0 {
|
||||||
|
t.Fatal("expected no tor logs")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("with existing file path", func(t *testing.T) {
|
||||||
|
m := &Measurer{}
|
||||||
|
logger := model.DiscardLogger
|
||||||
|
tk := &TestKeys{}
|
||||||
|
m.readTorLogs(logger, tk, filepath.Join("testdata", "tor.log"))
|
||||||
|
if count := len(tk.TorLogs); count != 9 {
|
||||||
|
t.Fatal("unexpected number of tor logs", count)
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGetSummaryKeys(t *testing.T) {
|
func TestGetSummaryKeys(t *testing.T) {
|
||||||
measurement := &model.Measurement{}
|
t.Run("in case of untyped nil TestKeys", func(t *testing.T) {
|
||||||
|
measurement := &model.Measurement{
|
||||||
|
TestKeys: nil,
|
||||||
|
}
|
||||||
|
m := &Measurer{}
|
||||||
|
_, err := m.GetSummaryKeys(measurement)
|
||||||
|
if !errors.Is(err, errInvalidTestKeysType) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("in case of typed nil TestKeys", func(t *testing.T) {
|
||||||
|
var tk *TestKeys
|
||||||
|
measurement := &model.Measurement{
|
||||||
|
TestKeys: tk,
|
||||||
|
}
|
||||||
|
m := &Measurer{}
|
||||||
|
_, err := m.GetSummaryKeys(measurement)
|
||||||
|
if !errors.Is(err, errNilTestKeys) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("in case of invalid TestKeys type", func(t *testing.T) {
|
||||||
|
measurement := &model.Measurement{
|
||||||
|
TestKeys: make(chan int),
|
||||||
|
}
|
||||||
|
m := &Measurer{}
|
||||||
|
_, err := m.GetSummaryKeys(measurement)
|
||||||
|
if !errors.Is(err, errInvalidTestKeysType) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("in case of success", func(t *testing.T) {
|
||||||
|
measurement := &model.Measurement{
|
||||||
|
TestKeys: &TestKeys{
|
||||||
|
Failure: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
m := &Measurer{}
|
m := &Measurer{}
|
||||||
sk, err := m.GetSummaryKeys(measurement)
|
sk, err := m.GetSummaryKeys(measurement)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -164,4 +324,23 @@ func TestGetSummaryKeys(t *testing.T) {
|
||||||
if rsk.IsAnomaly {
|
if rsk.IsAnomaly {
|
||||||
t.Fatal("expected no anomaly here")
|
t.Fatal("expected no anomaly here")
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("in case of failure", func(t *testing.T) {
|
||||||
|
failure := "generic_timeout_error"
|
||||||
|
measurement := &model.Measurement{
|
||||||
|
TestKeys: &TestKeys{
|
||||||
|
Failure: &failure,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
m := &Measurer{}
|
||||||
|
sk, err := m.GetSummaryKeys(measurement)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
rsk := sk.(SummaryKeys)
|
||||||
|
if !rsk.IsAnomaly {
|
||||||
|
t.Fatal("expected anomaly here")
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -111,7 +111,7 @@ func (g Getter) get(ctx context.Context, saver *trace.Saver) (TestKeys, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return tk, err
|
return tk, err
|
||||||
}
|
}
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: g.Config.Tunnel,
|
Name: g.Config.Tunnel,
|
||||||
Session: g.Session,
|
Session: g.Session,
|
||||||
TorArgs: g.Session.TorArgs(),
|
TorArgs: g.Session.TorArgs(),
|
||||||
|
|
|
@ -34,6 +34,7 @@ type Session struct {
|
||||||
MockableTempDir string
|
MockableTempDir string
|
||||||
MockableTorArgs []string
|
MockableTorArgs []string
|
||||||
MockableTorBinary string
|
MockableTorBinary string
|
||||||
|
MockableTunnelDir string
|
||||||
MockableUserAgent string
|
MockableUserAgent string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,6 +136,11 @@ func (sess *Session) TorBinary() string {
|
||||||
return sess.MockableTorBinary
|
return sess.MockableTorBinary
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TunnelDir implements ExperimentSession.TunnelDir.
|
||||||
|
func (sess *Session) TunnelDir() string {
|
||||||
|
return sess.MockableTunnelDir
|
||||||
|
}
|
||||||
|
|
||||||
// UserAgent implements ExperimentSession.UserAgent
|
// UserAgent implements ExperimentSession.UserAgent
|
||||||
func (sess *Session) UserAgent() string {
|
func (sess *Session) UserAgent() string {
|
||||||
return sess.MockableUserAgent
|
return sess.MockableUserAgent
|
||||||
|
|
|
@ -21,37 +21,6 @@ func (d *byteCounterDialer) DialContext(
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if exp := contextExperimentByteCounter(ctx); exp != nil {
|
conn = bytecounter.MaybeWrapWithContextByteCounters(ctx, conn)
|
||||||
conn = &bytecounter.Conn{Conn: conn, Counter: exp}
|
|
||||||
}
|
|
||||||
if sess := contextSessionByteCounter(ctx); sess != nil {
|
|
||||||
conn = &bytecounter.Conn{Conn: conn, Counter: sess}
|
|
||||||
}
|
|
||||||
return conn, nil
|
return conn, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type byteCounterSessionKey struct{}
|
|
||||||
|
|
||||||
// contextSessionByteCounter retrieves the session byte counter from the context
|
|
||||||
func contextSessionByteCounter(ctx context.Context) *bytecounter.Counter {
|
|
||||||
counter, _ := ctx.Value(byteCounterSessionKey{}).(*bytecounter.Counter)
|
|
||||||
return counter
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithSessionByteCounter assigns the session byte counter to the context.
|
|
||||||
func WithSessionByteCounter(ctx context.Context, counter *bytecounter.Counter) context.Context {
|
|
||||||
return context.WithValue(ctx, byteCounterSessionKey{}, counter)
|
|
||||||
}
|
|
||||||
|
|
||||||
type byteCounterExperimentKey struct{}
|
|
||||||
|
|
||||||
// contextExperimentByteCounter retrieves the experiment byte counter from the context
|
|
||||||
func contextExperimentByteCounter(ctx context.Context) *bytecounter.Counter {
|
|
||||||
counter, _ := ctx.Value(byteCounterExperimentKey{}).(*bytecounter.Counter)
|
|
||||||
return counter
|
|
||||||
}
|
|
||||||
|
|
||||||
// WithExperimentByteCounter assigns the experiment byte counter to the context.
|
|
||||||
func WithExperimentByteCounter(ctx context.Context, counter *bytecounter.Counter) context.Context {
|
|
||||||
return context.WithValue(ctx, byteCounterExperimentKey{}, counter)
|
|
||||||
}
|
|
||||||
|
|
|
@ -39,12 +39,12 @@ func TestByteCounterNormalUsage(t *testing.T) {
|
||||||
}
|
}
|
||||||
sess := bytecounter.New()
|
sess := bytecounter.New()
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
ctx = WithSessionByteCounter(ctx, sess)
|
ctx = bytecounter.WithSessionByteCounter(ctx, sess)
|
||||||
if err := dorequest(ctx, "http://www.google.com"); err != nil {
|
if err := dorequest(ctx, "http://www.google.com"); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
exp := bytecounter.New()
|
exp := bytecounter.New()
|
||||||
ctx = WithExperimentByteCounter(ctx, exp)
|
ctx = bytecounter.WithExperimentByteCounter(ctx, exp)
|
||||||
if err := dorequest(ctx, "http://facebook.com"); err != nil {
|
if err := dorequest(ctx, "http://facebook.com"); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -95,6 +95,9 @@ type Session struct {
|
||||||
// may need to pass to urlgetter when it uses a tor tunnel.
|
// may need to pass to urlgetter when it uses a tor tunnel.
|
||||||
torBinary string
|
torBinary string
|
||||||
|
|
||||||
|
// tunnelDir is the directory used by tunnels.
|
||||||
|
tunnelDir string
|
||||||
|
|
||||||
// tunnel is the optional tunnel that we may be using. It is created
|
// tunnel is the optional tunnel that we may be using. It is created
|
||||||
// by NewSession and it is cleaned up by Close.
|
// by NewSession and it is cleaned up by Close.
|
||||||
tunnel tunnel.Tunnel
|
tunnel tunnel.Tunnel
|
||||||
|
@ -163,6 +166,7 @@ func NewSession(ctx context.Context, config SessionConfig) (*Session, error) {
|
||||||
tempDir: tempDir,
|
tempDir: tempDir,
|
||||||
torArgs: config.TorArgs,
|
torArgs: config.TorArgs,
|
||||||
torBinary: config.TorBinary,
|
torBinary: config.TorBinary,
|
||||||
|
tunnelDir: config.TunnelDir,
|
||||||
}
|
}
|
||||||
proxyURL := config.ProxyURL
|
proxyURL := config.ProxyURL
|
||||||
if proxyURL != nil {
|
if proxyURL != nil {
|
||||||
|
@ -170,7 +174,7 @@ func NewSession(ctx context.Context, config SessionConfig) (*Session, error) {
|
||||||
case "psiphon", "tor", "fake":
|
case "psiphon", "tor", "fake":
|
||||||
config.Logger.Infof(
|
config.Logger.Infof(
|
||||||
"starting '%s' tunnel; please be patient...", proxyURL.Scheme)
|
"starting '%s' tunnel; please be patient...", proxyURL.Scheme)
|
||||||
tunnel, err := tunnel.Start(ctx, &tunnel.Config{
|
tunnel, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Logger: config.Logger,
|
Logger: config.Logger,
|
||||||
Name: proxyURL.Scheme,
|
Name: proxyURL.Scheme,
|
||||||
Session: &sessionTunnelEarlySession{},
|
Session: &sessionTunnelEarlySession{},
|
||||||
|
@ -204,6 +208,11 @@ func NewSession(ctx context.Context, config SessionConfig) (*Session, error) {
|
||||||
return sess, nil
|
return sess, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TunnelDir returns the persistent directory used by tunnels.
|
||||||
|
func (s *Session) TunnelDir() string {
|
||||||
|
return s.tunnelDir
|
||||||
|
}
|
||||||
|
|
||||||
// KibiBytesReceived accounts for the KibiBytes received by the HTTP clients
|
// KibiBytesReceived accounts for the KibiBytes received by the HTTP clients
|
||||||
// managed by this session so far, including experiments.
|
// managed by this session so far, including experiments.
|
||||||
func (s *Session) KibiBytesReceived() float64 {
|
func (s *Session) KibiBytesReceived() float64 {
|
||||||
|
|
|
@ -23,6 +23,7 @@ type ExperimentSession interface {
|
||||||
TempDir() string
|
TempDir() string
|
||||||
TorArgs() []string
|
TorArgs() []string
|
||||||
TorBinary() string
|
TorBinary() string
|
||||||
|
TunnelDir() string
|
||||||
UserAgent() string
|
UserAgent() string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +0,0 @@
|
||||||
package ptx
|
|
||||||
|
|
||||||
import (
|
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
|
||||||
)
|
|
||||||
|
|
||||||
// defaultLogger is the default silentLogger instance.
|
|
||||||
var defaultLogger model.Logger = model.DiscardLogger
|
|
|
@ -1,11 +0,0 @@
|
||||||
package ptx
|
|
||||||
|
|
||||||
import "testing"
|
|
||||||
|
|
||||||
func TestCoverSilentLogger(t *testing.T) {
|
|
||||||
// let us not be distracted by uncovered lines that can
|
|
||||||
// easily be covered, we can easily cover defaultLogger
|
|
||||||
defaultLogger.Debugf("foo")
|
|
||||||
defaultLogger.Infof("bar")
|
|
||||||
defaultLogger.Warnf("baz")
|
|
||||||
}
|
|
|
@ -125,7 +125,8 @@ type obfs4CancellableDialer struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// dial performs the dial.
|
// dial performs the dial.
|
||||||
func (d *obfs4CancellableDialer) dial(ctx context.Context, network, address string) (net.Conn, error) {
|
func (d *obfs4CancellableDialer) dial(
|
||||||
|
ctx context.Context, network, address string) (net.Conn, error) {
|
||||||
connch, errch := make(chan net.Conn), make(chan error, 1)
|
connch, errch := make(chan net.Conn), make(chan error, 1)
|
||||||
go func() {
|
go func() {
|
||||||
defer close(d.done) // signal we're joining
|
defer close(d.done) // signal we're joining
|
||||||
|
|
|
@ -74,7 +74,8 @@ func TestOBFS4DialerFailsWithConnectionErrorAndContextExpiration(t *testing.T) {
|
||||||
wg := &sync.WaitGroup{}
|
wg := &sync.WaitGroup{}
|
||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
o4d.UnderlyingDialer = &mocks.Dialer{
|
o4d.UnderlyingDialer = &mocks.Dialer{
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
MockDialContext: func(
|
||||||
|
ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
cancel()
|
cancel()
|
||||||
<-sigch
|
<-sigch
|
||||||
wg.Done()
|
wg.Done()
|
||||||
|
@ -114,8 +115,9 @@ func TestOBFS4DialerWorksWithContextExpiration(t *testing.T) {
|
||||||
called := &atomicx.Int64{}
|
called := &atomicx.Int64{}
|
||||||
o4d := DefaultTestingOBFS4Bridge()
|
o4d := DefaultTestingOBFS4Bridge()
|
||||||
o4d.UnderlyingDialer = &mocks.Dialer{
|
o4d.UnderlyingDialer = &mocks.Dialer{
|
||||||
MockDialContext: func(ctx context.Context, network string, address string) (net.Conn, error) {
|
MockDialContext: func(
|
||||||
// We cancel the context before returning the error, which makes
|
ctx context.Context, network string, address string) (net.Conn, error) {
|
||||||
|
// We cancel the context before returning the conn, which makes
|
||||||
// the context cancellation happen before us returning.
|
// the context cancellation happen before us returning.
|
||||||
cancel()
|
cancel()
|
||||||
conn, err := net.Dial(network, address)
|
conn, err := net.Dial(network, address)
|
||||||
|
|
|
@ -46,6 +46,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
pt "git.torproject.org/pluggable-transports/goptlib.git"
|
pt "git.torproject.org/pluggable-transports/goptlib.git"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/bytecounter"
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
"github.com/ooni/probe-cli/v3/internal/netxlite"
|
||||||
)
|
)
|
||||||
|
@ -69,15 +70,23 @@ type PTDialer interface {
|
||||||
// you fill the mandatory fields before using it. Do not modify public
|
// you fill the mandatory fields before using it. Do not modify public
|
||||||
// fields after you called Start, since this causes data races.
|
// fields after you called Start, since this causes data races.
|
||||||
type Listener struct {
|
type Listener struct {
|
||||||
|
// ExperimentByteCounter is the OPTIONAL byte counter that
|
||||||
|
// counts the bytes consumed by the experiment.
|
||||||
|
ExperimentByteCounter *bytecounter.Counter
|
||||||
|
|
||||||
|
// Logger is the OPTIONAL logger. When not set, this library
|
||||||
|
// will not emit logs. (But the underlying pluggable transport
|
||||||
|
// may still emit its own log messages.)
|
||||||
|
Logger model.Logger
|
||||||
|
|
||||||
// PTDialer is the MANDATORY pluggable transports dialer
|
// PTDialer is the MANDATORY pluggable transports dialer
|
||||||
// to use. Both SnowflakeDialer and OBFS4Dialer implement this
|
// to use. Both SnowflakeDialer and OBFS4Dialer implement this
|
||||||
// interface and can be thus safely used here.
|
// interface and can be thus safely used here.
|
||||||
PTDialer PTDialer
|
PTDialer PTDialer
|
||||||
|
|
||||||
// Logger is the optional logger. When not set, this library
|
// SessionByteCounter is the OPTIONAL byte counter that
|
||||||
// will not emit logs. (But the underlying pluggable transport
|
// counts the bytes consumed by the session.
|
||||||
// may still emit its own log messages.)
|
SessionByteCounter *bytecounter.Counter
|
||||||
Logger model.Logger
|
|
||||||
|
|
||||||
// mu provides mutual exclusion for accessing internals.
|
// mu provides mutual exclusion for accessing internals.
|
||||||
mu sync.Mutex
|
mu sync.Mutex
|
||||||
|
@ -100,7 +109,7 @@ func (lst *Listener) logger() model.Logger {
|
||||||
if lst.Logger != nil {
|
if lst.Logger != nil {
|
||||||
return lst.Logger
|
return lst.Logger
|
||||||
}
|
}
|
||||||
return defaultLogger
|
return model.DiscardLogger
|
||||||
}
|
}
|
||||||
|
|
||||||
// forward forwards the traffic from left to right and from right to left
|
// forward forwards the traffic from left to right and from right to left
|
||||||
|
@ -151,6 +160,11 @@ func (lst *Listener) handleSocksConn(ctx context.Context, socksConn ptxSocksConn
|
||||||
lst.logger().Warnf("ptx: ContextDialer.DialContext error: %s", err)
|
lst.logger().Warnf("ptx: ContextDialer.DialContext error: %s", err)
|
||||||
return err // used for testing
|
return err // used for testing
|
||||||
}
|
}
|
||||||
|
// We _must_ wrap the ptConn. Wrapping the socks conn leads us to
|
||||||
|
// count the sent bytes as received and the received bytes as sent:
|
||||||
|
// bytes flow in the opposite direction there for the socks conn.
|
||||||
|
ptConn = bytecounter.MaybeWrap(ptConn, lst.SessionByteCounter)
|
||||||
|
ptConn = bytecounter.MaybeWrap(ptConn, lst.ExperimentByteCounter)
|
||||||
lst.forwardWithContext(ctx, socksConn, ptConn) // transfer ownership
|
lst.forwardWithContext(ctx, socksConn, ptConn) // transfer ownership
|
||||||
return nil // used for testing
|
return nil // used for testing
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,20 +2,130 @@ package ptx
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
sflib "git.torproject.org/pluggable-transports/snowflake.git/v2/client/lib"
|
sflib "git.torproject.org/pluggable-transports/snowflake.git/v2/client/lib"
|
||||||
"github.com/ooni/probe-cli/v3/internal/stuninput"
|
"github.com/ooni/probe-cli/v3/internal/stuninput"
|
||||||
)
|
)
|
||||||
|
|
||||||
// SnowflakeDialer is a dialer for snowflake. When optional fields are
|
// SnowflakeRendezvousMethod is the method which with we perform the rendezvous.
|
||||||
// not specified, we use defaults from the snowflake repository.
|
type SnowflakeRendezvousMethod interface {
|
||||||
|
// Name is the name of the method.
|
||||||
|
Name() string
|
||||||
|
|
||||||
|
// AMPCacheURL returns a suitable AMP cache URL.
|
||||||
|
AMPCacheURL() string
|
||||||
|
|
||||||
|
// BrokerURL returns a suitable broker URL.
|
||||||
|
BrokerURL() string
|
||||||
|
|
||||||
|
// FrontDomain returns a suitable front domain.
|
||||||
|
FrontDomain() string
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSnowflakeRendezvousMethodDomainFronting is a rendezvous method
|
||||||
|
// that uses domain fronting to perform the rendezvous.
|
||||||
|
func NewSnowflakeRendezvousMethodDomainFronting() SnowflakeRendezvousMethod {
|
||||||
|
return &snowflakeRendezvousMethodDomainFronting{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type snowflakeRendezvousMethodDomainFronting struct{}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodDomainFronting) Name() string {
|
||||||
|
return "domain_fronting"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodDomainFronting) AMPCacheURL() string {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodDomainFronting) BrokerURL() string {
|
||||||
|
return "https://snowflake-broker.torproject.net.global.prod.fastly.net/"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodDomainFronting) FrontDomain() string {
|
||||||
|
return "cdn.sstatic.net"
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSnowflakeRendezvousMethodAMP is a rendezvous method that
|
||||||
|
// uses the AMP cache to perform the rendezvous.
|
||||||
|
func NewSnowflakeRendezvousMethodAMP() SnowflakeRendezvousMethod {
|
||||||
|
return &snowflakeRendezvousMethodAMP{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type snowflakeRendezvousMethodAMP struct{}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodAMP) Name() string {
|
||||||
|
return "amp"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodAMP) AMPCacheURL() string {
|
||||||
|
return "https://cdn.ampproject.org/"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodAMP) BrokerURL() string {
|
||||||
|
return "https://snowflake-broker.torproject.net/"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *snowflakeRendezvousMethodAMP) FrontDomain() string {
|
||||||
|
return "www.google.com"
|
||||||
|
}
|
||||||
|
|
||||||
|
// ErrSnowflakeNoSuchRendezvousMethod indicates the given rendezvous
|
||||||
|
// method is not supported by this implementation.
|
||||||
|
var ErrSnowflakeNoSuchRendezvousMethod = errors.New("ptx: unsupported rendezvous method")
|
||||||
|
|
||||||
|
// NewSnowflakeRendezvousMethod creates a new rendezvous method by name. We currently
|
||||||
|
// support the following rendezvous methods:
|
||||||
|
//
|
||||||
|
// 1. "domain_fronting" uses domain fronting with the sstatic.net CDN;
|
||||||
|
//
|
||||||
|
// 2. "" means default and it is currently equivalent to "domain_fronting" (but
|
||||||
|
// we don't guarantee that this default may change over time);
|
||||||
|
//
|
||||||
|
// 3. "amp" uses the AMP cache.
|
||||||
|
//
|
||||||
|
// Returns either a valid rendezvous method or an error.
|
||||||
|
func NewSnowflakeRendezvousMethod(method string) (SnowflakeRendezvousMethod, error) {
|
||||||
|
switch method {
|
||||||
|
case "domain_fronting", "":
|
||||||
|
return NewSnowflakeRendezvousMethodDomainFronting(), nil
|
||||||
|
case "amp":
|
||||||
|
return NewSnowflakeRendezvousMethodAMP(), nil
|
||||||
|
default:
|
||||||
|
return nil, ErrSnowflakeNoSuchRendezvousMethod
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SnowflakeDialer is a dialer for snowflake. You SHOULD either use a factory
|
||||||
|
// for constructing this type or set the fields marked as MANDATORY.
|
||||||
type SnowflakeDialer struct {
|
type SnowflakeDialer struct {
|
||||||
// newClientTransport is an optional hook for creating
|
// RendezvousMethod is the MANDATORY rendezvous method to use.
|
||||||
|
RendezvousMethod SnowflakeRendezvousMethod
|
||||||
|
|
||||||
|
// newClientTransport is an OPTIONAL hook for creating
|
||||||
// an alternative snowflakeTransport in testing.
|
// an alternative snowflakeTransport in testing.
|
||||||
newClientTransport func(config sflib.ClientConfig) (snowflakeTransport, error)
|
newClientTransport func(config sflib.ClientConfig) (snowflakeTransport, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// NewSnowflakeDialer creates a SnowflakeDialer with default settings.
|
||||||
|
func NewSnowflakeDialer() *SnowflakeDialer {
|
||||||
|
return &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
|
newClientTransport: nil,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewSnowflakeDialerWithRendezvousMethod creates a SnowflakeDialer
|
||||||
|
// using the given RendezvousMethod explicitly.
|
||||||
|
func NewSnowflakeDialerWithRendezvousMethod(m SnowflakeRendezvousMethod) *SnowflakeDialer {
|
||||||
|
return &SnowflakeDialer{
|
||||||
|
RendezvousMethod: m,
|
||||||
|
newClientTransport: nil,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// snowflakeTransport is anything that allows us to dial a snowflake
|
// snowflakeTransport is anything that allows us to dial a snowflake
|
||||||
type snowflakeTransport interface {
|
type snowflakeTransport interface {
|
||||||
Dial() (net.Conn, error)
|
Dial() (net.Conn, error)
|
||||||
|
@ -32,9 +142,9 @@ func (d *SnowflakeDialer) dialContext(
|
||||||
ctx context.Context) (net.Conn, chan interface{}, error) {
|
ctx context.Context) (net.Conn, chan interface{}, error) {
|
||||||
done := make(chan interface{})
|
done := make(chan interface{})
|
||||||
txp, err := d.newSnowflakeClient(sflib.ClientConfig{
|
txp, err := d.newSnowflakeClient(sflib.ClientConfig{
|
||||||
BrokerURL: d.brokerURL(),
|
BrokerURL: d.RendezvousMethod.BrokerURL(),
|
||||||
AmpCacheURL: d.ampCacheURL(),
|
AmpCacheURL: d.RendezvousMethod.AMPCacheURL(),
|
||||||
FrontDomain: d.frontDomain(),
|
FrontDomain: d.RendezvousMethod.FrontDomain(),
|
||||||
ICEAddresses: d.iceAddresses(),
|
ICEAddresses: d.iceAddresses(),
|
||||||
KeepLocalAddresses: false,
|
KeepLocalAddresses: false,
|
||||||
Max: d.maxSnowflakes(),
|
Max: d.maxSnowflakes(),
|
||||||
|
@ -68,35 +178,14 @@ func (d *SnowflakeDialer) dialContext(
|
||||||
|
|
||||||
// newSnowflakeClient allows us to call a mock rather than
|
// newSnowflakeClient allows us to call a mock rather than
|
||||||
// the real sflib.NewSnowflakeClient.
|
// the real sflib.NewSnowflakeClient.
|
||||||
func (d *SnowflakeDialer) newSnowflakeClient(config sflib.ClientConfig) (snowflakeTransport, error) {
|
func (d *SnowflakeDialer) newSnowflakeClient(
|
||||||
|
config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
if d.newClientTransport != nil {
|
if d.newClientTransport != nil {
|
||||||
return d.newClientTransport(config)
|
return d.newClientTransport(config)
|
||||||
}
|
}
|
||||||
return sflib.NewSnowflakeClient(config)
|
return sflib.NewSnowflakeClient(config)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ampCacheURL returns a suitable AMP cache URL.
|
|
||||||
func (d *SnowflakeDialer) ampCacheURL() string {
|
|
||||||
// I tried using the following AMP cache and always got:
|
|
||||||
//
|
|
||||||
// 2022/01/19 16:51:28 AMP cache rendezvous response: 500 Internal Server Error
|
|
||||||
//
|
|
||||||
// So I disabled the AMP cache until we figure it out.
|
|
||||||
//
|
|
||||||
//return "https://cdn.ampproject.org/"
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
|
|
||||||
// brokerURL returns a suitable broker URL.
|
|
||||||
func (d *SnowflakeDialer) brokerURL() string {
|
|
||||||
return "https://snowflake-broker.torproject.net.global.prod.fastly.net/"
|
|
||||||
}
|
|
||||||
|
|
||||||
// frontDomain returns a suitable front domain.
|
|
||||||
func (d *SnowflakeDialer) frontDomain() string {
|
|
||||||
return "cdn.sstatic.net"
|
|
||||||
}
|
|
||||||
|
|
||||||
// iceAddresses returns suitable ICE addresses.
|
// iceAddresses returns suitable ICE addresses.
|
||||||
func (d *SnowflakeDialer) iceAddresses() []string {
|
func (d *SnowflakeDialer) iceAddresses() []string {
|
||||||
return stuninput.AsSnowflakeInput()
|
return stuninput.AsSnowflakeInput()
|
||||||
|
|
|
@ -11,12 +11,107 @@ import (
|
||||||
"github.com/ooni/probe-cli/v3/internal/model/mocks"
|
"github.com/ooni/probe-cli/v3/internal/model/mocks"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestSnowflakeMethodDomainFronting(t *testing.T) {
|
||||||
|
meth := NewSnowflakeRendezvousMethodDomainFronting()
|
||||||
|
if meth.AMPCacheURL() != "" {
|
||||||
|
t.Fatal("invalid amp cache URL")
|
||||||
|
}
|
||||||
|
const brokerURL = "https://snowflake-broker.torproject.net.global.prod.fastly.net/"
|
||||||
|
if meth.BrokerURL() != brokerURL {
|
||||||
|
t.Fatal("invalid broker URL")
|
||||||
|
}
|
||||||
|
const frontDomain = "cdn.sstatic.net"
|
||||||
|
if meth.FrontDomain() != frontDomain {
|
||||||
|
t.Fatal("invalid front domain")
|
||||||
|
}
|
||||||
|
if meth.Name() != "domain_fronting" {
|
||||||
|
t.Fatal("invalid name")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSnowflakeMethodAMP(t *testing.T) {
|
||||||
|
meth := NewSnowflakeRendezvousMethodAMP()
|
||||||
|
const ampCacheURL = "https://cdn.ampproject.org/"
|
||||||
|
if meth.AMPCacheURL() != ampCacheURL {
|
||||||
|
t.Fatal("invalid amp cache URL")
|
||||||
|
}
|
||||||
|
const brokerURL = "https://snowflake-broker.torproject.net/"
|
||||||
|
if meth.BrokerURL() != brokerURL {
|
||||||
|
t.Fatal("invalid broker URL")
|
||||||
|
}
|
||||||
|
const frontDomain = "www.google.com"
|
||||||
|
if meth.FrontDomain() != frontDomain {
|
||||||
|
t.Fatal("invalid front domain")
|
||||||
|
}
|
||||||
|
if meth.Name() != "amp" {
|
||||||
|
t.Fatal("invalid name")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewSnowflakeRendezvousMethod(t *testing.T) {
|
||||||
|
t.Run("for domain_fronted", func(t *testing.T) {
|
||||||
|
meth, err := NewSnowflakeRendezvousMethod("domain_fronting")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if _, ok := meth.(*snowflakeRendezvousMethodDomainFronting); !ok {
|
||||||
|
t.Fatal("unexpected method type")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("for empty string", func(t *testing.T) {
|
||||||
|
meth, err := NewSnowflakeRendezvousMethod("")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if _, ok := meth.(*snowflakeRendezvousMethodDomainFronting); !ok {
|
||||||
|
t.Fatal("unexpected method type")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("for amp", func(t *testing.T) {
|
||||||
|
meth, err := NewSnowflakeRendezvousMethod("amp")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
if _, ok := meth.(*snowflakeRendezvousMethodAMP); !ok {
|
||||||
|
t.Fatal("unexpected method type")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("for another value", func(t *testing.T) {
|
||||||
|
meth, err := NewSnowflakeRendezvousMethod("amptani")
|
||||||
|
if !errors.Is(err, ErrSnowflakeNoSuchRendezvousMethod) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
if meth != nil {
|
||||||
|
t.Fatal("unexpected method value")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewSnowflakeDialer(t *testing.T) {
|
||||||
|
dialer := NewSnowflakeDialer()
|
||||||
|
_, ok := dialer.RendezvousMethod.(*snowflakeRendezvousMethodDomainFronting)
|
||||||
|
if !ok {
|
||||||
|
t.Fatal("invalid rendezvous method type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestNewSnowflakeDialerWithRendezvousMethod(t *testing.T) {
|
||||||
|
meth := NewSnowflakeRendezvousMethodAMP()
|
||||||
|
dialer := NewSnowflakeDialerWithRendezvousMethod(meth)
|
||||||
|
if meth != dialer.RendezvousMethod {
|
||||||
|
t.Fatal("invalid rendezvous method value")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestSnowflakeDialerWorks(t *testing.T) {
|
func TestSnowflakeDialerWorks(t *testing.T) {
|
||||||
// This test may sadly run for a very long time (~10s)
|
// This test may sadly run for a very long time (~10s)
|
||||||
if testing.Short() {
|
if testing.Short() {
|
||||||
t.Skip("skip test in short mode")
|
t.Skip("skip test in short mode")
|
||||||
}
|
}
|
||||||
sfd := &SnowflakeDialer{}
|
sfd := NewSnowflakeDialer()
|
||||||
conn, err := sfd.DialContext(context.Background())
|
conn, err := sfd.DialContext(context.Background())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
|
@ -48,6 +143,7 @@ var _ snowflakeTransport = &mockableSnowflakeTransport{}
|
||||||
|
|
||||||
func TestSnowflakeDialerWorksWithMocks(t *testing.T) {
|
func TestSnowflakeDialerWorksWithMocks(t *testing.T) {
|
||||||
sfd := &SnowflakeDialer{
|
sfd := &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
return &mockableSnowflakeTransport{
|
return &mockableSnowflakeTransport{
|
||||||
MockDial: func() (net.Conn, error) {
|
MockDial: func() (net.Conn, error) {
|
||||||
|
@ -80,6 +176,7 @@ func TestSnowflakeDialerWorksWithMocks(t *testing.T) {
|
||||||
func TestSnowflakeDialerCannotCreateTransport(t *testing.T) {
|
func TestSnowflakeDialerCannotCreateTransport(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
sfd := &SnowflakeDialer{
|
sfd := &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
return nil, expected
|
return nil, expected
|
||||||
},
|
},
|
||||||
|
@ -96,6 +193,7 @@ func TestSnowflakeDialerCannotCreateTransport(t *testing.T) {
|
||||||
func TestSnowflakeDialerCannotCreateConnWithNoContextExpiration(t *testing.T) {
|
func TestSnowflakeDialerCannotCreateConnWithNoContextExpiration(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
sfd := &SnowflakeDialer{
|
sfd := &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
return &mockableSnowflakeTransport{
|
return &mockableSnowflakeTransport{
|
||||||
MockDial: func() (net.Conn, error) {
|
MockDial: func() (net.Conn, error) {
|
||||||
|
@ -118,6 +216,7 @@ func TestSnowflakeDialerCannotCreateConnWithContextExpiration(t *testing.T) {
|
||||||
defer cancel()
|
defer cancel()
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
sfd := &SnowflakeDialer{
|
sfd := &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
return &mockableSnowflakeTransport{
|
return &mockableSnowflakeTransport{
|
||||||
MockDial: func() (net.Conn, error) {
|
MockDial: func() (net.Conn, error) {
|
||||||
|
@ -141,6 +240,7 @@ func TestSnowflakeDialerWorksWithWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
defer cancel()
|
defer cancel()
|
||||||
sfd := &SnowflakeDialer{
|
sfd := &SnowflakeDialer{
|
||||||
|
RendezvousMethod: NewSnowflakeRendezvousMethodDomainFronting(),
|
||||||
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
newClientTransport: func(config sflib.ClientConfig) (snowflakeTransport, error) {
|
||||||
return &mockableSnowflakeTransport{
|
return &mockableSnowflakeTransport{
|
||||||
MockDial: func() (net.Conn, error) {
|
MockDial: func() (net.Conn, error) {
|
||||||
|
|
|
@ -2,6 +2,7 @@ package tunnel
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"net"
|
"net"
|
||||||
"os"
|
"os"
|
||||||
|
|
||||||
|
@ -66,6 +67,9 @@ type Config struct {
|
||||||
// testTorStart allows us to mock tor.Start.
|
// testTorStart allows us to mock tor.Start.
|
||||||
testTorStart func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error)
|
testTorStart func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error)
|
||||||
|
|
||||||
|
// testTorProtocolInfo allows us to mock getting protocol info.
|
||||||
|
testTorProtocolInfo func(tor *tor.Tor) (*control.ProtocolInfo, error)
|
||||||
|
|
||||||
// testTorEnableNetwork allows us to fake a failure when
|
// testTorEnableNetwork allows us to fake a failure when
|
||||||
// telling to the tor daemon to enable the network.
|
// telling to the tor daemon to enable the network.
|
||||||
testTorEnableNetwork func(ctx context.Context, tor *tor.Tor, wait bool) error
|
testTorEnableNetwork func(ctx context.Context, tor *tor.Tor, wait bool) error
|
||||||
|
@ -163,6 +167,21 @@ func (c *Config) torStart(ctx context.Context, conf *tor.StartConf) (*tor.Tor, e
|
||||||
return tor.Start(ctx, conf)
|
return tor.Start(ctx, conf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// errNoTorControl indicate you passed us a tor with a nil control field.
|
||||||
|
var errNoTorControl = errors.New("tunnel: no tor control")
|
||||||
|
|
||||||
|
// torProtocolInfo calls either testTorProtocolInfo or the
|
||||||
|
// proper function to get back protocol information.
|
||||||
|
func (c *Config) torProtocolInfo(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
if c.testTorProtocolInfo != nil {
|
||||||
|
return c.testTorProtocolInfo(tor)
|
||||||
|
}
|
||||||
|
if tor.Control == nil {
|
||||||
|
return nil, errNoTorControl
|
||||||
|
}
|
||||||
|
return tor.Control.ProtocolInfo()
|
||||||
|
}
|
||||||
|
|
||||||
// torEnableNetwork calls either testTorEnableNetwork or tor.EnableNetwork.
|
// torEnableNetwork calls either testTorEnableNetwork or tor.EnableNetwork.
|
||||||
func (c *Config) torEnableNetwork(ctx context.Context, tor *tor.Tor, wait bool) error {
|
func (c *Config) torEnableNetwork(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
if c.testTorEnableNetwork != nil {
|
if c.testTorEnableNetwork != nil {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/apex/log"
|
"github.com/apex/log"
|
||||||
|
"github.com/cretz/bine/tor"
|
||||||
"github.com/ooni/probe-cli/v3/internal/model"
|
"github.com/ooni/probe-cli/v3/internal/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -108,3 +109,16 @@ func TestConfigTorBinary(t *testing.T) {
|
||||||
verifyExpectations(t, config, expected, nil)
|
verifyExpectations(t, config, expected, nil)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestConfigTorProtocolInfo(t *testing.T) {
|
||||||
|
t.Run("with nil Control field", func(t *testing.T) {
|
||||||
|
config := &Config{}
|
||||||
|
protocolInfo, err := config.torProtocolInfo(&tor.Tor{})
|
||||||
|
if !errors.Is(err, errNoTorControl) {
|
||||||
|
t.Fatal("unexpected error", err)
|
||||||
|
}
|
||||||
|
if protocolInfo != nil {
|
||||||
|
t.Fatal("expected nil protocol info")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -39,7 +39,7 @@ func (t *fakeTunnel) SOCKS5ProxyURL() *url.URL {
|
||||||
}
|
}
|
||||||
|
|
||||||
// fakeStart starts the fake tunnel.
|
// fakeStart starts the fake tunnel.
|
||||||
func fakeStart(ctx context.Context, config *Config) (Tunnel, error) {
|
func fakeStart(ctx context.Context, config *Config) (Tunnel, DebugInfo, error) {
|
||||||
// do the same things other tunnels do:
|
// do the same things other tunnels do:
|
||||||
//
|
//
|
||||||
// 1. abort if context is cancelled
|
// 1. abort if context is cancelled
|
||||||
|
@ -50,25 +50,30 @@ func fakeStart(ctx context.Context, config *Config) (Tunnel, error) {
|
||||||
//
|
//
|
||||||
// after that, it's all fake and we just create a simple
|
// after that, it's all fake and we just create a simple
|
||||||
// socks5 server that we can use
|
// socks5 server that we can use
|
||||||
|
debugInfo := DebugInfo{
|
||||||
|
LogFilePath: "",
|
||||||
|
Name: "fake",
|
||||||
|
Version: "",
|
||||||
|
}
|
||||||
select {
|
select {
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
return nil, ctx.Err() // simplifies unit testing this code
|
return nil, debugInfo, ctx.Err() // simplifies unit testing this code
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
if config.TunnelDir == "" {
|
if config.TunnelDir == "" {
|
||||||
return nil, ErrEmptyTunnelDir
|
return nil, debugInfo, ErrEmptyTunnelDir
|
||||||
}
|
}
|
||||||
if err := config.mkdirAll(config.TunnelDir, 0700); err != nil {
|
if err := config.mkdirAll(config.TunnelDir, 0700); err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
server, err := config.socks5New(&socks5.Config{})
|
server, err := config.socks5New(&socks5.Config{})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
listener, err := config.netListen("tcp", "127.0.0.1:0")
|
listener, err := config.netListen("tcp", "127.0.0.1:0")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
bootstrapTime := time.Since(start)
|
bootstrapTime := time.Since(start)
|
||||||
go server.Serve(listener)
|
go server.Serve(listener)
|
||||||
|
@ -76,5 +81,5 @@ func fakeStart(ctx context.Context, config *Config) (Tunnel, error) {
|
||||||
addr: listener.Addr(),
|
addr: listener.Addr(),
|
||||||
bootstrapTime: bootstrapTime,
|
bootstrapTime: bootstrapTime,
|
||||||
listener: listener,
|
listener: listener,
|
||||||
}, nil
|
}, debugInfo, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ func TestFakeStartStop(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
tunnel, err := tunnel.Start(context.Background(), &tunnel.Config{
|
tunnel, _, err := tunnel.Start(context.Background(), &tunnel.Config{
|
||||||
Name: "fake",
|
Name: "fake",
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: tunnelDir,
|
TunnelDir: tunnelDir,
|
||||||
|
|
|
@ -14,7 +14,7 @@ func TestFakeWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // immediately fail
|
cancel() // immediately fail
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := fakeStart(ctx, &Config{
|
tunnel, _, err := fakeStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
})
|
})
|
||||||
|
@ -29,7 +29,7 @@ func TestFakeWithCancelledContext(t *testing.T) {
|
||||||
func TestFakeWithEmptyTunnelDir(t *testing.T) {
|
func TestFakeWithEmptyTunnelDir(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := fakeStart(ctx, &Config{
|
tunnel, _, err := fakeStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "",
|
TunnelDir: "",
|
||||||
})
|
})
|
||||||
|
@ -45,7 +45,7 @@ func TestFakeWithFailingMkdirAll(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := fakeStart(ctx, &Config{
|
tunnel, _, err := fakeStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testMkdirAll: func(dir string, mode os.FileMode) error {
|
testMkdirAll: func(dir string, mode os.FileMode) error {
|
||||||
|
@ -64,7 +64,7 @@ func TestFakeSocks5NewFails(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := fakeStart(ctx, &Config{
|
tunnel, _, err := fakeStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testSocks5New: func(conf *socks5.Config) (*socks5.Server, error) {
|
testSocks5New: func(conf *socks5.Config) (*socks5.Server, error) {
|
||||||
|
@ -83,7 +83,7 @@ func TestFakeNetListenFails(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := fakeStart(ctx, &Config{
|
tunnel, _, err := fakeStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testNetListen: func(network, address string) (net.Listener, error) {
|
testNetListen: func(network, address string) (net.Listener, error) {
|
||||||
|
|
37
internal/tunnel/mocks/mocks.go
Normal file
37
internal/tunnel/mocks/mocks.go
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
// Package mocks contains mocks for tunnel.
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/url"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/tunnel"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Tunnel allows mocking a tunnel.
|
||||||
|
type Tunnel struct {
|
||||||
|
// MockBootstrapTime allows to mock BootstrapTime.
|
||||||
|
MockBootstrapTime func() time.Duration
|
||||||
|
|
||||||
|
// MockSOCKS5ProxyURL allows to mock Socks5ProxyURL.
|
||||||
|
MockSOCKS5ProxyURL func() *url.URL
|
||||||
|
|
||||||
|
// MockStop allows to mock Stop.
|
||||||
|
MockStop func()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *Tunnel) BootstrapTime() time.Duration {
|
||||||
|
return t.MockBootstrapTime()
|
||||||
|
}
|
||||||
|
|
||||||
|
// SOCKS5ProxyURL implements Tunnel.SOCKS5ProxyURL.
|
||||||
|
func (t *Tunnel) SOCKS5ProxyURL() *url.URL {
|
||||||
|
return t.MockSOCKS5ProxyURL()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Stop implements Tunnel.Stop.
|
||||||
|
func (t *Tunnel) Stop() {
|
||||||
|
t.MockStop()
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ tunnel.Tunnel = &Tunnel{}
|
60
internal/tunnel/mocks/mocks_test.go
Normal file
60
internal/tunnel/mocks/mocks_test.go
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/url"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/google/go-cmp/cmp"
|
||||||
|
"github.com/ooni/probe-cli/v3/internal/atomicx"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestTunnel(t *testing.T) {
|
||||||
|
t.Run("BootstrapTime", func(t *testing.T) {
|
||||||
|
var expected time.Duration = 114
|
||||||
|
tun := &Tunnel{
|
||||||
|
MockBootstrapTime: func() time.Duration {
|
||||||
|
return expected
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if tun.BootstrapTime() != expected {
|
||||||
|
t.Fatal("invalid BootstrapTime")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("SOCKS5ProxyURL", func(t *testing.T) {
|
||||||
|
expected := &url.URL{
|
||||||
|
Scheme: "https",
|
||||||
|
Opaque: "",
|
||||||
|
User: &url.Userinfo{},
|
||||||
|
Host: "www.google.com",
|
||||||
|
Path: "/robots.txt",
|
||||||
|
RawPath: "",
|
||||||
|
ForceQuery: false,
|
||||||
|
RawQuery: "",
|
||||||
|
Fragment: "",
|
||||||
|
RawFragment: "",
|
||||||
|
}
|
||||||
|
tun := &Tunnel{
|
||||||
|
MockSOCKS5ProxyURL: func() *url.URL {
|
||||||
|
return expected
|
||||||
|
},
|
||||||
|
}
|
||||||
|
if diff := cmp.Diff(expected.String(), tun.SOCKS5ProxyURL().String()); diff != "" {
|
||||||
|
t.Fatal(diff)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("Stop", func(t *testing.T) {
|
||||||
|
called := &atomicx.Int64{}
|
||||||
|
tun := &Tunnel{
|
||||||
|
MockStop: func() {
|
||||||
|
called.Add(1)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
tun.Stop()
|
||||||
|
if called.Load() != 1 {
|
||||||
|
t.Fatal("not called")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
|
@ -30,30 +30,38 @@ func psiphonMakeWorkingDir(config *Config) (string, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// psiphonStart starts the psiphon tunnel.
|
// psiphonStart starts the psiphon tunnel.
|
||||||
func psiphonStart(ctx context.Context, config *Config) (Tunnel, error) {
|
func psiphonStart(ctx context.Context, config *Config) (Tunnel, DebugInfo, error) {
|
||||||
|
debugInfo := DebugInfo{
|
||||||
|
LogFilePath: "",
|
||||||
|
Name: "psiphon",
|
||||||
|
Version: "",
|
||||||
|
}
|
||||||
select {
|
select {
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
return nil, ctx.Err() // simplifies unit testing this code
|
return nil, debugInfo, ctx.Err() // simplifies unit testing this code
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
if config.TunnelDir == "" {
|
if config.TunnelDir == "" {
|
||||||
return nil, ErrEmptyTunnelDir
|
return nil, debugInfo, ErrEmptyTunnelDir
|
||||||
}
|
}
|
||||||
configJSON, err := config.Session.FetchPsiphonConfig(ctx)
|
configJSON, err := config.Session.FetchPsiphonConfig(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
workdir, err := psiphonMakeWorkingDir(config)
|
workdir, err := psiphonMakeWorkingDir(config)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
tunnel, err := config.startPsiphon(ctx, configJSON, workdir)
|
tunnel, err := config.startPsiphon(ctx, configJSON, workdir)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
stop := time.Now()
|
stop := time.Now()
|
||||||
return &psiphonTunnel{tunnel: tunnel, bootstrapTime: stop.Sub(start)}, nil
|
return &psiphonTunnel{
|
||||||
|
tunnel: tunnel,
|
||||||
|
bootstrapTime: stop.Sub(start),
|
||||||
|
}, debugInfo, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stop is an idempotent method that shuts down the tunnel
|
// Stop is an idempotent method that shuts down the tunnel
|
||||||
|
|
|
@ -28,7 +28,7 @@ func TestPsiphonStartStop(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
tunnel, err := tunnel.Start(context.Background(), &tunnel.Config{
|
tunnel, _, err := tunnel.Start(context.Background(), &tunnel.Config{
|
||||||
Name: "psiphon",
|
Name: "psiphon",
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: tunnelDir,
|
TunnelDir: tunnelDir,
|
||||||
|
|
|
@ -13,7 +13,7 @@ func TestPsiphonWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // immediately fail
|
cancel() // immediately fail
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := psiphonStart(ctx, &Config{
|
tunnel, _, err := psiphonStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
})
|
})
|
||||||
|
@ -28,7 +28,7 @@ func TestPsiphonWithCancelledContext(t *testing.T) {
|
||||||
func TestPsiphonWithEmptyTunnelDir(t *testing.T) {
|
func TestPsiphonWithEmptyTunnelDir(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
sess := &MockableSession{}
|
sess := &MockableSession{}
|
||||||
tunnel, err := psiphonStart(ctx, &Config{
|
tunnel, _, err := psiphonStart(ctx, &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "",
|
TunnelDir: "",
|
||||||
})
|
})
|
||||||
|
@ -45,7 +45,7 @@ func TestPsiphonFetchPsiphonConfigFailure(t *testing.T) {
|
||||||
sess := &MockableSession{
|
sess := &MockableSession{
|
||||||
Err: expected,
|
Err: expected,
|
||||||
}
|
}
|
||||||
tunnel, err := psiphonStart(context.Background(), &Config{
|
tunnel, _, err := psiphonStart(context.Background(), &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
})
|
})
|
||||||
|
@ -62,7 +62,7 @@ func TestPsiphonMkdirAllFailure(t *testing.T) {
|
||||||
sess := &MockableSession{
|
sess := &MockableSession{
|
||||||
Result: []byte(`{}`),
|
Result: []byte(`{}`),
|
||||||
}
|
}
|
||||||
tunnel, err := psiphonStart(context.Background(), &Config{
|
tunnel, _, err := psiphonStart(context.Background(), &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testMkdirAll: func(path string, perm os.FileMode) error {
|
testMkdirAll: func(path string, perm os.FileMode) error {
|
||||||
|
@ -82,7 +82,7 @@ func TestPsiphonStartFailure(t *testing.T) {
|
||||||
sess := &MockableSession{
|
sess := &MockableSession{
|
||||||
Result: []byte(`{}`),
|
Result: []byte(`{}`),
|
||||||
}
|
}
|
||||||
tunnel, err := psiphonStart(context.Background(), &Config{
|
tunnel, _, err := psiphonStart(context.Background(), &Config{
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testStartPsiphon: func(ctx context.Context, config []byte,
|
testStartPsiphon: func(ctx context.Context, config []byte,
|
||||||
|
|
|
@ -55,17 +55,23 @@ var ErrTorReturnedUnsupportedProxy = errors.New(
|
||||||
"tor returned unsupported proxy")
|
"tor returned unsupported proxy")
|
||||||
|
|
||||||
// torStart starts the tor tunnel.
|
// torStart starts the tor tunnel.
|
||||||
func torStart(ctx context.Context, config *Config) (Tunnel, error) {
|
func torStart(ctx context.Context, config *Config) (Tunnel, DebugInfo, error) {
|
||||||
|
debugInfo := DebugInfo{
|
||||||
|
LogFilePath: "",
|
||||||
|
Name: "tor",
|
||||||
|
Version: "",
|
||||||
|
}
|
||||||
select {
|
select {
|
||||||
case <-ctx.Done():
|
case <-ctx.Done():
|
||||||
return nil, ctx.Err() // allows to write unit tests using this code
|
return nil, debugInfo, ctx.Err() // allows to write unit tests using this code
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
if config.TunnelDir == "" {
|
if config.TunnelDir == "" {
|
||||||
return nil, ErrEmptyTunnelDir
|
return nil, debugInfo, ErrEmptyTunnelDir
|
||||||
}
|
}
|
||||||
stateDir := filepath.Join(config.TunnelDir, "tor")
|
stateDir := filepath.Join(config.TunnelDir, "tor")
|
||||||
logfile := filepath.Join(stateDir, "tor.log")
|
logfile := filepath.Join(stateDir, "tor.log")
|
||||||
|
debugInfo.LogFilePath = logfile
|
||||||
maybeCleanupTunnelDir(stateDir, logfile)
|
maybeCleanupTunnelDir(stateDir, logfile)
|
||||||
extraArgs := append([]string{}, config.TorArgs...)
|
extraArgs := append([]string{}, config.TorArgs...)
|
||||||
extraArgs = append(extraArgs, "Log")
|
extraArgs = append(extraArgs, "Log")
|
||||||
|
@ -74,39 +80,44 @@ func torStart(ctx context.Context, config *Config) (Tunnel, error) {
|
||||||
extraArgs = append(extraArgs, fmt.Sprintf(`notice file %s`, logfile))
|
extraArgs = append(extraArgs, fmt.Sprintf(`notice file %s`, logfile))
|
||||||
torStartConf, err := getTorStartConf(config, stateDir, extraArgs)
|
torStartConf, err := getTorStartConf(config, stateDir, extraArgs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
instance, err := config.torStart(ctx, torStartConf)
|
instance, err := config.torStart(ctx, torStartConf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
|
protoInfo, err := config.torProtocolInfo(instance)
|
||||||
|
if err != nil {
|
||||||
|
return nil, debugInfo, err
|
||||||
|
}
|
||||||
|
debugInfo.Version = protoInfo.TorVersion
|
||||||
instance.StopProcessOnClose = true
|
instance.StopProcessOnClose = true
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
if err := config.torEnableNetwork(ctx, instance, true); err != nil {
|
if err := config.torEnableNetwork(ctx, instance, true); err != nil {
|
||||||
instance.Close()
|
instance.Close()
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
stop := time.Now()
|
stop := time.Now()
|
||||||
// Adapted from <https://git.io/Jfc7N>
|
// Adapted from <https://git.io/Jfc7N>
|
||||||
info, err := config.torGetInfo(instance.Control, "net/listeners/socks")
|
info, err := config.torGetInfo(instance.Control, "net/listeners/socks")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
instance.Close()
|
instance.Close()
|
||||||
return nil, err
|
return nil, debugInfo, err
|
||||||
}
|
}
|
||||||
if len(info) != 1 || info[0].Key != "net/listeners/socks" {
|
if len(info) != 1 || info[0].Key != "net/listeners/socks" {
|
||||||
instance.Close()
|
instance.Close()
|
||||||
return nil, ErrTorUnableToGetSOCKSProxyAddress
|
return nil, debugInfo, ErrTorUnableToGetSOCKSProxyAddress
|
||||||
}
|
}
|
||||||
proxyAddress := info[0].Val
|
proxyAddress := info[0].Val
|
||||||
if strings.HasPrefix(proxyAddress, "unix:") {
|
if strings.HasPrefix(proxyAddress, "unix:") {
|
||||||
instance.Close()
|
instance.Close()
|
||||||
return nil, ErrTorReturnedUnsupportedProxy
|
return nil, debugInfo, ErrTorReturnedUnsupportedProxy
|
||||||
}
|
}
|
||||||
return &torTunnel{
|
return &torTunnel{
|
||||||
bootstrapTime: stop.Sub(start),
|
bootstrapTime: stop.Sub(start),
|
||||||
instance: instance,
|
instance: instance,
|
||||||
proxy: &url.URL{Scheme: "socks5", Host: proxyAddress},
|
proxy: &url.URL{Scheme: "socks5", Host: proxyAddress},
|
||||||
}, nil
|
}, debugInfo, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// maybeCleanupTunnelDir removes stale files inside
|
// maybeCleanupTunnelDir removes stale files inside
|
||||||
|
|
|
@ -33,7 +33,7 @@ func TestTorStartStop(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
tunnel, err := tunnel.Start(context.Background(), &tunnel.Config{
|
tunnel, _, err := tunnel.Start(context.Background(), &tunnel.Config{
|
||||||
Name: "tor",
|
Name: "tor",
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TorBinary: torBinaryPath,
|
TorBinary: torBinaryPath,
|
||||||
|
|
|
@ -50,7 +50,7 @@ func TestTorTunnelNonNil(t *testing.T) {
|
||||||
func TestTorWithCancelledContext(t *testing.T) {
|
func TestTorWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // fail immediately
|
cancel() // fail immediately
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
})
|
})
|
||||||
|
@ -64,7 +64,7 @@ func TestTorWithCancelledContext(t *testing.T) {
|
||||||
|
|
||||||
func TestTorWithEmptyTunnelDir(t *testing.T) {
|
func TestTorWithEmptyTunnelDir(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "",
|
TunnelDir: "",
|
||||||
})
|
})
|
||||||
|
@ -78,7 +78,7 @@ func TestTorWithEmptyTunnelDir(t *testing.T) {
|
||||||
|
|
||||||
func TestTorBinaryNotFoundFailure(t *testing.T) {
|
func TestTorBinaryNotFoundFailure(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TorBinary: "/nonexistent/directory/tor",
|
TorBinary: "/nonexistent/directory/tor",
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
|
@ -94,7 +94,7 @@ func TestTorBinaryNotFoundFailure(t *testing.T) {
|
||||||
func TestTorStartFailure(t *testing.T) {
|
func TestTorStartFailure(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -112,10 +112,10 @@ func TestTorStartFailure(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestTorEnableNetworkFailure(t *testing.T) {
|
func TestTorGetProtocolInfoFailure(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -124,6 +124,33 @@ func TestTorEnableNetworkFailure(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return nil, expected
|
||||||
|
},
|
||||||
|
})
|
||||||
|
if !errors.Is(err, expected) {
|
||||||
|
t.Fatal("not the error we expected")
|
||||||
|
}
|
||||||
|
if tun != nil {
|
||||||
|
t.Fatal("expected nil tunnel here")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTorEnableNetworkFailure(t *testing.T) {
|
||||||
|
expected := errors.New("mocked error")
|
||||||
|
ctx := context.Background()
|
||||||
|
tun, _, err := torStart(ctx, &Config{
|
||||||
|
Session: &MockableSession{},
|
||||||
|
TunnelDir: "testdata",
|
||||||
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
return "/usr/local/bin/tor", nil
|
||||||
|
},
|
||||||
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
|
return &tor.Tor{}, nil
|
||||||
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return expected
|
return expected
|
||||||
},
|
},
|
||||||
|
@ -139,7 +166,7 @@ func TestTorEnableNetworkFailure(t *testing.T) {
|
||||||
func TestTorGetInfoFailure(t *testing.T) {
|
func TestTorGetInfoFailure(t *testing.T) {
|
||||||
expected := errors.New("mocked error")
|
expected := errors.New("mocked error")
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -148,6 +175,9 @@ func TestTorGetInfoFailure(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
@ -165,7 +195,7 @@ func TestTorGetInfoFailure(t *testing.T) {
|
||||||
|
|
||||||
func TestTorGetInfoInvalidNumberOfKeys(t *testing.T) {
|
func TestTorGetInfoInvalidNumberOfKeys(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -174,6 +204,9 @@ func TestTorGetInfoInvalidNumberOfKeys(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
@ -191,7 +224,7 @@ func TestTorGetInfoInvalidNumberOfKeys(t *testing.T) {
|
||||||
|
|
||||||
func TestTorGetInfoInvalidKey(t *testing.T) {
|
func TestTorGetInfoInvalidKey(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -200,6 +233,9 @@ func TestTorGetInfoInvalidKey(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
@ -217,7 +253,7 @@ func TestTorGetInfoInvalidKey(t *testing.T) {
|
||||||
|
|
||||||
func TestTorGetInfoInvalidProxyType(t *testing.T) {
|
func TestTorGetInfoInvalidProxyType(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -226,6 +262,9 @@ func TestTorGetInfoInvalidProxyType(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
@ -243,7 +282,7 @@ func TestTorGetInfoInvalidProxyType(t *testing.T) {
|
||||||
|
|
||||||
func TestTorUnsupportedProxy(t *testing.T) {
|
func TestTorUnsupportedProxy(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := torStart(ctx, &Config{
|
tun, _, err := torStart(ctx, &Config{
|
||||||
Session: &MockableSession{},
|
Session: &MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
testExecabsLookPath: func(name string) (string, error) {
|
testExecabsLookPath: func(name string) (string, error) {
|
||||||
|
@ -252,6 +291,9 @@ func TestTorUnsupportedProxy(t *testing.T) {
|
||||||
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
testTorStart: func(ctx context.Context, conf *tor.StartConf) (*tor.Tor, error) {
|
||||||
return &tor.Tor{}, nil
|
return &tor.Tor{}, nil
|
||||||
},
|
},
|
||||||
|
testTorProtocolInfo: func(tor *tor.Tor) (*control.ProtocolInfo, error) {
|
||||||
|
return &control.ProtocolInfo{}, nil
|
||||||
|
},
|
||||||
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
testTorEnableNetwork: func(ctx context.Context, tor *tor.Tor, wait bool) error {
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
|
|
|
@ -74,6 +74,22 @@ var ErrEmptyTunnelDir = errors.New("TunnelDir is empty")
|
||||||
// is not supported by this package.
|
// is not supported by this package.
|
||||||
var ErrUnsupportedTunnelName = errors.New("unsupported tunnel name")
|
var ErrUnsupportedTunnelName = errors.New("unsupported tunnel name")
|
||||||
|
|
||||||
|
// DebugInfo contains information useful to debug issues
|
||||||
|
// when starting up a given tunnel fails.
|
||||||
|
type DebugInfo struct {
|
||||||
|
// LogFilePath is the path to the log file, which MAY
|
||||||
|
// be empty in case we don't have a log file.
|
||||||
|
LogFilePath string
|
||||||
|
|
||||||
|
// Name is the name of the tunnel and will always
|
||||||
|
// be properly set by the code.
|
||||||
|
Name string
|
||||||
|
|
||||||
|
// Version is the tunnel version. This field MAY be
|
||||||
|
// empty if we don't know the version.
|
||||||
|
Version string
|
||||||
|
}
|
||||||
|
|
||||||
// Start starts a new tunnel by name or returns an error. We currently
|
// Start starts a new tunnel by name or returns an error. We currently
|
||||||
// support the following tunnels:
|
// support the following tunnels:
|
||||||
//
|
//
|
||||||
|
@ -94,7 +110,15 @@ var ErrUnsupportedTunnelName = errors.New("unsupported tunnel name")
|
||||||
// The "fake" tunnel is a fake tunnel that just exposes a
|
// The "fake" tunnel is a fake tunnel that just exposes a
|
||||||
// SOCKS5 proxy and then connects directly to server. We use
|
// SOCKS5 proxy and then connects directly to server. We use
|
||||||
// this special kind of tunnel to implement tests.
|
// this special kind of tunnel to implement tests.
|
||||||
func Start(ctx context.Context, config *Config) (Tunnel, error) {
|
//
|
||||||
|
// The return value is a triple:
|
||||||
|
//
|
||||||
|
// 1. a valid Tunnel on success, nil on failure;
|
||||||
|
//
|
||||||
|
// 2. debugging information (both on success and failure);
|
||||||
|
//
|
||||||
|
// 3. nil on success, an error on failure.
|
||||||
|
func Start(ctx context.Context, config *Config) (Tunnel, DebugInfo, error) {
|
||||||
switch config.Name {
|
switch config.Name {
|
||||||
case "fake":
|
case "fake":
|
||||||
return fakeStart(ctx, config)
|
return fakeStart(ctx, config)
|
||||||
|
@ -103,6 +127,7 @@ func Start(ctx context.Context, config *Config) (Tunnel, error) {
|
||||||
case "tor":
|
case "tor":
|
||||||
return torStart(ctx, config)
|
return torStart(ctx, config)
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("%w: %s", ErrUnsupportedTunnelName, config.Name)
|
di := DebugInfo{}
|
||||||
|
return nil, di, fmt.Errorf("%w: %s", ErrUnsupportedTunnelName, config.Name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ import (
|
||||||
|
|
||||||
func TestStartNoTunnel(t *testing.T) {
|
func TestStartNoTunnel(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: "",
|
Name: "",
|
||||||
Session: &tunnel.MockableSession{},
|
Session: &tunnel.MockableSession{},
|
||||||
})
|
})
|
||||||
|
@ -25,7 +25,7 @@ func TestStartNoTunnel(t *testing.T) {
|
||||||
func TestStartPsiphonWithCancelledContext(t *testing.T) {
|
func TestStartPsiphonWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // fail immediately
|
cancel() // fail immediately
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: "psiphon",
|
Name: "psiphon",
|
||||||
Session: &tunnel.MockableSession{},
|
Session: &tunnel.MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
|
@ -41,7 +41,7 @@ func TestStartPsiphonWithCancelledContext(t *testing.T) {
|
||||||
func TestStartTorWithCancelledContext(t *testing.T) {
|
func TestStartTorWithCancelledContext(t *testing.T) {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
cancel() // fail immediately
|
cancel() // fail immediately
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: "tor",
|
Name: "tor",
|
||||||
Session: &tunnel.MockableSession{},
|
Session: &tunnel.MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
|
@ -56,7 +56,7 @@ func TestStartTorWithCancelledContext(t *testing.T) {
|
||||||
|
|
||||||
func TestStartInvalidTunnel(t *testing.T) {
|
func TestStartInvalidTunnel(t *testing.T) {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: "antani",
|
Name: "antani",
|
||||||
Session: &tunnel.MockableSession{},
|
Session: &tunnel.MockableSession{},
|
||||||
TunnelDir: "testdata",
|
TunnelDir: "testdata",
|
||||||
|
|
|
@ -76,11 +76,11 @@ As a first step, we create a dialer for snowflake using the
|
||||||
`ptx` package. This dialer will allow us to create a `net.Conn`-like
|
`ptx` package. This dialer will allow us to create a `net.Conn`-like
|
||||||
network connection where traffic is sent using the Snowflake
|
network connection where traffic is sent using the Snowflake
|
||||||
pluggable transport. There are several optional fields in
|
pluggable transport. There are several optional fields in
|
||||||
`SnowflakeDialer`, but we don't need to override the default
|
`SnowflakeDialer`; the `NewSnowflakeDialer` constructor will
|
||||||
values, so we can just use a default-initialized struct.
|
give us a suitable configured dialer with default settings.
|
||||||
|
|
||||||
```Go
|
```Go
|
||||||
sfdialer := &ptx.SnowflakeDialer{}
|
sfdialer := ptx.NewSnowflakeDialer()
|
||||||
```
|
```
|
||||||
|
|
||||||
Let us now create a listener. The `ptx.Listener` is a listener
|
Let us now create a listener. The `ptx.Listener` is a listener
|
||||||
|
|
|
@ -141,11 +141,11 @@ func (m *Measurer) run(ctx context.Context,
|
||||||
// `ptx` package. This dialer will allow us to create a `net.Conn`-like
|
// `ptx` package. This dialer will allow us to create a `net.Conn`-like
|
||||||
// network connection where traffic is sent using the Snowflake
|
// network connection where traffic is sent using the Snowflake
|
||||||
// pluggable transport. There are several optional fields in
|
// pluggable transport. There are several optional fields in
|
||||||
// `SnowflakeDialer`, but we don't need to override the default
|
// `SnowflakeDialer`; the `NewSnowflakeDialer` constructor will
|
||||||
// values, so we can just use a default-initialized struct.
|
// give us a suitable configured dialer with default settings.
|
||||||
//
|
//
|
||||||
// ```Go
|
// ```Go
|
||||||
sfdialer := &ptx.SnowflakeDialer{}
|
sfdialer := ptx.NewSnowflakeDialer()
|
||||||
// ```
|
// ```
|
||||||
//
|
//
|
||||||
// Let us now create a listener. The `ptx.Listener` is a listener
|
// Let us now create a listener. The `ptx.Listener` is a listener
|
||||||
|
@ -181,7 +181,7 @@ func (m *Measurer) run(ctx context.Context,
|
||||||
// pluggable transport created by `ptl` and `sfdialer`.
|
// pluggable transport created by `ptl` and `sfdialer`.
|
||||||
//
|
//
|
||||||
// ```Go
|
// ```Go
|
||||||
tun, err := tunnel.Start(ctx, &tunnel.Config{
|
tun, _, err := tunnel.Start(ctx, &tunnel.Config{
|
||||||
Name: "tor",
|
Name: "tor",
|
||||||
Session: sess,
|
Session: sess,
|
||||||
TunnelDir: path.Join(sess.TempDir(), "torsf"),
|
TunnelDir: path.Join(sess.TempDir(), "torsf"),
|
||||||
|
|
Loading…
Reference in New Issue
Block a user