mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-05 10:20:36 +02:00
339 lines
9.5 KiB
Go
339 lines
9.5 KiB
Go
|
package quic
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"crypto/tls"
|
||
|
"errors"
|
||
|
"net"
|
||
|
"sync"
|
||
|
"time"
|
||
|
|
||
|
"github.com/lucas-clemente/quic-go/crypto"
|
||
|
"github.com/lucas-clemente/quic-go/handshake"
|
||
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
||
|
"github.com/lucas-clemente/quic-go/protocol"
|
||
|
"github.com/lucas-clemente/quic-go/qerr"
|
||
|
)
|
||
|
|
||
|
// packetHandler handles packets
|
||
|
type packetHandler interface {
|
||
|
Session
|
||
|
handlePacket(*receivedPacket)
|
||
|
run() error
|
||
|
closeRemote(error)
|
||
|
}
|
||
|
|
||
|
// A Listener of QUIC
|
||
|
type server struct {
|
||
|
tlsConf *tls.Config
|
||
|
config *Config
|
||
|
|
||
|
conn net.PacketConn
|
||
|
|
||
|
certChain crypto.CertChain
|
||
|
scfg *handshake.ServerConfig
|
||
|
|
||
|
sessions map[protocol.ConnectionID]packetHandler
|
||
|
sessionsMutex sync.RWMutex
|
||
|
deleteClosedSessionsAfter time.Duration
|
||
|
|
||
|
serverError error
|
||
|
sessionQueue chan Session
|
||
|
errorChan chan struct{}
|
||
|
|
||
|
newSession func(conn connection, v protocol.VersionNumber, connectionID protocol.ConnectionID, sCfg *handshake.ServerConfig, tlsConf *tls.Config, config *Config) (packetHandler, <-chan handshakeEvent, error)
|
||
|
}
|
||
|
|
||
|
var _ Listener = &server{}
|
||
|
|
||
|
// ListenAddr creates a QUIC server listening on a given address.
|
||
|
// The listener is not active until Serve() is called.
|
||
|
// The tls.Config must not be nil, the quic.Config may be nil.
|
||
|
func ListenAddr(addr string, tlsConf *tls.Config, config *Config) (Listener, error) {
|
||
|
udpAddr, err := net.ResolveUDPAddr("udp", addr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
conn, err := net.ListenUDP("udp", udpAddr)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return Listen(conn, tlsConf, config)
|
||
|
}
|
||
|
|
||
|
// Listen listens for QUIC connections on a given net.PacketConn.
|
||
|
// The listener is not active until Serve() is called.
|
||
|
// The tls.Config must not be nil, the quic.Config may be nil.
|
||
|
func Listen(conn net.PacketConn, tlsConf *tls.Config, config *Config) (Listener, error) {
|
||
|
certChain := crypto.NewCertChain(tlsConf)
|
||
|
kex, err := crypto.NewCurve25519KEX()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
scfg, err := handshake.NewServerConfig(kex, certChain)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
s := &server{
|
||
|
conn: conn,
|
||
|
tlsConf: tlsConf,
|
||
|
config: populateServerConfig(config),
|
||
|
certChain: certChain,
|
||
|
scfg: scfg,
|
||
|
sessions: map[protocol.ConnectionID]packetHandler{},
|
||
|
newSession: newSession,
|
||
|
deleteClosedSessionsAfter: protocol.ClosedSessionDeleteTimeout,
|
||
|
sessionQueue: make(chan Session, 5),
|
||
|
errorChan: make(chan struct{}),
|
||
|
}
|
||
|
go s.serve()
|
||
|
return s, nil
|
||
|
}
|
||
|
|
||
|
var defaultAcceptSTK = func(clientAddr net.Addr, stk *STK) bool {
|
||
|
if stk == nil {
|
||
|
return false
|
||
|
}
|
||
|
if time.Now().After(stk.sentTime.Add(protocol.STKExpiryTime)) {
|
||
|
return false
|
||
|
}
|
||
|
var sourceAddr string
|
||
|
if udpAddr, ok := clientAddr.(*net.UDPAddr); ok {
|
||
|
sourceAddr = udpAddr.IP.String()
|
||
|
} else {
|
||
|
sourceAddr = clientAddr.String()
|
||
|
}
|
||
|
return sourceAddr == stk.remoteAddr
|
||
|
}
|
||
|
|
||
|
// populateServerConfig populates fields in the quic.Config with their default values, if none are set
|
||
|
// it may be called with nil
|
||
|
func populateServerConfig(config *Config) *Config {
|
||
|
if config == nil {
|
||
|
config = &Config{}
|
||
|
}
|
||
|
versions := config.Versions
|
||
|
if len(versions) == 0 {
|
||
|
versions = protocol.SupportedVersions
|
||
|
}
|
||
|
|
||
|
vsa := defaultAcceptSTK
|
||
|
if config.AcceptSTK != nil {
|
||
|
vsa = config.AcceptSTK
|
||
|
}
|
||
|
|
||
|
handshakeTimeout := protocol.DefaultHandshakeTimeout
|
||
|
if config.HandshakeTimeout != 0 {
|
||
|
handshakeTimeout = config.HandshakeTimeout
|
||
|
}
|
||
|
|
||
|
maxReceiveStreamFlowControlWindow := config.MaxReceiveStreamFlowControlWindow
|
||
|
if maxReceiveStreamFlowControlWindow == 0 {
|
||
|
maxReceiveStreamFlowControlWindow = protocol.DefaultMaxReceiveStreamFlowControlWindowServer
|
||
|
}
|
||
|
maxReceiveConnectionFlowControlWindow := config.MaxReceiveConnectionFlowControlWindow
|
||
|
if maxReceiveConnectionFlowControlWindow == 0 {
|
||
|
maxReceiveConnectionFlowControlWindow = protocol.DefaultMaxReceiveConnectionFlowControlWindowServer
|
||
|
}
|
||
|
|
||
|
return &Config{
|
||
|
Versions: versions,
|
||
|
HandshakeTimeout: handshakeTimeout,
|
||
|
AcceptSTK: vsa,
|
||
|
MaxReceiveStreamFlowControlWindow: maxReceiveStreamFlowControlWindow,
|
||
|
MaxReceiveConnectionFlowControlWindow: maxReceiveConnectionFlowControlWindow,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// serve listens on an existing PacketConn
|
||
|
func (s *server) serve() {
|
||
|
for {
|
||
|
data := getPacketBuffer()
|
||
|
data = data[:protocol.MaxReceivePacketSize]
|
||
|
// The packet size should not exceed protocol.MaxReceivePacketSize bytes
|
||
|
// If it does, we only read a truncated packet, which will then end up undecryptable
|
||
|
n, remoteAddr, err := s.conn.ReadFrom(data)
|
||
|
if err != nil {
|
||
|
s.serverError = err
|
||
|
close(s.errorChan)
|
||
|
_ = s.Close()
|
||
|
return
|
||
|
}
|
||
|
data = data[:n]
|
||
|
if err := s.handlePacket(s.conn, remoteAddr, data); err != nil {
|
||
|
utils.Errorf("error handling packet: %s", err.Error())
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Accept returns newly openend sessions
|
||
|
func (s *server) Accept() (Session, error) {
|
||
|
var sess Session
|
||
|
select {
|
||
|
case sess = <-s.sessionQueue:
|
||
|
return sess, nil
|
||
|
case <-s.errorChan:
|
||
|
return nil, s.serverError
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// 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()
|
||
|
|
||
|
if s.conn == nil {
|
||
|
return nil
|
||
|
}
|
||
|
return s.conn.Close()
|
||
|
}
|
||
|
|
||
|
// Addr returns the server's network address
|
||
|
func (s *server) Addr() net.Addr {
|
||
|
return s.conn.LocalAddr()
|
||
|
}
|
||
|
|
||
|
func (s *server) handlePacket(pconn net.PacketConn, remoteAddr net.Addr, packet []byte) error {
|
||
|
rcvTime := time.Now()
|
||
|
|
||
|
r := bytes.NewReader(packet)
|
||
|
hdr, err := ParsePublicHeader(r, protocol.PerspectiveClient)
|
||
|
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()
|
||
|
|
||
|
// ignore all Public Reset packets
|
||
|
if hdr.ResetFlag {
|
||
|
if ok {
|
||
|
var pr *publicReset
|
||
|
pr, err = parsePublicReset(r)
|
||
|
if err != nil {
|
||
|
utils.Infof("Received a Public Reset for connection %x. An error occurred parsing the packet.")
|
||
|
} else {
|
||
|
utils.Infof("Received a Public Reset for connection %x, rejected packet number: 0x%x.", hdr.ConnectionID, pr.rejectedPacketNumber)
|
||
|
}
|
||
|
} else {
|
||
|
utils.Infof("Received Public Reset for unknown connection %x.", hdr.ConnectionID)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// 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(s.config.Versions, hdr.VersionNumber) {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Send Version Negotiation Packet if the client is speaking a different protocol version
|
||
|
if hdr.VersionFlag && !protocol.IsSupportedVersion(s.config.Versions, hdr.VersionNumber) {
|
||
|
// drop packets that are too small to be valid first packets
|
||
|
if len(packet) < protocol.ClientHelloMinimumSize+len(hdr.Raw) {
|
||
|
return errors.New("dropping small packet with unknown version")
|
||
|
}
|
||
|
utils.Infof("Client offered version %d, sending VersionNegotiationPacket", hdr.VersionNumber)
|
||
|
_, err = pconn.WriteTo(composeVersionNegotiation(hdr.ConnectionID, s.config.Versions), remoteAddr)
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if !ok {
|
||
|
if !hdr.VersionFlag {
|
||
|
_, err = pconn.WriteTo(writePublicReset(hdr.ConnectionID, hdr.PacketNumber, 0), remoteAddr)
|
||
|
return err
|
||
|
}
|
||
|
version := hdr.VersionNumber
|
||
|
if !protocol.IsSupportedVersion(s.config.Versions, version) {
|
||
|
return errors.New("Server BUG: negotiated version not supported")
|
||
|
}
|
||
|
|
||
|
utils.Infof("Serving new connection: %x, version %d from %v", hdr.ConnectionID, version, remoteAddr)
|
||
|
var handshakeChan <-chan handshakeEvent
|
||
|
session, handshakeChan, err = s.newSession(
|
||
|
&conn{pconn: pconn, currentAddr: remoteAddr},
|
||
|
version,
|
||
|
hdr.ConnectionID,
|
||
|
s.scfg,
|
||
|
s.tlsConf,
|
||
|
s.config,
|
||
|
)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
s.sessionsMutex.Lock()
|
||
|
s.sessions[hdr.ConnectionID] = session
|
||
|
s.sessionsMutex.Unlock()
|
||
|
|
||
|
go func() {
|
||
|
// session.run() returns as soon as the session is closed
|
||
|
_ = session.run()
|
||
|
s.removeConnection(hdr.ConnectionID)
|
||
|
}()
|
||
|
|
||
|
go func() {
|
||
|
for {
|
||
|
ev := <-handshakeChan
|
||
|
if ev.err != nil {
|
||
|
return
|
||
|
}
|
||
|
if ev.encLevel == protocol.EncryptionForwardSecure {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
s.sessionQueue <- session
|
||
|
}()
|
||
|
}
|
||
|
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) removeConnection(id protocol.ConnectionID) {
|
||
|
s.sessionsMutex.Lock()
|
||
|
s.sessions[id] = nil
|
||
|
s.sessionsMutex.Unlock()
|
||
|
|
||
|
time.AfterFunc(s.deleteClosedSessionsAfter, func() {
|
||
|
s.sessionsMutex.Lock()
|
||
|
delete(s.sessions, id)
|
||
|
s.sessionsMutex.Unlock()
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func composeVersionNegotiation(connectionID protocol.ConnectionID, versions []protocol.VersionNumber) []byte {
|
||
|
fullReply := &bytes.Buffer{}
|
||
|
responsePublicHeader := PublicHeader{
|
||
|
ConnectionID: connectionID,
|
||
|
PacketNumber: 1,
|
||
|
VersionFlag: true,
|
||
|
}
|
||
|
err := responsePublicHeader.Write(fullReply, protocol.VersionWhatever, protocol.PerspectiveServer)
|
||
|
if err != nil {
|
||
|
utils.Errorf("error composing version negotiation packet: %s", err.Error())
|
||
|
}
|
||
|
for _, v := range versions {
|
||
|
utils.WriteUint32(fullReply, protocol.VersionNumberToTag(v))
|
||
|
}
|
||
|
return fullReply.Bytes()
|
||
|
}
|