server.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. package quic
  2. import (
  3. "bytes"
  4. "crypto/tls"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "net"
  9. "sync"
  10. "time"
  11. "github.com/lucas-clemente/quic-go/internal/handshake"
  12. "github.com/lucas-clemente/quic-go/internal/protocol"
  13. "github.com/lucas-clemente/quic-go/internal/utils"
  14. "github.com/lucas-clemente/quic-go/internal/wire"
  15. )
  16. // packetHandler handles packets
  17. type packetHandler interface {
  18. handlePacket(*receivedPacket)
  19. io.Closer
  20. destroy(error)
  21. GetPerspective() protocol.Perspective
  22. }
  23. type unknownPacketHandler interface {
  24. handlePacket(*receivedPacket)
  25. closeWithError(error) error
  26. }
  27. type packetHandlerManager interface {
  28. Add(protocol.ConnectionID, packetHandler)
  29. Retire(protocol.ConnectionID)
  30. Remove(protocol.ConnectionID)
  31. SetServer(unknownPacketHandler)
  32. CloseServer()
  33. }
  34. type quicSession interface {
  35. Session
  36. handlePacket(*receivedPacket)
  37. GetVersion() protocol.VersionNumber
  38. run() error
  39. destroy(error)
  40. closeRemote(error)
  41. }
  42. type sessionRunner interface {
  43. onHandshakeComplete(Session)
  44. retireConnectionID(protocol.ConnectionID)
  45. removeConnectionID(protocol.ConnectionID)
  46. }
  47. type runner struct {
  48. onHandshakeCompleteImpl func(Session)
  49. retireConnectionIDImpl func(protocol.ConnectionID)
  50. removeConnectionIDImpl func(protocol.ConnectionID)
  51. }
  52. func (r *runner) onHandshakeComplete(s Session) { r.onHandshakeCompleteImpl(s) }
  53. func (r *runner) retireConnectionID(c protocol.ConnectionID) { r.retireConnectionIDImpl(c) }
  54. func (r *runner) removeConnectionID(c protocol.ConnectionID) { r.removeConnectionIDImpl(c) }
  55. var _ sessionRunner = &runner{}
  56. // A Listener of QUIC
  57. type server struct {
  58. mutex sync.Mutex
  59. tlsConf *tls.Config
  60. config *Config
  61. conn net.PacketConn
  62. // If the server is started with ListenAddr, we create a packet conn.
  63. // If it is started with Listen, we take a packet conn as a parameter.
  64. createdPacketConn bool
  65. cookieGenerator *handshake.CookieGenerator
  66. sessionHandler packetHandlerManager
  67. // set as a member, so they can be set in the tests
  68. newSession func(connection, sessionRunner, protocol.ConnectionID /* original connection ID */, protocol.ConnectionID /* destination connection ID */, protocol.ConnectionID /* source connection ID */, *Config, *tls.Config, *handshake.TransportParameters, utils.Logger, protocol.VersionNumber) (quicSession, error)
  69. serverError error
  70. errorChan chan struct{}
  71. closed bool
  72. sessionQueue chan Session
  73. sessionRunner sessionRunner
  74. logger utils.Logger
  75. }
  76. var _ Listener = &server{}
  77. var _ unknownPacketHandler = &server{}
  78. // ListenAddr creates a QUIC server listening on a given address.
  79. // The tls.Config must not be nil and must contain a certificate configuration.
  80. // The quic.Config may be nil, in that case the default values will be used.
  81. func ListenAddr(addr string, tlsConf *tls.Config, config *Config) (Listener, error) {
  82. udpAddr, err := net.ResolveUDPAddr("udp", addr)
  83. if err != nil {
  84. return nil, err
  85. }
  86. conn, err := net.ListenUDP("udp", udpAddr)
  87. if err != nil {
  88. return nil, err
  89. }
  90. serv, err := listen(conn, tlsConf, config)
  91. if err != nil {
  92. return nil, err
  93. }
  94. serv.createdPacketConn = true
  95. return serv, nil
  96. }
  97. // Listen listens for QUIC connections on a given net.PacketConn.
  98. // A single PacketConn only be used for a single call to Listen.
  99. // The PacketConn can be used for simultaneous calls to Dial.
  100. // QUIC connection IDs are used for demultiplexing the different connections.
  101. // The tls.Config must not be nil and must contain a certificate configuration.
  102. // The quic.Config may be nil, in that case the default values will be used.
  103. func Listen(conn net.PacketConn, tlsConf *tls.Config, config *Config) (Listener, error) {
  104. return listen(conn, tlsConf, config)
  105. }
  106. func listen(conn net.PacketConn, tlsConf *tls.Config, config *Config) (*server, error) {
  107. // TODO(#1655): only require that tls.Config.Certificates or tls.Config.GetCertificate is set
  108. if tlsConf == nil || len(tlsConf.Certificates) == 0 {
  109. return nil, errors.New("quic: Certificates not set in tls.Config")
  110. }
  111. config = populateServerConfig(config)
  112. for _, v := range config.Versions {
  113. if !protocol.IsValidVersion(v) {
  114. return nil, fmt.Errorf("%s is not a valid QUIC version", v)
  115. }
  116. }
  117. sessionHandler, err := getMultiplexer().AddConn(conn, config.ConnectionIDLength)
  118. if err != nil {
  119. return nil, err
  120. }
  121. s := &server{
  122. conn: conn,
  123. tlsConf: tlsConf,
  124. config: config,
  125. sessionHandler: sessionHandler,
  126. sessionQueue: make(chan Session, 5),
  127. errorChan: make(chan struct{}),
  128. newSession: newSession,
  129. logger: utils.DefaultLogger.WithPrefix("server"),
  130. }
  131. if err := s.setup(); err != nil {
  132. return nil, err
  133. }
  134. sessionHandler.SetServer(s)
  135. s.logger.Debugf("Listening for %s connections on %s", conn.LocalAddr().Network(), conn.LocalAddr().String())
  136. return s, nil
  137. }
  138. func (s *server) setup() error {
  139. s.sessionRunner = &runner{
  140. onHandshakeCompleteImpl: func(sess Session) { s.sessionQueue <- sess },
  141. retireConnectionIDImpl: s.sessionHandler.Retire,
  142. removeConnectionIDImpl: s.sessionHandler.Remove,
  143. }
  144. cookieGenerator, err := handshake.NewCookieGenerator()
  145. if err != nil {
  146. return err
  147. }
  148. s.cookieGenerator = cookieGenerator
  149. return nil
  150. }
  151. var defaultAcceptCookie = func(clientAddr net.Addr, cookie *Cookie) bool {
  152. if cookie == nil {
  153. return false
  154. }
  155. if time.Now().After(cookie.SentTime.Add(protocol.CookieExpiryTime)) {
  156. return false
  157. }
  158. var sourceAddr string
  159. if udpAddr, ok := clientAddr.(*net.UDPAddr); ok {
  160. sourceAddr = udpAddr.IP.String()
  161. } else {
  162. sourceAddr = clientAddr.String()
  163. }
  164. return sourceAddr == cookie.RemoteAddr
  165. }
  166. // populateServerConfig populates fields in the quic.Config with their default values, if none are set
  167. // it may be called with nil
  168. func populateServerConfig(config *Config) *Config {
  169. if config == nil {
  170. config = &Config{}
  171. }
  172. versions := config.Versions
  173. if len(versions) == 0 {
  174. versions = protocol.SupportedVersions
  175. }
  176. vsa := defaultAcceptCookie
  177. if config.AcceptCookie != nil {
  178. vsa = config.AcceptCookie
  179. }
  180. handshakeTimeout := protocol.DefaultHandshakeTimeout
  181. if config.HandshakeTimeout != 0 {
  182. handshakeTimeout = config.HandshakeTimeout
  183. }
  184. idleTimeout := protocol.DefaultIdleTimeout
  185. if config.IdleTimeout != 0 {
  186. idleTimeout = config.IdleTimeout
  187. }
  188. maxReceiveStreamFlowControlWindow := config.MaxReceiveStreamFlowControlWindow
  189. if maxReceiveStreamFlowControlWindow == 0 {
  190. maxReceiveStreamFlowControlWindow = protocol.DefaultMaxReceiveStreamFlowControlWindow
  191. }
  192. maxReceiveConnectionFlowControlWindow := config.MaxReceiveConnectionFlowControlWindow
  193. if maxReceiveConnectionFlowControlWindow == 0 {
  194. maxReceiveConnectionFlowControlWindow = protocol.DefaultMaxReceiveConnectionFlowControlWindow
  195. }
  196. maxIncomingStreams := config.MaxIncomingStreams
  197. if maxIncomingStreams == 0 {
  198. maxIncomingStreams = protocol.DefaultMaxIncomingStreams
  199. } else if maxIncomingStreams < 0 {
  200. maxIncomingStreams = 0
  201. }
  202. maxIncomingUniStreams := config.MaxIncomingUniStreams
  203. if maxIncomingUniStreams == 0 {
  204. maxIncomingUniStreams = protocol.DefaultMaxIncomingUniStreams
  205. } else if maxIncomingUniStreams < 0 {
  206. maxIncomingUniStreams = 0
  207. }
  208. connIDLen := config.ConnectionIDLength
  209. if connIDLen == 0 {
  210. connIDLen = protocol.DefaultConnectionIDLength
  211. }
  212. return &Config{
  213. Versions: versions,
  214. HandshakeTimeout: handshakeTimeout,
  215. IdleTimeout: idleTimeout,
  216. AcceptCookie: vsa,
  217. KeepAlive: config.KeepAlive,
  218. MaxReceiveStreamFlowControlWindow: maxReceiveStreamFlowControlWindow,
  219. MaxReceiveConnectionFlowControlWindow: maxReceiveConnectionFlowControlWindow,
  220. MaxIncomingStreams: maxIncomingStreams,
  221. MaxIncomingUniStreams: maxIncomingUniStreams,
  222. ConnectionIDLength: connIDLen,
  223. }
  224. }
  225. // Accept returns newly openend sessions
  226. func (s *server) Accept() (Session, error) {
  227. var sess Session
  228. select {
  229. case sess = <-s.sessionQueue:
  230. return sess, nil
  231. case <-s.errorChan:
  232. return nil, s.serverError
  233. }
  234. }
  235. // Close the server
  236. func (s *server) Close() error {
  237. s.mutex.Lock()
  238. defer s.mutex.Unlock()
  239. if s.closed {
  240. return nil
  241. }
  242. return s.closeWithMutex()
  243. }
  244. func (s *server) closeWithMutex() error {
  245. s.sessionHandler.CloseServer()
  246. if s.serverError == nil {
  247. s.serverError = errors.New("server closed")
  248. }
  249. var err error
  250. // If the server was started with ListenAddr, we created the packet conn.
  251. // We need to close it in order to make the go routine reading from that conn return.
  252. if s.createdPacketConn {
  253. err = s.conn.Close()
  254. }
  255. s.closed = true
  256. close(s.errorChan)
  257. return err
  258. }
  259. func (s *server) closeWithError(e error) error {
  260. s.mutex.Lock()
  261. defer s.mutex.Unlock()
  262. if s.closed {
  263. return nil
  264. }
  265. s.serverError = e
  266. return s.closeWithMutex()
  267. }
  268. // Addr returns the server's network address
  269. func (s *server) Addr() net.Addr {
  270. return s.conn.LocalAddr()
  271. }
  272. func (s *server) handlePacket(p *receivedPacket) {
  273. hdr := p.hdr
  274. // send a Version Negotiation Packet if the client is speaking a different protocol version
  275. if !protocol.IsSupportedVersion(s.config.Versions, hdr.Version) {
  276. go s.sendVersionNegotiationPacket(p)
  277. return
  278. }
  279. if hdr.Type == protocol.PacketTypeInitial {
  280. go s.handleInitial(p)
  281. }
  282. // TODO(#943): send Stateless Reset
  283. }
  284. func (s *server) handleInitial(p *receivedPacket) {
  285. // TODO: add a check that DestConnID == SrcConnID
  286. s.logger.Debugf("<- Received Initial packet.")
  287. sess, connID, err := s.handleInitialImpl(p)
  288. if err != nil {
  289. s.logger.Errorf("Error occurred handling initial packet: %s", err)
  290. return
  291. }
  292. if sess == nil { // a retry was done
  293. return
  294. }
  295. serverSession := newServerSession(sess, s.config, s.logger)
  296. s.sessionHandler.Add(connID, serverSession)
  297. }
  298. func (s *server) handleInitialImpl(p *receivedPacket) (quicSession, protocol.ConnectionID, error) {
  299. hdr := p.hdr
  300. if len(hdr.Token) == 0 && hdr.DestConnectionID.Len() < protocol.MinConnectionIDLenInitial {
  301. return nil, nil, errors.New("dropping Initial packet with too short connection ID")
  302. }
  303. if len(p.data) < protocol.MinInitialPacketSize {
  304. return nil, nil, errors.New("dropping too small Initial packet")
  305. }
  306. var cookie *Cookie
  307. var origDestConnectionID protocol.ConnectionID
  308. if len(hdr.Token) > 0 {
  309. c, err := s.cookieGenerator.DecodeToken(hdr.Token)
  310. if err == nil {
  311. cookie = &Cookie{
  312. RemoteAddr: c.RemoteAddr,
  313. SentTime: c.SentTime,
  314. }
  315. origDestConnectionID = c.OriginalDestConnectionID
  316. }
  317. }
  318. if !s.config.AcceptCookie(p.remoteAddr, cookie) {
  319. // Log the Initial packet now.
  320. // If no Retry is sent, the packet will be logged by the session.
  321. (&wire.ExtendedHeader{Header: *p.hdr}).Log(s.logger)
  322. return nil, nil, s.sendRetry(p.remoteAddr, hdr)
  323. }
  324. connID, err := protocol.GenerateConnectionID(s.config.ConnectionIDLength)
  325. if err != nil {
  326. return nil, nil, err
  327. }
  328. s.logger.Debugf("Changing connection ID to %s.", connID)
  329. sess, err := s.createNewSession(
  330. p.remoteAddr,
  331. origDestConnectionID,
  332. hdr.DestConnectionID,
  333. hdr.SrcConnectionID,
  334. connID,
  335. hdr.Version,
  336. )
  337. if err != nil {
  338. return nil, nil, err
  339. }
  340. sess.handlePacket(p)
  341. return sess, connID, nil
  342. }
  343. func (s *server) createNewSession(
  344. remoteAddr net.Addr,
  345. origDestConnID protocol.ConnectionID,
  346. clientDestConnID protocol.ConnectionID,
  347. destConnID protocol.ConnectionID,
  348. srcConnID protocol.ConnectionID,
  349. version protocol.VersionNumber,
  350. ) (quicSession, error) {
  351. params := &handshake.TransportParameters{
  352. InitialMaxStreamDataBidiLocal: protocol.InitialMaxStreamData,
  353. InitialMaxStreamDataBidiRemote: protocol.InitialMaxStreamData,
  354. InitialMaxStreamDataUni: protocol.InitialMaxStreamData,
  355. InitialMaxData: protocol.InitialMaxData,
  356. IdleTimeout: s.config.IdleTimeout,
  357. MaxBidiStreams: uint64(s.config.MaxIncomingStreams),
  358. MaxUniStreams: uint64(s.config.MaxIncomingUniStreams),
  359. DisableMigration: true,
  360. // TODO(#855): generate a real token
  361. StatelessResetToken: bytes.Repeat([]byte{42}, 16),
  362. OriginalConnectionID: origDestConnID,
  363. }
  364. sess, err := s.newSession(
  365. &conn{pconn: s.conn, currentAddr: remoteAddr},
  366. s.sessionRunner,
  367. clientDestConnID,
  368. destConnID,
  369. srcConnID,
  370. s.config,
  371. s.tlsConf,
  372. params,
  373. s.logger,
  374. version,
  375. )
  376. if err != nil {
  377. return nil, err
  378. }
  379. go sess.run()
  380. return sess, nil
  381. }
  382. func (s *server) sendRetry(remoteAddr net.Addr, hdr *wire.Header) error {
  383. token, err := s.cookieGenerator.NewToken(remoteAddr, hdr.DestConnectionID)
  384. if err != nil {
  385. return err
  386. }
  387. connID, err := protocol.GenerateConnectionID(s.config.ConnectionIDLength)
  388. if err != nil {
  389. return err
  390. }
  391. replyHdr := &wire.ExtendedHeader{}
  392. replyHdr.IsLongHeader = true
  393. replyHdr.Type = protocol.PacketTypeRetry
  394. replyHdr.Version = hdr.Version
  395. replyHdr.SrcConnectionID = connID
  396. replyHdr.DestConnectionID = hdr.SrcConnectionID
  397. replyHdr.OrigDestConnectionID = hdr.DestConnectionID
  398. replyHdr.Token = token
  399. s.logger.Debugf("Changing connection ID to %s.\n-> Sending Retry", connID)
  400. replyHdr.Log(s.logger)
  401. buf := &bytes.Buffer{}
  402. if err := replyHdr.Write(buf, hdr.Version); err != nil {
  403. return err
  404. }
  405. if _, err := s.conn.WriteTo(buf.Bytes(), remoteAddr); err != nil {
  406. s.logger.Debugf("Error sending Retry: %s", err)
  407. }
  408. return nil
  409. }
  410. func (s *server) sendVersionNegotiationPacket(p *receivedPacket) {
  411. hdr := p.hdr
  412. s.logger.Debugf("Client offered version %s, sending Version Negotiation", hdr.Version)
  413. data, err := wire.ComposeVersionNegotiation(hdr.SrcConnectionID, hdr.DestConnectionID, s.config.Versions)
  414. if err != nil {
  415. s.logger.Debugf("Error composing Version Negotiation: %s", err)
  416. return
  417. }
  418. if _, err := s.conn.WriteTo(data, p.remoteAddr); err != nil {
  419. s.logger.Debugf("Error sending Version Negotiation: %s", err)
  420. }
  421. }