mirror of
https://github.com/yusing/godoxy.git
synced 2026-03-26 19:31:46 +01:00
133 lines
2.4 KiB
Go
Executable File
133 lines
2.4 KiB
Go
Executable File
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
const tcpDialTimeout = 5 * time.Second
|
|
|
|
type TCPRoute struct {
|
|
*StreamRouteBase
|
|
listener net.Listener
|
|
connChan chan net.Conn
|
|
}
|
|
|
|
func NewTCPRoute(config *ProxyConfig) (StreamRoute, error) {
|
|
base, err := newStreamRouteBase(config)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if base.TargetScheme != StreamType_TCP {
|
|
return nil, fmt.Errorf("tcp to %s not yet supported", base.TargetScheme)
|
|
}
|
|
return &TCPRoute{
|
|
StreamRouteBase: base,
|
|
listener: nil,
|
|
connChan: make(chan net.Conn),
|
|
}, nil
|
|
}
|
|
|
|
func (route *TCPRoute) Start() {
|
|
route.setupListen()
|
|
in, err := net.Listen("tcp", fmt.Sprintf(":%v", route.ListeningPort))
|
|
if err != nil {
|
|
route.l.Error(err)
|
|
return
|
|
}
|
|
route.listener = in
|
|
route.wg.Add(2)
|
|
go route.grAcceptConnections()
|
|
go route.grHandleConnections()
|
|
}
|
|
|
|
func (route *TCPRoute) Stop() {
|
|
stopListening(route)
|
|
streamRoutes.Delete(route.id)
|
|
}
|
|
|
|
func (route *TCPRoute) closeListeners() {
|
|
if route.listener == nil {
|
|
return
|
|
}
|
|
route.listener.Close()
|
|
route.listener = nil
|
|
}
|
|
|
|
func (route *TCPRoute) grAcceptConnections() {
|
|
defer route.wg.Done()
|
|
|
|
for {
|
|
select {
|
|
case <-route.stopChann:
|
|
return
|
|
default:
|
|
conn, err := route.listener.Accept()
|
|
if err != nil {
|
|
route.l.Error(err)
|
|
continue
|
|
}
|
|
route.connChan <- conn
|
|
}
|
|
}
|
|
}
|
|
|
|
func (route *TCPRoute) grHandleConnections() {
|
|
defer route.wg.Done()
|
|
|
|
for {
|
|
select {
|
|
case <-route.stopChann:
|
|
return
|
|
case conn := <-route.connChan:
|
|
route.wg.Add(1)
|
|
go route.grHandleConnection(conn)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (route *TCPRoute) grHandleConnection(clientConn net.Conn) {
|
|
defer clientConn.Close()
|
|
defer route.wg.Done()
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), tcpDialTimeout)
|
|
defer cancel()
|
|
|
|
serverAddr := fmt.Sprintf("%s:%v", route.TargetHost, route.TargetPort)
|
|
dialer := &net.Dialer{}
|
|
serverConn, err := dialer.DialContext(ctx, route.TargetScheme, serverAddr)
|
|
if err != nil {
|
|
route.l.WithField("stage", "dial").Infof("%v", err)
|
|
return
|
|
}
|
|
route.tcpPipe(clientConn, serverConn)
|
|
}
|
|
|
|
func (route *TCPRoute) tcpPipe(src net.Conn, dest net.Conn) {
|
|
close := func() {
|
|
src.Close()
|
|
dest.Close()
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(2) // Number of goroutines
|
|
|
|
go func() {
|
|
_, err := io.Copy(src, dest)
|
|
route.l.Error(err)
|
|
close()
|
|
wg.Done()
|
|
}()
|
|
go func() {
|
|
_, err := io.Copy(dest, src)
|
|
route.l.Error(err)
|
|
close()
|
|
wg.Done()
|
|
}()
|
|
wg.Wait()
|
|
}
|