mirror of
https://github.com/v2fly/v2ray-core.git
synced 2024-10-20 10:43:56 -04:00
196 lines
4.3 KiB
Go
196 lines
4.3 KiB
Go
package kcp
|
|
|
|
import (
|
|
"context"
|
|
"crypto/cipher"
|
|
gotls "crypto/tls"
|
|
"sync"
|
|
|
|
"github.com/v2fly/v2ray-core/v4/common"
|
|
"github.com/v2fly/v2ray-core/v4/common/buf"
|
|
"github.com/v2fly/v2ray-core/v4/common/net"
|
|
"github.com/v2fly/v2ray-core/v4/transport/internet"
|
|
"github.com/v2fly/v2ray-core/v4/transport/internet/tls"
|
|
"github.com/v2fly/v2ray-core/v4/transport/internet/udp"
|
|
)
|
|
|
|
type ConnectionID struct {
|
|
Remote net.Address
|
|
Port net.Port
|
|
Conv uint16
|
|
}
|
|
|
|
// Listener defines a server listening for connections
|
|
type Listener struct {
|
|
sync.Mutex
|
|
sessions map[ConnectionID]*Connection
|
|
hub *udp.Hub
|
|
tlsConfig *gotls.Config
|
|
config *Config
|
|
reader PacketReader
|
|
header internet.PacketHeader
|
|
security cipher.AEAD
|
|
addConn internet.ConnHandler
|
|
}
|
|
|
|
func NewListener(ctx context.Context, address net.Address, port net.Port, streamSettings *internet.MemoryStreamConfig, addConn internet.ConnHandler) (*Listener, error) {
|
|
kcpSettings := streamSettings.ProtocolSettings.(*Config)
|
|
header, err := kcpSettings.GetPackerHeader()
|
|
if err != nil {
|
|
return nil, newError("failed to create packet header").Base(err).AtError()
|
|
}
|
|
security, err := kcpSettings.GetSecurity()
|
|
if err != nil {
|
|
return nil, newError("failed to create security").Base(err).AtError()
|
|
}
|
|
l := &Listener{
|
|
header: header,
|
|
security: security,
|
|
reader: &KCPPacketReader{
|
|
Header: header,
|
|
Security: security,
|
|
},
|
|
sessions: make(map[ConnectionID]*Connection),
|
|
config: kcpSettings,
|
|
addConn: addConn,
|
|
}
|
|
|
|
if config := tls.ConfigFromStreamSettings(streamSettings); config != nil {
|
|
l.tlsConfig = config.GetTLSConfig()
|
|
}
|
|
|
|
hub, err := udp.ListenUDP(ctx, address, port, streamSettings, udp.HubCapacity(1024))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
l.Lock()
|
|
l.hub = hub
|
|
l.Unlock()
|
|
newError("listening on ", address, ":", port).WriteToLog()
|
|
|
|
go l.handlePackets()
|
|
|
|
return l, nil
|
|
}
|
|
|
|
func (l *Listener) handlePackets() {
|
|
receive := l.hub.Receive()
|
|
for payload := range receive {
|
|
l.OnReceive(payload.Payload, payload.Source)
|
|
}
|
|
}
|
|
|
|
func (l *Listener) OnReceive(payload *buf.Buffer, src net.Destination) {
|
|
segments := l.reader.Read(payload.Bytes())
|
|
payload.Release()
|
|
|
|
if len(segments) == 0 {
|
|
newError("discarding invalid payload from ", src).WriteToLog()
|
|
return
|
|
}
|
|
|
|
conv := segments[0].Conversation()
|
|
cmd := segments[0].Command()
|
|
|
|
id := ConnectionID{
|
|
Remote: src.Address,
|
|
Port: src.Port,
|
|
Conv: conv,
|
|
}
|
|
|
|
l.Lock()
|
|
defer l.Unlock()
|
|
|
|
conn, found := l.sessions[id]
|
|
|
|
if !found {
|
|
if cmd == CommandTerminate {
|
|
return
|
|
}
|
|
writer := &Writer{
|
|
id: id,
|
|
hub: l.hub,
|
|
dest: src,
|
|
listener: l,
|
|
}
|
|
remoteAddr := &net.UDPAddr{
|
|
IP: src.Address.IP(),
|
|
Port: int(src.Port),
|
|
}
|
|
localAddr := l.hub.Addr()
|
|
conn = NewConnection(ConnMetadata{
|
|
LocalAddr: localAddr,
|
|
RemoteAddr: remoteAddr,
|
|
Conversation: conv,
|
|
}, &KCPPacketWriter{
|
|
Header: l.header,
|
|
Security: l.security,
|
|
Writer: writer,
|
|
}, writer, l.config)
|
|
var netConn internet.Connection = conn
|
|
if l.tlsConfig != nil {
|
|
netConn = tls.Server(conn, l.tlsConfig)
|
|
}
|
|
|
|
l.addConn(netConn)
|
|
l.sessions[id] = conn
|
|
}
|
|
conn.Input(segments)
|
|
}
|
|
|
|
func (l *Listener) Remove(id ConnectionID) {
|
|
l.Lock()
|
|
delete(l.sessions, id)
|
|
l.Unlock()
|
|
}
|
|
|
|
// Close stops listening on the UDP address. Already Accepted connections are not closed.
|
|
func (l *Listener) Close() error {
|
|
l.hub.Close()
|
|
|
|
l.Lock()
|
|
defer l.Unlock()
|
|
|
|
for _, conn := range l.sessions {
|
|
go conn.Terminate()
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (l *Listener) ActiveConnections() int {
|
|
l.Lock()
|
|
defer l.Unlock()
|
|
|
|
return len(l.sessions)
|
|
}
|
|
|
|
// Addr returns the listener's network address, The Addr returned is shared by all invocations of Addr, so do not modify it.
|
|
func (l *Listener) Addr() net.Addr {
|
|
return l.hub.Addr()
|
|
}
|
|
|
|
type Writer struct {
|
|
id ConnectionID
|
|
dest net.Destination
|
|
hub *udp.Hub
|
|
listener *Listener
|
|
}
|
|
|
|
func (w *Writer) Write(payload []byte) (int, error) {
|
|
return w.hub.WriteTo(payload, w.dest)
|
|
}
|
|
|
|
func (w *Writer) Close() error {
|
|
w.listener.Remove(w.id)
|
|
return nil
|
|
}
|
|
|
|
func ListenKCP(ctx context.Context, address net.Address, port net.Port, streamSettings *internet.MemoryStreamConfig, addConn internet.ConnHandler) (internet.Listener, error) {
|
|
return NewListener(ctx, address, port, streamSettings, addConn)
|
|
}
|
|
|
|
func init() {
|
|
common.Must(internet.RegisterTransportListener(protocolName, ListenKCP))
|
|
}
|