From d7ed47d1efcc55a28de3f9eeee282ab1e246c6f1 Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Wed, 4 Sep 2024 18:52:57 -0400 Subject: [PATCH 1/7] Add DNS and TCP reports to test-connectivity --- x/examples/test-connectivity/main.go | 104 ++++++++++++++++++++++++--- 1 file changed, 95 insertions(+), 9 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index 45b4ea8d..beb233c5 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -24,6 +24,7 @@ import ( "log" "net" "net/http" + "net/http/httptrace" "net/url" "os" "path" @@ -31,6 +32,7 @@ import ( "time" "github.com/Jigsaw-Code/outline-sdk/dns" + "github.com/Jigsaw-Code/outline-sdk/transport" "github.com/Jigsaw-Code/outline-sdk/x/config" "github.com/Jigsaw-Code/outline-sdk/x/connectivity" "github.com/Jigsaw-Code/outline-sdk/x/report" @@ -41,6 +43,12 @@ var debugLog log.Logger = *log.New(io.Discard, "", 0) // var errorLog log.Logger = *log.New(os.Stderr, "[ERROR] ", log.LstdFlags|log.Lmicroseconds|log.Lshortfile) type connectivityReport struct { + Test testReport `json:"test"` + DNSQueries []dnsReport `json:"dns_queries,omitempty"` + TCPConnections []tcpReport `json:"tcp_connections,omitempty"` +} + +type testReport struct { // Inputs Resolver string `json:"resolver"` Proto string `json:"proto"` @@ -53,6 +61,19 @@ type connectivityReport struct { Error *errorJSON `json:"error"` } +type dnsReport struct { + QueryName string `json:"query_name"` + AnswerIPs []string `json:"answer_ips"` + Error string `json:"error"` +} + +type tcpReport struct { + Hostname string `json:"hostname"` + IP string `json:"ip"` + Port string `json:"port"` + Error string `json:"error"` +} + type errorJSON struct { // TODO: add Shadowsocks/Transport error Op string `json:"op,omitempty"` @@ -84,7 +105,7 @@ func unwrapAll(err error) error { } func (r connectivityReport) IsSuccess() bool { - if r.Error == nil { + if r.Test.Error == nil { return true } else { return false @@ -161,21 +182,82 @@ func main() { success := false jsonEncoder := json.NewEncoder(os.Stdout) jsonEncoder.SetEscapeHTML(false) - configToDialer := config.NewDefaultConfigToDialer() for _, resolverHost := range strings.Split(*resolverFlag, ",") { resolverHost := strings.TrimSpace(resolverHost) resolverAddress := net.JoinHostPort(resolverHost, "53") for _, proto := range strings.Split(*protoFlag, ",") { proto = strings.TrimSpace(proto) var resolver dns.Resolver + dnsReports := make([]dnsReport, 0) + tcpReports := make([]tcpReport, 0) switch proto { case "tcp": + configToDialer := config.NewDefaultConfigToDialer() + configToDialer.BaseStreamDialer = transport.FuncStreamDialer(func(ctx context.Context, addr string) (transport.StreamConn, error) { + hostname, _, err := net.SplitHostPort(addr) + if err != nil { + return nil, err + } + ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSDone: func(di httptrace.DNSDoneInfo) { + report := dnsReport{ + QueryName: hostname, + } + if di.Err != nil { + report.Error = di.Err.Error() + } + for _, ip := range di.Addrs { + report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) + } + dnsReports = append(dnsReports, report) + }, + ConnectDone: func(network, addr string, connErr error) { + ip, port, err := net.SplitHostPort(addr) + if err != nil { + return + } + report := tcpReport{ + Hostname: hostname, + IP: ip, + Port: port, + } + if connErr != nil { + report.Error = connErr.Error() + } + tcpReports = append(tcpReports, report) + }, + }) + return (&transport.TCPDialer{}).DialStream(ctx, addr) + }) streamDialer, err := configToDialer.NewStreamDialer(*transportFlag) if err != nil { log.Fatalf("Failed to create StreamDialer: %v", err) } resolver = dns.NewTCPResolver(streamDialer, resolverAddress) + case "udp": + configToDialer := config.NewDefaultConfigToDialer() + configToDialer.BasePacketDialer = transport.FuncPacketDialer(func(ctx context.Context, addr string) (net.Conn, error) { + hostname, _, err := net.SplitHostPort(addr) + if err != nil { + return nil, err + } + ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSDone: func(di httptrace.DNSDoneInfo) { + report := dnsReport{ + QueryName: hostname, + } + if di.Err != nil { + report.Error = di.Err.Error() + } + for _, ip := range di.Addrs { + report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) + } + dnsReports = append(dnsReports, report) + }, + }) + return (&transport.UDPDialer{}).DialPacket(ctx, addr) + }) packetDialer, err := configToDialer.NewPacketDialer(*transportFlag) if err != nil { log.Fatalf("Failed to create PacketDialer: %v", err) @@ -199,13 +281,17 @@ func main() { log.Fatalf("Failed to sanitize config: %v", err) } var r report.Report = connectivityReport{ - Resolver: resolverAddress, - Proto: proto, - Time: startTime.UTC().Truncate(time.Second), - // TODO(fortuna): Add sanitized config: - Transport: sanitizedConfig, - DurationMs: testDuration.Milliseconds(), - Error: makeErrorRecord(result), + Test: testReport{ + Resolver: resolverAddress, + Proto: proto, + Time: startTime.UTC().Truncate(time.Second), + // TODO(fortuna): Add sanitized config: + Transport: sanitizedConfig, + DurationMs: testDuration.Milliseconds(), + Error: makeErrorRecord(result), + }, + DNSQueries: dnsReports, + TCPConnections: tcpReports, } if reportCollector != nil { err = reportCollector.Collect(context.Background(), r) From 089afc6af39a39842b22231aae9ac4fc97b285c6 Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Wed, 4 Sep 2024 19:05:27 -0400 Subject: [PATCH 2/7] Add DNS duration and start time --- x/examples/test-connectivity/main.go | 24 +++++++++++++++++++----- 1 file changed, 19 insertions(+), 5 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index beb233c5..24516ddf 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -62,9 +62,11 @@ type testReport struct { } type dnsReport struct { - QueryName string `json:"query_name"` - AnswerIPs []string `json:"answer_ips"` - Error string `json:"error"` + QueryName string `json:"query_name"` + Time time.Time `json:"time"` + DurationMs int64 `json:"duration_ms"` + AnswerIPs []string `json:"answer_ips"` + Error string `json:"error"` } type tcpReport struct { @@ -198,10 +200,16 @@ func main() { if err != nil { return nil, err } + var dnsStart time.Time ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSStart: func(di httptrace.DNSStartInfo) { + dnsStart = time.Now() + }, DNSDone: func(di httptrace.DNSDoneInfo) { report := dnsReport{ - QueryName: hostname, + QueryName: hostname, + Time: dnsStart.UTC().Truncate(time.Second), + DurationMs: time.Since(dnsStart).Milliseconds(), } if di.Err != nil { report.Error = di.Err.Error() @@ -242,10 +250,16 @@ func main() { if err != nil { return nil, err } + var dnsStart time.Time ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSStart: func(di httptrace.DNSStartInfo) { + dnsStart = time.Now() + }, DNSDone: func(di httptrace.DNSDoneInfo) { report := dnsReport{ - QueryName: hostname, + QueryName: hostname, + Time: dnsStart.UTC().Truncate(time.Second), + DurationMs: time.Since(dnsStart).Milliseconds(), } if di.Err != nil { report.Error = di.Err.Error() From 01d1495d9c3f4dd9edf3345ba4e89b3481fc8cac Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Wed, 4 Sep 2024 19:10:18 -0400 Subject: [PATCH 3/7] Add TODOs --- x/examples/test-connectivity/main.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index 24516ddf..a841268a 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -217,6 +217,7 @@ func main() { for _, ip := range di.Addrs { report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) } + // TODO(fortuna): Use a Mutex. dnsReports = append(dnsReports, report) }, ConnectDone: func(network, addr string, connErr error) { @@ -232,6 +233,7 @@ func main() { if connErr != nil { report.Error = connErr.Error() } + // TODO(fortuna): Use a Mutex. tcpReports = append(tcpReports, report) }, }) @@ -267,6 +269,7 @@ func main() { for _, ip := range di.Addrs { report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) } + // TODO(fortuna): Use a Mutex. dnsReports = append(dnsReports, report) }, }) From 8548f45a657a1919620140c44298740e2160de5f Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Mon, 9 Sep 2024 14:31:36 -0400 Subject: [PATCH 4/7] decouple --- x/examples/test-connectivity/main.go | 70 ++++++++++++++++++---------- 1 file changed, 45 insertions(+), 25 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index a841268a..5e2d8527 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -120,6 +120,29 @@ func init() { flag.PrintDefaults() } } +func newTCPTraceDialer( + onDNS func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo), + onDial func(ctx context.Context, network, addr string, connErr error)) transport.StreamDialer { + dialer := &transport.TCPDialer{} + var onDNSDone func(di httptrace.DNSDoneInfo) + return transport.FuncStreamDialer(func(ctx context.Context, addr string) (transport.StreamConn, error) { + ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSStart: func(di httptrace.DNSStartInfo) { + onDNSDone = onDNS(ctx, di.Host) + }, + DNSDone: func(di httptrace.DNSDoneInfo) { + if onDNSDone != nil { + onDNSDone(di) + onDNSDone = nil + } + }, + ConnectDone: func(network, addr string, connErr error) { + onDial(ctx, network, addr, connErr) + }, + }) + return dialer.DialStream(ctx, addr) + }) +} func main() { verboseFlag := flag.Bool("v", false, "Enable debug output") @@ -200,12 +223,9 @@ func main() { if err != nil { return nil, err } - var dnsStart time.Time - ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ - DNSStart: func(di httptrace.DNSStartInfo) { - dnsStart = time.Now() - }, - DNSDone: func(di httptrace.DNSDoneInfo) { + onDNS := func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo) { + dnsStart := time.Now() + return func(di httptrace.DNSDoneInfo) { report := dnsReport{ QueryName: hostname, Time: dnsStart.UTC().Truncate(time.Second), @@ -219,25 +239,25 @@ func main() { } // TODO(fortuna): Use a Mutex. dnsReports = append(dnsReports, report) - }, - ConnectDone: func(network, addr string, connErr error) { - ip, port, err := net.SplitHostPort(addr) - if err != nil { - return - } - report := tcpReport{ - Hostname: hostname, - IP: ip, - Port: port, - } - if connErr != nil { - report.Error = connErr.Error() - } - // TODO(fortuna): Use a Mutex. - tcpReports = append(tcpReports, report) - }, - }) - return (&transport.TCPDialer{}).DialStream(ctx, addr) + } + } + onDial := func(ctx context.Context, network, addr string, connErr error) { + ip, port, err := net.SplitHostPort(addr) + if err != nil { + return + } + report := tcpReport{ + Hostname: hostname, + IP: ip, + Port: port, + } + if connErr != nil { + report.Error = connErr.Error() + } + // TODO(fortuna): Use a Mutex. + tcpReports = append(tcpReports, report) + } + return newTCPTraceDialer(onDNS, onDial).DialStream(ctx, addr) }) streamDialer, err := configToDialer.NewStreamDialer(*transportFlag) if err != nil { From acdc62201c16952ebbf4176759a2d8c9f33fa8a8 Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Fri, 20 Sep 2024 10:51:10 -0400 Subject: [PATCH 5/7] Add mutex and move trace --- x/examples/test-connectivity/main.go | 134 ++++++++++++++------------- 1 file changed, 69 insertions(+), 65 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index 5e2d8527..57d5b284 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -29,6 +29,7 @@ import ( "os" "path" "strings" + "sync" "time" "github.com/Jigsaw-Code/outline-sdk/dns" @@ -213,52 +214,84 @@ func main() { for _, proto := range strings.Split(*protoFlag, ",") { proto = strings.TrimSpace(proto) var resolver dns.Resolver + var mu sync.Mutex dnsReports := make([]dnsReport, 0) tcpReports := make([]tcpReport, 0) - switch proto { - case "tcp": - configToDialer := config.NewDefaultConfigToDialer() - configToDialer.BaseStreamDialer = transport.FuncStreamDialer(func(ctx context.Context, addr string) (transport.StreamConn, error) { - hostname, _, err := net.SplitHostPort(addr) + configToDialer := config.NewDefaultConfigToDialer() + configToDialer.BaseStreamDialer = transport.FuncStreamDialer(func(ctx context.Context, addr string) (transport.StreamConn, error) { + hostname, _, err := net.SplitHostPort(addr) + if err != nil { + return nil, err + } + onDNS := func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo) { + dnsStart := time.Now() + return func(di httptrace.DNSDoneInfo) { + report := dnsReport{ + QueryName: hostname, + Time: dnsStart.UTC().Truncate(time.Second), + DurationMs: time.Since(dnsStart).Milliseconds(), + } + if di.Err != nil { + report.Error = di.Err.Error() + } + for _, ip := range di.Addrs { + report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) + } + mu.Lock() + dnsReports = append(dnsReports, report) + mu.Unlock() + } + } + onDial := func(ctx context.Context, network, addr string, connErr error) { + ip, port, err := net.SplitHostPort(addr) if err != nil { - return nil, err + return } - onDNS := func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo) { - dnsStart := time.Now() - return func(di httptrace.DNSDoneInfo) { - report := dnsReport{ - QueryName: hostname, - Time: dnsStart.UTC().Truncate(time.Second), - DurationMs: time.Since(dnsStart).Milliseconds(), - } - if di.Err != nil { - report.Error = di.Err.Error() - } - for _, ip := range di.Addrs { - report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) - } - // TODO(fortuna): Use a Mutex. - dnsReports = append(dnsReports, report) - } + report := tcpReport{ + Hostname: hostname, + IP: ip, + Port: port, + } + if connErr != nil { + report.Error = connErr.Error() } - onDial := func(ctx context.Context, network, addr string, connErr error) { - ip, port, err := net.SplitHostPort(addr) - if err != nil { - return + mu.Lock() + tcpReports = append(tcpReports, report) + mu.Unlock() + } + return newTCPTraceDialer(onDNS, onDial).DialStream(ctx, addr) + }) + configToDialer.BasePacketDialer = transport.FuncPacketDialer(func(ctx context.Context, addr string) (net.Conn, error) { + hostname, _, err := net.SplitHostPort(addr) + if err != nil { + return nil, err + } + var dnsStart time.Time + ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSStart: func(di httptrace.DNSStartInfo) { + dnsStart = time.Now() + }, + DNSDone: func(di httptrace.DNSDoneInfo) { + report := dnsReport{ + QueryName: hostname, + Time: dnsStart.UTC().Truncate(time.Second), + DurationMs: time.Since(dnsStart).Milliseconds(), } - report := tcpReport{ - Hostname: hostname, - IP: ip, - Port: port, + if di.Err != nil { + report.Error = di.Err.Error() } - if connErr != nil { - report.Error = connErr.Error() + for _, ip := range di.Addrs { + report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) } // TODO(fortuna): Use a Mutex. - tcpReports = append(tcpReports, report) - } - return newTCPTraceDialer(onDNS, onDial).DialStream(ctx, addr) + dnsReports = append(dnsReports, report) + }, }) + return (&transport.UDPDialer{}).DialPacket(ctx, addr) + }) + + switch proto { + case "tcp": streamDialer, err := configToDialer.NewStreamDialer(*transportFlag) if err != nil { log.Fatalf("Failed to create StreamDialer: %v", err) @@ -266,35 +299,6 @@ func main() { resolver = dns.NewTCPResolver(streamDialer, resolverAddress) case "udp": - configToDialer := config.NewDefaultConfigToDialer() - configToDialer.BasePacketDialer = transport.FuncPacketDialer(func(ctx context.Context, addr string) (net.Conn, error) { - hostname, _, err := net.SplitHostPort(addr) - if err != nil { - return nil, err - } - var dnsStart time.Time - ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ - DNSStart: func(di httptrace.DNSStartInfo) { - dnsStart = time.Now() - }, - DNSDone: func(di httptrace.DNSDoneInfo) { - report := dnsReport{ - QueryName: hostname, - Time: dnsStart.UTC().Truncate(time.Second), - DurationMs: time.Since(dnsStart).Milliseconds(), - } - if di.Err != nil { - report.Error = di.Err.Error() - } - for _, ip := range di.Addrs { - report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) - } - // TODO(fortuna): Use a Mutex. - dnsReports = append(dnsReports, report) - }, - }) - return (&transport.UDPDialer{}).DialPacket(ctx, addr) - }) packetDialer, err := configToDialer.NewPacketDialer(*transportFlag) if err != nil { log.Fatalf("Failed to create PacketDialer: %v", err) From d8b3e5083b46c8e631f4cbf1f49645469dcd4ded Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Fri, 20 Sep 2024 10:51:58 -0400 Subject: [PATCH 6/7] Use slog --- x/examples/test-connectivity/main.go | 52 ++++++++++++++++------------ x/go.mod | 2 ++ x/go.sum | 2 ++ 3 files changed, 34 insertions(+), 22 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index 57d5b284..7cb57019 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -20,8 +20,7 @@ import ( "errors" "flag" "fmt" - "io" - "log" + "log/slog" "net" "net/http" "net/http/httptrace" @@ -37,12 +36,10 @@ import ( "github.com/Jigsaw-Code/outline-sdk/x/config" "github.com/Jigsaw-Code/outline-sdk/x/connectivity" "github.com/Jigsaw-Code/outline-sdk/x/report" + "github.com/lmittmann/tint" + "golang.org/x/term" ) -var debugLog log.Logger = *log.New(io.Discard, "", 0) - -// var errorLog log.Logger = *log.New(os.Stderr, "[ERROR] ", log.LstdFlags|log.Lmicroseconds|log.Lshortfile) - type connectivityReport struct { Test testReport `json:"test"` DNSQueries []dnsReport `json:"dns_queries,omitempty"` @@ -157,28 +154,34 @@ func main() { flag.Parse() + logLevel := slog.LevelInfo + if *verboseFlag { + logLevel = slog.LevelDebug + } + slog.SetDefault(slog.New(tint.NewHandler( + os.Stderr, + &tint.Options{NoColor: !term.IsTerminal(int(os.Stderr.Fd())), Level: logLevel}, + ))) + // Perform custom range validation for sampling rate if *reportSuccessFlag < 0.0 || *reportSuccessFlag > 1.0 { - fmt.Println("Error: report-success-rate must be between 0 and 1.") + slog.Error("Error: report-success-rate must be between 0 and 1.", "report-success-rate", *reportSuccessFlag) flag.Usage() - return + os.Exit(1) } if *reportFailureFlag < 0.0 || *reportFailureFlag > 1.0 { - fmt.Println("Error: report-failure-rate must be between 0 and 1.") + slog.Error("Error: report-failure-rate must be between 0 and 1.", "report-failure-rate", *reportFailureFlag) flag.Usage() - return - } - - if *verboseFlag { - debugLog = *log.New(os.Stderr, "[DEBUG] ", log.LstdFlags|log.Lmicroseconds|log.Lshortfile) + os.Exit(1) } var reportCollector report.Collector if *reportToFlag != "" { collectorURL, err := url.Parse(*reportToFlag) if err != nil { - debugLog.Printf("Failed to parse collector URL: %v", err) + slog.Error("Failed to parse collector URL", "url", err) + os.Exit(1) } remoteCollector := &report.RemoteCollector{ CollectorURL: collectorURL, @@ -294,32 +297,37 @@ func main() { case "tcp": streamDialer, err := configToDialer.NewStreamDialer(*transportFlag) if err != nil { - log.Fatalf("Failed to create StreamDialer: %v", err) + slog.Error("Failed to create StreamDialer", "error", err) + os.Exit(1) } resolver = dns.NewTCPResolver(streamDialer, resolverAddress) case "udp": packetDialer, err := configToDialer.NewPacketDialer(*transportFlag) if err != nil { - log.Fatalf("Failed to create PacketDialer: %v", err) + slog.Error("Failed to create PacketDialer", "error", err) + os.Exit(1) } resolver = dns.NewUDPResolver(packetDialer, resolverAddress) default: - log.Fatalf(`Invalid proto %v. Must be "tcp" or "udp"`, proto) + slog.Error(`Invalid proto. Must be "tcp" or "udp"`, "proto", proto) + os.Exit(1) } startTime := time.Now() result, err := connectivity.TestConnectivityWithResolver(context.Background(), resolver, *domainFlag) if err != nil { - log.Fatalf("Connectivity test failed to run: %v", err) + slog.Error("Connectivity test failed to run", "error", err) + os.Exit(1) } testDuration := time.Since(startTime) if result == nil { success = true } - debugLog.Printf("Test %v %v result: %v", proto, resolverAddress, result) + slog.Debug("Test done", "proto", proto, "resolver", resolverAddress, "result", result) sanitizedConfig, err := config.SanitizeConfig(*transportFlag) if err != nil { - log.Fatalf("Failed to sanitize config: %v", err) + slog.Error("Failed to sanitize config", "error", err) + os.Exit(1) } var r report.Report = connectivityReport{ Test: testReport{ @@ -337,7 +345,7 @@ func main() { if reportCollector != nil { err = reportCollector.Collect(context.Background(), r) if err != nil { - debugLog.Printf("Failed to collect report: %v\n", err) + slog.Warn("Failed to collect report", "error", err) } } } diff --git a/x/go.mod b/x/go.mod index 0b322d15..a1dfe72a 100644 --- a/x/go.mod +++ b/x/go.mod @@ -7,12 +7,14 @@ require ( // Use github.com/Psiphon-Labs/psiphon-tunnel-core@staging-client as per // https://github.com/Psiphon-Labs/psiphon-tunnel-core/?tab=readme-ov-file#using-psiphon-with-go-modules github.com/Psiphon-Labs/psiphon-tunnel-core v1.0.11-0.20240619172145-03cade11f647 + github.com/lmittmann/tint v1.0.5 github.com/songgao/water v0.0.0-20200317203138-2b4b6d7c09d8 github.com/stretchr/testify v1.9.0 github.com/vishvananda/netlink v1.1.0 golang.org/x/mobile v0.0.0-20240520174638-fa72addaaa1b golang.org/x/net v0.25.0 golang.org/x/sys v0.20.0 + golang.org/x/term v0.20.0 ) require ( diff --git a/x/go.sum b/x/go.sum index 7c6f5f80..033308fb 100644 --- a/x/go.sum +++ b/x/go.sum @@ -104,6 +104,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/libp2p/go-reuseport v0.4.0 h1:nR5KU7hD0WxXCJbmw7r2rhRYruNRl2koHw8fQscQm2s= github.com/libp2p/go-reuseport v0.4.0/go.mod h1:ZtI03j/wO5hZVDFo2jKywN6bYKWLOy8Se6DrI2E1cLU= +github.com/lmittmann/tint v1.0.5 h1:NQclAutOfYsqs2F1Lenue6OoWCajs5wJcP3DfWVpePw= +github.com/lmittmann/tint v1.0.5/go.mod h1:HIS3gSy7qNwGCj+5oRjAutErFBl4BzdQP6cJZ0NfMwE= github.com/marusama/semaphore v0.0.0-20171214154724-565ffd8e868a h1:6SRny9FLB1eWasPyDUqBQnMi9NhXU01XIlB0ao89YoI= github.com/marusama/semaphore v0.0.0-20171214154724-565ffd8e868a/go.mod h1:TmeOqAKoDinfPfSohs14CO3VcEf7o+Bem6JiNe05yrQ= github.com/mdlayher/netlink v1.4.2-0.20210930205308-a81a8c23d40a h1:yk5OmRew64lWdeNanQ3l0hDgUt1E8MfipPhh/GO9Tuw= From 6139f8feefde2547e7d3414e458be82027ac851f Mon Sep 17 00:00:00 2001 From: Vinicius Fortuna Date: Fri, 20 Sep 2024 11:06:03 -0400 Subject: [PATCH 7/7] Refactor --- x/examples/test-connectivity/main.go | 85 +++++++++++++--------------- 1 file changed, 40 insertions(+), 45 deletions(-) diff --git a/x/examples/test-connectivity/main.go b/x/examples/test-connectivity/main.go index 5451e2af..1edd5584 100644 --- a/x/examples/test-connectivity/main.go +++ b/x/examples/test-connectivity/main.go @@ -142,6 +142,26 @@ func newTCPTraceDialer( }) } +func newUDPTraceDialer( + onDNS func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo)) transport.PacketDialer { + dialer := &transport.UDPDialer{} + var onDNSDone func(di httptrace.DNSDoneInfo) + return transport.FuncPacketDialer(func(ctx context.Context, addr string) (net.Conn, error) { + ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ + DNSStart: func(di httptrace.DNSStartInfo) { + onDNSDone = onDNS(ctx, di.Host) + }, + DNSDone: func(di httptrace.DNSDoneInfo) { + if onDNSDone != nil { + onDNSDone(di) + onDNSDone = nil + } + }, + }) + return dialer.DialPacket(ctx, addr) + }) +} + func main() { verboseFlag := flag.Bool("v", false, "Enable debug output") transportFlag := flag.String("transport", "", "Transport config") @@ -221,30 +241,30 @@ func main() { dnsReports := make([]dnsReport, 0) tcpReports := make([]tcpReport, 0) configToDialer := configurl.NewDefaultConfigToDialer() + onDNS := func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo) { + dnsStart := time.Now() + return func(di httptrace.DNSDoneInfo) { + report := dnsReport{ + QueryName: domain, + Time: dnsStart.UTC().Truncate(time.Second), + DurationMs: time.Since(dnsStart).Milliseconds(), + } + if di.Err != nil { + report.Error = di.Err.Error() + } + for _, ip := range di.Addrs { + report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) + } + mu.Lock() + dnsReports = append(dnsReports, report) + mu.Unlock() + } + } configToDialer.BaseStreamDialer = transport.FuncStreamDialer(func(ctx context.Context, addr string) (transport.StreamConn, error) { hostname, _, err := net.SplitHostPort(addr) if err != nil { return nil, err } - onDNS := func(ctx context.Context, domain string) func(di httptrace.DNSDoneInfo) { - dnsStart := time.Now() - return func(di httptrace.DNSDoneInfo) { - report := dnsReport{ - QueryName: hostname, - Time: dnsStart.UTC().Truncate(time.Second), - DurationMs: time.Since(dnsStart).Milliseconds(), - } - if di.Err != nil { - report.Error = di.Err.Error() - } - for _, ip := range di.Addrs { - report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) - } - mu.Lock() - dnsReports = append(dnsReports, report) - mu.Unlock() - } - } onDial := func(ctx context.Context, network, addr string, connErr error) { ip, port, err := net.SplitHostPort(addr) if err != nil { @@ -265,32 +285,7 @@ func main() { return newTCPTraceDialer(onDNS, onDial).DialStream(ctx, addr) }) configToDialer.BasePacketDialer = transport.FuncPacketDialer(func(ctx context.Context, addr string) (net.Conn, error) { - hostname, _, err := net.SplitHostPort(addr) - if err != nil { - return nil, err - } - var dnsStart time.Time - ctx = httptrace.WithClientTrace(ctx, &httptrace.ClientTrace{ - DNSStart: func(di httptrace.DNSStartInfo) { - dnsStart = time.Now() - }, - DNSDone: func(di httptrace.DNSDoneInfo) { - report := dnsReport{ - QueryName: hostname, - Time: dnsStart.UTC().Truncate(time.Second), - DurationMs: time.Since(dnsStart).Milliseconds(), - } - if di.Err != nil { - report.Error = di.Err.Error() - } - for _, ip := range di.Addrs { - report.AnswerIPs = append(report.AnswerIPs, ip.IP.String()) - } - // TODO(fortuna): Use a Mutex. - dnsReports = append(dnsReports, report) - }, - }) - return (&transport.UDPDialer{}).DialPacket(ctx, addr) + return newUDPTraceDialer(onDNS).DialPacket(ctx, addr) }) switch proto {