mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
222 lines
5.4 KiB
Go
222 lines
5.4 KiB
Go
package quic
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/tls"
|
|
"errors"
|
|
"net"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/lucas-clemente/quic-go/crypto"
|
|
"github.com/lucas-clemente/quic-go/handshake"
|
|
"github.com/lucas-clemente/quic-go/protocol"
|
|
"github.com/lucas-clemente/quic-go/qerr"
|
|
"github.com/lucas-clemente/quic-go/utils"
|
|
)
|
|
|
|
// packetHandler handles packets
|
|
type packetHandler interface {
|
|
handlePacket(*receivedPacket)
|
|
run()
|
|
Close(error) error
|
|
}
|
|
|
|
// A Server of QUIC
|
|
type Server struct {
|
|
addr *net.UDPAddr
|
|
|
|
conn *net.UDPConn
|
|
connMutex sync.Mutex
|
|
|
|
signer crypto.Signer
|
|
scfg *handshake.ServerConfig
|
|
|
|
sessions map[protocol.ConnectionID]packetHandler
|
|
sessionsMutex sync.RWMutex
|
|
|
|
streamCallback StreamCallback
|
|
|
|
newSession func(conn connection, v protocol.VersionNumber, connectionID protocol.ConnectionID, sCfg *handshake.ServerConfig, streamCallback StreamCallback, closeCallback closeCallback) (packetHandler, error)
|
|
}
|
|
|
|
// NewServer makes a new server
|
|
func NewServer(addr string, tlsConfig *tls.Config, cb StreamCallback) (*Server, error) {
|
|
signer, err := crypto.NewProofSource(tlsConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
kex, err := crypto.NewCurve25519KEX()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
scfg, err := handshake.NewServerConfig(kex, signer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
udpAddr, err := net.ResolveUDPAddr("udp", addr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Server{
|
|
addr: udpAddr,
|
|
signer: signer,
|
|
scfg: scfg,
|
|
streamCallback: cb,
|
|
sessions: map[protocol.ConnectionID]packetHandler{},
|
|
newSession: newSession,
|
|
}, nil
|
|
}
|
|
|
|
// ListenAndServe listens and serves a connection
|
|
func (s *Server) ListenAndServe() error {
|
|
conn, err := net.ListenUDP("udp", s.addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return s.Serve(conn)
|
|
}
|
|
|
|
// Serve on an existing UDP connection.
|
|
func (s *Server) Serve(conn *net.UDPConn) error {
|
|
s.connMutex.Lock()
|
|
s.conn = conn
|
|
s.connMutex.Unlock()
|
|
|
|
for {
|
|
data := getPacketBuffer()
|
|
data = data[:protocol.MaxPacketSize]
|
|
n, remoteAddr, err := conn.ReadFromUDP(data)
|
|
if err != nil {
|
|
if strings.HasSuffix(err.Error(), "use of closed network connection") {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
data = data[:n]
|
|
if err := s.handlePacket(conn, remoteAddr, data); err != nil {
|
|
utils.Errorf("error handling packet: %s", err.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
// Close the server
|
|
func (s *Server) Close() error {
|
|
s.sessionsMutex.Lock()
|
|
for _, session := range s.sessions {
|
|
if session != nil {
|
|
s.sessionsMutex.Unlock()
|
|
_ = session.Close(nil)
|
|
s.sessionsMutex.Lock()
|
|
}
|
|
}
|
|
s.sessionsMutex.Unlock()
|
|
|
|
s.connMutex.Lock()
|
|
conn := s.conn
|
|
s.conn = nil
|
|
s.connMutex.Unlock()
|
|
|
|
if conn == nil {
|
|
return nil
|
|
}
|
|
return conn.Close()
|
|
}
|
|
|
|
func (s *Server) handlePacket(conn *net.UDPConn, remoteAddr *net.UDPAddr, packet []byte) error {
|
|
if protocol.ByteCount(len(packet)) > protocol.MaxPacketSize {
|
|
return qerr.PacketTooLarge
|
|
}
|
|
|
|
rcvTime := time.Now()
|
|
|
|
r := bytes.NewReader(packet)
|
|
|
|
hdr, err := ParsePublicHeader(r)
|
|
if err != nil {
|
|
return qerr.Error(qerr.InvalidPacketHeader, err.Error())
|
|
}
|
|
hdr.Raw = packet[:len(packet)-r.Len()]
|
|
|
|
s.sessionsMutex.RLock()
|
|
session, ok := s.sessions[hdr.ConnectionID]
|
|
s.sessionsMutex.RUnlock()
|
|
|
|
// a session is only created once the client sent a supported version
|
|
// if we receive a packet for a connection that already has session, it's probably an old packet that was sent by the client before the version was negotiated
|
|
// it is safe to drop it
|
|
if ok && hdr.VersionFlag && !protocol.IsSupportedVersion(hdr.VersionNumber) {
|
|
return nil
|
|
}
|
|
|
|
// Send Version Negotiation Packet if the client is speaking a different protocol version
|
|
if hdr.VersionFlag && !protocol.IsSupportedVersion(hdr.VersionNumber) {
|
|
utils.Infof("Client offered version %d, sending VersionNegotiationPacket", hdr.VersionNumber)
|
|
_, err = conn.WriteToUDP(composeVersionNegotiation(hdr.ConnectionID), remoteAddr)
|
|
return err
|
|
}
|
|
|
|
if !ok {
|
|
if !hdr.VersionFlag {
|
|
_, err = conn.WriteToUDP(writePublicReset(hdr.ConnectionID, hdr.PacketNumber, 0), remoteAddr)
|
|
return err
|
|
}
|
|
version := hdr.VersionNumber
|
|
if !protocol.IsSupportedVersion(version) {
|
|
return errors.New("Server BUG: negotiated version not supported")
|
|
}
|
|
|
|
utils.Infof("Serving new connection: %x, version %d from %v", hdr.ConnectionID, version, remoteAddr)
|
|
session, err = s.newSession(
|
|
&udpConn{conn: conn, currentAddr: remoteAddr},
|
|
version,
|
|
hdr.ConnectionID,
|
|
s.scfg,
|
|
s.streamCallback,
|
|
s.closeCallback,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
go session.run()
|
|
s.sessionsMutex.Lock()
|
|
s.sessions[hdr.ConnectionID] = session
|
|
s.sessionsMutex.Unlock()
|
|
}
|
|
if session == nil {
|
|
// Late packet for closed session
|
|
return nil
|
|
}
|
|
session.handlePacket(&receivedPacket{
|
|
remoteAddr: remoteAddr,
|
|
publicHeader: hdr,
|
|
data: packet[len(packet)-r.Len():],
|
|
rcvTime: rcvTime,
|
|
})
|
|
return nil
|
|
}
|
|
|
|
func (s *Server) closeCallback(id protocol.ConnectionID) {
|
|
s.sessionsMutex.Lock()
|
|
s.sessions[id] = nil
|
|
s.sessionsMutex.Unlock()
|
|
}
|
|
|
|
func composeVersionNegotiation(connectionID protocol.ConnectionID) []byte {
|
|
fullReply := &bytes.Buffer{}
|
|
responsePublicHeader := PublicHeader{
|
|
ConnectionID: connectionID,
|
|
PacketNumber: 1,
|
|
VersionFlag: true,
|
|
}
|
|
err := responsePublicHeader.Write(fullReply, protocol.Version35)
|
|
if err != nil {
|
|
utils.Errorf("error composing version negotiation packet: %s", err.Error())
|
|
}
|
|
fullReply.Write(protocol.SupportedVersionsAsTags)
|
|
return fullReply.Bytes()
|
|
}
|