quic_listen.go 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. // Copyright (C) 2019 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at http://mozilla.org/MPL/2.0/.
  6. // +build go1.12
  7. package connections
  8. import (
  9. "crypto/tls"
  10. "net"
  11. "net/url"
  12. "strings"
  13. "sync"
  14. "sync/atomic"
  15. "time"
  16. "github.com/lucas-clemente/quic-go"
  17. "github.com/syncthing/syncthing/lib/config"
  18. "github.com/syncthing/syncthing/lib/connections/registry"
  19. "github.com/syncthing/syncthing/lib/nat"
  20. "github.com/syncthing/syncthing/lib/stun"
  21. )
  22. func init() {
  23. factory := &quicListenerFactory{}
  24. for _, scheme := range []string{"quic", "quic4", "quic6"} {
  25. listeners[scheme] = factory
  26. }
  27. }
  28. type quicListener struct {
  29. nat atomic.Value
  30. onAddressesChangedNotifier
  31. uri *url.URL
  32. cfg config.Wrapper
  33. tlsCfg *tls.Config
  34. stop chan struct{}
  35. conns chan internalConn
  36. factory listenerFactory
  37. address *url.URL
  38. err error
  39. mut sync.Mutex
  40. }
  41. func (t *quicListener) OnNATTypeChanged(natType stun.NATType) {
  42. if natType != stun.NATUnknown {
  43. l.Infof("%s detected NAT type: %s", t.uri, natType)
  44. }
  45. t.nat.Store(natType)
  46. }
  47. func (t *quicListener) OnExternalAddressChanged(address *stun.Host, via string) {
  48. var uri *url.URL
  49. if address != nil {
  50. copy := *t.uri
  51. uri = &copy
  52. uri.Host = address.TransportAddr()
  53. }
  54. t.mut.Lock()
  55. existingAddress := t.address
  56. t.address = uri
  57. t.mut.Unlock()
  58. if uri != nil && (existingAddress == nil || existingAddress.String() != uri.String()) {
  59. l.Infof("%s resolved external address %s (via %s)", t.uri, uri.String(), via)
  60. t.notifyAddressesChanged(t)
  61. } else if uri == nil && existingAddress != nil {
  62. t.notifyAddressesChanged(t)
  63. }
  64. }
  65. func (t *quicListener) Serve() {
  66. t.mut.Lock()
  67. t.err = nil
  68. t.mut.Unlock()
  69. network := strings.Replace(t.uri.Scheme, "quic", "udp", -1)
  70. packetConn, err := net.ListenPacket(network, t.uri.Host)
  71. if err != nil {
  72. t.mut.Lock()
  73. t.err = err
  74. t.mut.Unlock()
  75. l.Infoln("Listen (BEP/quic):", err)
  76. return
  77. }
  78. defer func() { _ = packetConn.Close() }()
  79. svc, conn := stun.New(t.cfg, t, packetConn)
  80. defer func() { _ = conn.Close() }()
  81. go svc.Serve()
  82. defer svc.Stop()
  83. registry.Register(t.uri.Scheme, conn)
  84. defer registry.Unregister(t.uri.Scheme, conn)
  85. listener, err := quic.Listen(conn, t.tlsCfg, quicConfig)
  86. if err != nil {
  87. t.mut.Lock()
  88. t.err = err
  89. t.mut.Unlock()
  90. l.Infoln("Listen (BEP/quic):", err)
  91. return
  92. }
  93. l.Infof("QUIC listener (%v) starting", packetConn.LocalAddr())
  94. defer l.Infof("QUIC listener (%v) shutting down", packetConn.LocalAddr())
  95. // Accept is forever, so handle stops externally.
  96. go func() {
  97. select {
  98. case <-t.stop:
  99. _ = listener.Close()
  100. }
  101. }()
  102. for {
  103. // Blocks forever, see https://github.com/lucas-clemente/quic-go/issues/1915
  104. session, err := listener.Accept()
  105. select {
  106. case <-t.stop:
  107. if err == nil {
  108. _ = session.Close()
  109. }
  110. return
  111. default:
  112. }
  113. if err != nil {
  114. if err, ok := err.(net.Error); !ok || !err.Timeout() {
  115. l.Warnln("Listen (BEP/quic): Accepting connection:", err)
  116. }
  117. continue
  118. }
  119. l.Debugln("connect from", session.RemoteAddr())
  120. // Accept blocks forever, give it 10s to do it's thing.
  121. ok := make(chan struct{})
  122. go func() {
  123. select {
  124. case <-ok:
  125. return
  126. case <-t.stop:
  127. _ = session.Close()
  128. case <-time.After(10 * time.Second):
  129. l.Debugln("timed out waiting for AcceptStream on", session.RemoteAddr())
  130. _ = session.Close()
  131. }
  132. }()
  133. stream, err := session.AcceptStream()
  134. close(ok)
  135. if err != nil {
  136. l.Debugln("failed to accept stream from", session.RemoteAddr(), err.Error())
  137. _ = session.Close()
  138. continue
  139. }
  140. t.conns <- internalConn{&quicTlsConn{session, stream, nil}, connTypeQUICServer, quicPriority}
  141. }
  142. }
  143. func (t *quicListener) Stop() {
  144. close(t.stop)
  145. }
  146. func (t *quicListener) URI() *url.URL {
  147. return t.uri
  148. }
  149. func (t *quicListener) WANAddresses() []*url.URL {
  150. uris := t.LANAddresses()
  151. t.mut.Lock()
  152. if t.address != nil {
  153. uris = append(uris, t.address)
  154. }
  155. t.mut.Unlock()
  156. return uris
  157. }
  158. func (t *quicListener) LANAddresses() []*url.URL {
  159. return []*url.URL{t.uri}
  160. }
  161. func (t *quicListener) Error() error {
  162. t.mut.Lock()
  163. err := t.err
  164. t.mut.Unlock()
  165. return err
  166. }
  167. func (t *quicListener) String() string {
  168. return t.uri.String()
  169. }
  170. func (t *quicListener) Factory() listenerFactory {
  171. return t.factory
  172. }
  173. func (t *quicListener) NATType() string {
  174. v := t.nat.Load().(stun.NATType)
  175. if v == stun.NATUnknown || v == stun.NATError {
  176. return "unknown"
  177. }
  178. return v.String()
  179. }
  180. type quicListenerFactory struct{}
  181. func (f *quicListenerFactory) Valid(config.Configuration) error {
  182. return nil
  183. }
  184. func (f *quicListenerFactory) New(uri *url.URL, cfg config.Wrapper, tlsCfg *tls.Config, conns chan internalConn, natService *nat.Service) genericListener {
  185. l := &quicListener{
  186. uri: fixupPort(uri, config.DefaultQUICPort),
  187. cfg: cfg,
  188. tlsCfg: tlsCfg,
  189. conns: conns,
  190. stop: make(chan struct{}),
  191. factory: f,
  192. }
  193. l.nat.Store(stun.NATUnknown)
  194. return l
  195. }
  196. func (quicListenerFactory) Enabled(cfg config.Configuration) bool {
  197. return true
  198. }