2021-04-03 19:57:21 +02:00
|
|
|
package tunnel
|
2021-02-02 12:05:47 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"net/url"
|
|
|
|
"path/filepath"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ooni/psiphon/oopsi/github.com/Psiphon-Labs/psiphon-tunnel-core/ClientLibrary/clientlib"
|
|
|
|
)
|
|
|
|
|
2021-04-03 19:57:21 +02:00
|
|
|
// psiphonTunnel is a psiphon tunnel
|
|
|
|
type psiphonTunnel struct {
|
2021-04-03 21:25:08 +02:00
|
|
|
// tunnel is the underlying psiphon tunnel
|
|
|
|
tunnel *clientlib.PsiphonTunnel
|
|
|
|
|
|
|
|
// duration is the duration of the bootstrap
|
2021-02-02 12:05:47 +01:00
|
|
|
duration time.Duration
|
|
|
|
}
|
|
|
|
|
2021-04-03 21:34:19 +02:00
|
|
|
// TODO(bassosimone): _always_ wiping the state directory
|
|
|
|
// here is absolutely wrong. This prevents us from reusing
|
|
|
|
// an existing psiphon cache existing on disk. We want to
|
|
|
|
// delete the directory _only_ in the psiphon nettest.
|
|
|
|
|
|
|
|
// psiphonMakeWorkingDir creates the working directory
|
|
|
|
func psiphonMakeWorkingDir(config *Config) (string, error) {
|
2021-02-02 12:05:47 +01:00
|
|
|
const testdirname = "oonipsiphon"
|
2021-04-03 21:34:19 +02:00
|
|
|
baseWorkDir := config.WorkDir
|
|
|
|
if baseWorkDir == "" {
|
|
|
|
baseWorkDir = config.Session.TempDir()
|
|
|
|
}
|
|
|
|
workdir := filepath.Join(baseWorkDir, testdirname)
|
2021-04-03 21:09:34 +02:00
|
|
|
if err := config.removeAll(workdir); err != nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
return "", err
|
|
|
|
}
|
2021-04-03 21:09:34 +02:00
|
|
|
if err := config.mkdirAll(workdir, 0700); err != nil {
|
2021-02-02 12:05:47 +01:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return workdir, nil
|
|
|
|
}
|
|
|
|
|
2021-04-03 19:57:21 +02:00
|
|
|
// psiphonStart starts the psiphon tunnel.
|
2021-04-03 21:09:34 +02:00
|
|
|
func psiphonStart(ctx context.Context, config *Config) (Tunnel, error) {
|
2021-02-02 12:05:47 +01:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return nil, ctx.Err() // simplifies unit testing this code
|
|
|
|
default:
|
|
|
|
}
|
2021-04-03 21:09:34 +02:00
|
|
|
configJSON, err := config.Session.FetchPsiphonConfig(ctx)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-04-03 21:34:19 +02:00
|
|
|
workdir, err := psiphonMakeWorkingDir(config)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
start := time.Now()
|
2021-04-03 21:09:34 +02:00
|
|
|
tunnel, err := config.startPsiphon(ctx, configJSON, workdir)
|
2021-02-02 12:05:47 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
stop := time.Now()
|
2021-04-03 19:57:21 +02:00
|
|
|
return &psiphonTunnel{tunnel: tunnel, duration: stop.Sub(start)}, nil
|
2021-02-02 12:05:47 +01:00
|
|
|
}
|
|
|
|
|
2021-04-03 21:09:34 +02:00
|
|
|
// TODO(bassosimone): define the NullTunnel rather than relying on
|
|
|
|
// this magic that a nil psiphonTunnel works.
|
|
|
|
|
2021-02-02 12:05:47 +01:00
|
|
|
// Stop is an idempotent method that shuts down the tunnel
|
2021-04-03 19:57:21 +02:00
|
|
|
func (t *psiphonTunnel) Stop() {
|
2021-02-02 12:05:47 +01:00
|
|
|
if t != nil {
|
|
|
|
t.tunnel.Stop()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SOCKS5ProxyURL returns the SOCKS5 proxy URL.
|
2021-04-03 19:57:21 +02:00
|
|
|
func (t *psiphonTunnel) SOCKS5ProxyURL() (proxyURL *url.URL) {
|
2021-02-02 12:05:47 +01:00
|
|
|
if t != nil {
|
|
|
|
proxyURL = &url.URL{
|
|
|
|
Scheme: "socks5",
|
|
|
|
Host: net.JoinHostPort(
|
|
|
|
"127.0.0.1", fmt.Sprintf("%d", t.tunnel.SOCKSProxyPort)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// BootstrapTime returns the bootstrap time
|
2021-04-03 19:57:21 +02:00
|
|
|
func (t *psiphonTunnel) BootstrapTime() (duration time.Duration) {
|
2021-02-02 12:05:47 +01:00
|
|
|
if t != nil {
|
|
|
|
duration = t.duration
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|