mirror of
https://github.com/yusing/godoxy.git
synced 2025-06-10 21:32:33 +02:00
316 lines
6.5 KiB
Go
316 lines
6.5 KiB
Go
package stream
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"maps"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/rs/zerolog"
|
|
nettypes "github.com/yusing/go-proxy/internal/net/types"
|
|
"github.com/yusing/go-proxy/internal/utils/synk"
|
|
"go.uber.org/atomic"
|
|
)
|
|
|
|
type UDPUDPStream struct {
|
|
name string
|
|
listener *net.UDPConn
|
|
|
|
laddr *net.UDPAddr
|
|
dst *net.UDPAddr
|
|
|
|
preDial nettypes.HookFunc
|
|
onRead nettypes.HookFunc
|
|
|
|
cleanUpTicker *time.Ticker
|
|
|
|
conns map[string]*udpUDPConn
|
|
closed atomic.Bool
|
|
mu sync.Mutex
|
|
}
|
|
|
|
type udpUDPConn struct {
|
|
srcAddr *net.UDPAddr
|
|
dstConn *net.UDPConn
|
|
listener *net.UDPConn
|
|
lastUsed atomic.Time
|
|
closed atomic.Bool
|
|
mu sync.Mutex
|
|
}
|
|
|
|
const (
|
|
udpBufferSize = 16 * 1024
|
|
udpIdleTimeout = 5 * time.Minute // Longer timeout for game sessions
|
|
udpCleanupInterval = 1 * time.Minute
|
|
udpReadTimeout = 30 * time.Second
|
|
)
|
|
|
|
var bufPool = synk.NewBytesPool()
|
|
|
|
func NewUDPUDPStream(listenAddr, dstAddr string) (nettypes.Stream, error) {
|
|
dst, err := net.ResolveUDPAddr("udp", dstAddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
laddr, err := net.ResolveUDPAddr("udp", listenAddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &UDPUDPStream{
|
|
laddr: laddr,
|
|
dst: dst,
|
|
conns: make(map[string]*udpUDPConn),
|
|
}, nil
|
|
}
|
|
|
|
func (s *UDPUDPStream) ListenAndServe(ctx context.Context, preDial, onRead nettypes.HookFunc) {
|
|
listener, err := net.ListenUDP("udp", s.laddr)
|
|
if err != nil {
|
|
logErr(s, err, "failed to listen")
|
|
return
|
|
}
|
|
s.listener = listener
|
|
s.preDial = preDial
|
|
s.onRead = onRead
|
|
go s.listen(ctx)
|
|
go s.cleanUp(ctx)
|
|
}
|
|
|
|
func (s *UDPUDPStream) Close() error {
|
|
if s.closed.Swap(true) || s.listener == nil {
|
|
return nil
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
s.mu.Lock()
|
|
for _, conn := range s.conns {
|
|
wg.Add(1)
|
|
go func(c *udpUDPConn) {
|
|
defer wg.Done()
|
|
c.Close()
|
|
}(conn)
|
|
}
|
|
clear(s.conns)
|
|
s.mu.Unlock()
|
|
|
|
wg.Wait()
|
|
|
|
return s.listener.Close()
|
|
}
|
|
|
|
func (s *UDPUDPStream) LocalAddr() net.Addr {
|
|
if s.listener == nil {
|
|
return s.laddr
|
|
}
|
|
return s.listener.LocalAddr()
|
|
}
|
|
|
|
func (s *UDPUDPStream) MarshalZerologObject(e *zerolog.Event) {
|
|
e.Str("protocol", "udp-udp").Str("name", s.name).Str("dst", s.dst.String())
|
|
}
|
|
|
|
func (s *UDPUDPStream) listen(ctx context.Context) {
|
|
buf := bufPool.GetSized(udpBufferSize)
|
|
defer bufPool.Put(buf)
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
n, srcAddr, err := s.listener.ReadFromUDP(buf)
|
|
if err != nil {
|
|
if s.closed.Load() {
|
|
return
|
|
}
|
|
logErr(s, err, "failed to read from listener")
|
|
continue
|
|
}
|
|
|
|
logDebugf(s, "read %d bytes from %s", n, srcAddr)
|
|
|
|
if s.onRead != nil {
|
|
if err := s.onRead(ctx); err != nil {
|
|
logErr(s, err, "failed to on read")
|
|
continue
|
|
}
|
|
}
|
|
|
|
// Get or create connection, passing the initial data
|
|
go s.getOrCreateConnection(ctx, srcAddr, bytes.Clone(buf[:n]))
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *UDPUDPStream) getOrCreateConnection(ctx context.Context, srcAddr *net.UDPAddr, initialData []byte) {
|
|
key := srcAddr.String()
|
|
|
|
s.mu.Lock()
|
|
if conn, ok := s.conns[key]; ok {
|
|
s.mu.Unlock()
|
|
// Forward packet for existing connection
|
|
go conn.forwardToDestination(initialData)
|
|
return
|
|
}
|
|
|
|
defer s.mu.Unlock()
|
|
// Create new connection with initial data
|
|
conn, ok := s.createConnection(ctx, srcAddr, initialData)
|
|
if ok && !conn.closed.Load() {
|
|
s.conns[key] = conn
|
|
}
|
|
}
|
|
|
|
func (s *UDPUDPStream) createConnection(ctx context.Context, srcAddr *net.UDPAddr, initialData []byte) (*udpUDPConn, bool) {
|
|
// Apply pre-dial if configured
|
|
if s.preDial != nil {
|
|
if err := s.preDial(ctx); err != nil {
|
|
logErr(s, err, "failed to pre-dial")
|
|
return nil, false
|
|
}
|
|
}
|
|
|
|
// Create UDP connection to destination
|
|
dstConn, err := net.DialUDP("udp", nil, s.dst)
|
|
if err != nil {
|
|
logErr(s, err, "failed to dial dst")
|
|
return nil, false
|
|
}
|
|
|
|
conn := &udpUDPConn{
|
|
srcAddr: srcAddr,
|
|
dstConn: dstConn,
|
|
listener: s.listener,
|
|
}
|
|
conn.lastUsed.Store(time.Now())
|
|
|
|
// Send initial data before starting response handler
|
|
if !conn.forwardToDestination(initialData) {
|
|
dstConn.Close()
|
|
return nil, false
|
|
}
|
|
|
|
// Start response handler after initial data is sent
|
|
go conn.handleResponses(ctx)
|
|
|
|
logDebugf(s, "created new connection from %s", srcAddr.String())
|
|
return conn, true
|
|
}
|
|
|
|
func (conn *udpUDPConn) MarshalZerologObject(e *zerolog.Event) {
|
|
e.Stringer("src", conn.srcAddr).Stringer("dst", conn.dstConn.RemoteAddr())
|
|
}
|
|
|
|
func (conn *udpUDPConn) handleResponses(ctx context.Context) {
|
|
buf := bufPool.GetSized(udpBufferSize)
|
|
defer bufPool.Put(buf)
|
|
|
|
defer conn.Close()
|
|
|
|
for {
|
|
if conn.closed.Load() {
|
|
return
|
|
}
|
|
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
default:
|
|
// Set a reasonable timeout for reads
|
|
_ = conn.dstConn.SetReadDeadline(time.Now().Add(udpReadTimeout))
|
|
|
|
n, err := conn.dstConn.Read(buf)
|
|
if err != nil {
|
|
if !conn.closed.Load() {
|
|
logErr(conn, err, "failed to read from dst")
|
|
}
|
|
return
|
|
}
|
|
|
|
// Clear deadline after successful read
|
|
_ = conn.dstConn.SetReadDeadline(time.Time{})
|
|
|
|
// Forward response back to client using the listener
|
|
_, err = conn.listener.WriteToUDP(buf[:n], conn.srcAddr)
|
|
if err != nil {
|
|
if !conn.closed.Load() {
|
|
logErrf(conn, err, "failed to write %d bytes to client", n)
|
|
}
|
|
return
|
|
}
|
|
|
|
conn.lastUsed.Store(time.Now())
|
|
logDebugf(conn, "forwarded response to client, %d bytes", n)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *UDPUDPStream) cleanUp(ctx context.Context) {
|
|
s.cleanUpTicker = time.NewTicker(udpCleanupInterval)
|
|
defer s.cleanUpTicker.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case <-s.cleanUpTicker.C:
|
|
s.mu.Lock()
|
|
conns := maps.Clone(s.conns)
|
|
s.mu.Unlock()
|
|
|
|
removed := []string(nil)
|
|
for key, conn := range conns {
|
|
if conn.Expired() {
|
|
conn.Close()
|
|
removed = append(removed, key)
|
|
}
|
|
}
|
|
|
|
s.mu.Lock()
|
|
for _, key := range removed {
|
|
logDebugf(s, "cleaning up expired connection: %s", key)
|
|
delete(s.conns, key)
|
|
}
|
|
s.mu.Unlock()
|
|
}
|
|
}
|
|
}
|
|
|
|
func (conn *udpUDPConn) forwardToDestination(data []byte) bool {
|
|
conn.mu.Lock()
|
|
defer conn.mu.Unlock()
|
|
|
|
if conn.closed.Load() {
|
|
return false
|
|
}
|
|
|
|
_, err := conn.dstConn.Write(data)
|
|
if err != nil {
|
|
logErrf(conn, err, "failed to write %d bytes to dst", len(data))
|
|
return false
|
|
}
|
|
|
|
conn.lastUsed.Store(time.Now())
|
|
logDebugf(conn, "forwarded %d bytes to dst", len(data))
|
|
return true
|
|
}
|
|
|
|
func (conn *udpUDPConn) Expired() bool {
|
|
return time.Since(conn.lastUsed.Load()) > udpIdleTimeout
|
|
}
|
|
|
|
func (conn *udpUDPConn) Close() {
|
|
conn.mu.Lock()
|
|
defer conn.mu.Unlock()
|
|
|
|
if conn.closed.Load() {
|
|
return
|
|
}
|
|
|
|
conn.closed.Store(true)
|
|
|
|
conn.dstConn.Close()
|
|
conn.dstConn = nil
|
|
}
|