Files
unchain/client/socks5_bind.go
Eric Zhou b60caf4869
Some checks failed
Go / build (push) Has been cancelled
构建image和发布 / build-and-push (push) Has been cancelled
fix memory leak
2025-08-26 09:25:57 +08:00

67 lines
1.5 KiB
Go

package client
import (
"io"
"log/slog"
"net"
"sync"
)
func relayBind(s5 net.Conn, _ *Socks5Request) {
bindListener, err := net.Listen("tcp4", ":0")
if err != nil {
slog.Error("bind tcp failed", "err", err)
socks5Response(s5, net.IPv4zero, 0, socks5ReplyFail)
return
}
defer bindListener.Close()
//first reply
localAddr := bindListener.Addr().(*net.TCPAddr)
socks5Response(s5, localAddr.IP, localAddr.Port, socks5ReplyOkay)
targetConn, err := bindListener.Accept()
if err != nil {
slog.Error("bind tcp failed", "err", err)
return
}
defer targetConn.Close()
//sec reply
targetAddr := targetConn.RemoteAddr().(*net.TCPAddr)
socks5Response(s5, targetAddr.IP, targetAddr.Port, socks5ReplyOkay)
var wg sync.WaitGroup
// Use a done channel to ensure both goroutines exit when one finishes
done := make(chan struct{})
wg.Add(2)
go func() {
defer wg.Done()
defer func() {
close(done) // Signal the other goroutine to exit
targetConn.Close() // Force close to interrupt the other goroutine
s5.Close()
}()
_, err := io.Copy(targetConn, s5)
if err != nil {
slog.Error("bind tcp failed", "err", err)
}
}()
go func() {
defer wg.Done()
defer func() {
targetConn.Close() // Force close to interrupt the other goroutine
s5.Close()
}()
select {
case <-done:
return // Other goroutine finished, exit
default:
}
_, err := io.Copy(s5, targetConn)
if err != nil {
slog.Error("bind tcp failed", "err", err)
}
}()
wg.Wait()
}