2024-07-25 21:25:56 +00:00
|
|
|
package socks5
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/puzpuzpuz/xsync/v3"
|
|
|
|
"log/slog"
|
|
|
|
"net"
|
|
|
|
)
|
|
|
|
|
|
|
|
type TCPListener struct {
|
|
|
|
listener net.Listener
|
|
|
|
connectChannels *xsync.MapOf[string, chan net.Conn] // todo -- use [16]byte for uuid instead of string
|
|
|
|
}
|
|
|
|
|
2024-07-26 19:21:50 +00:00
|
|
|
func NewTCPListener(address string) (*TCPListener, error) {
|
|
|
|
l, err := net.Listen("tcp", address)
|
2024-07-25 21:25:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &TCPListener{
|
|
|
|
listener: l,
|
|
|
|
connectChannels: xsync.NewMapOf[string, chan net.Conn](),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *TCPListener) AddConnectChannel(uuid uuid.UUID, ch chan net.Conn) {
|
|
|
|
l.connectChannels.Store(uuid.String(), ch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start starts the listener
|
|
|
|
func (l *TCPListener) Start() {
|
|
|
|
for {
|
|
|
|
conn, err := l.listener.Accept()
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
go l.handleConnection(conn)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// handleConnection handles the connection
|
|
|
|
func (l *TCPListener) handleConnection(conn net.Conn) {
|
|
|
|
//defer conn.Close()
|
|
|
|
for {
|
|
|
|
// read uuid from the connection
|
|
|
|
readbuffer := make([]byte, 36)
|
|
|
|
|
|
|
|
_, err := conn.Read(readbuffer)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// check if uuid is in the map
|
|
|
|
ch, ok := l.connectChannels.Load(string(readbuffer))
|
|
|
|
if !ok {
|
|
|
|
slog.Error("uuid not found in map")
|
2024-08-01 12:35:28 +00:00
|
|
|
continue
|
2024-07-25 21:25:56 +00:00
|
|
|
}
|
2024-08-01 12:35:28 +00:00
|
|
|
slog.Info("uuid found in map")
|
|
|
|
conn.Write([]byte{1})
|
2024-07-25 21:25:56 +00:00
|
|
|
// send the connection to the channel
|
|
|
|
ch <- conn
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|