mirror of
https://github.com/yusing/godoxy.git
synced 2026-04-24 17:28:31 +02:00
feat(idlesleep): support idlesleep for stream routes, rewritten and fixed stream implementation
This commit is contained in:
316
internal/route/stream/udp_udp.go
Normal file
316
internal/route/stream/udp_udp.go
Normal file
@@ -0,0 +1,316 @@
|
||||
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
|
||||
}
|
||||
Reference in New Issue
Block a user