dynamic.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. // Copyright (C) 2015 Audrius Butkevicius and Contributors (see the CONTRIBUTORS file).
  2. package client
  3. import (
  4. "context"
  5. "crypto/tls"
  6. "encoding/json"
  7. "errors"
  8. "fmt"
  9. "net/http"
  10. "net/url"
  11. "slices"
  12. "sync"
  13. "time"
  14. "github.com/syncthing/syncthing/lib/osutil"
  15. "github.com/syncthing/syncthing/lib/rand"
  16. "github.com/syncthing/syncthing/lib/relay/protocol"
  17. )
  18. type dynamicClient struct {
  19. commonClient
  20. pooladdr *url.URL
  21. certs []tls.Certificate
  22. timeout time.Duration
  23. mut sync.RWMutex // Protects client.
  24. client *staticClient
  25. }
  26. func newDynamicClient(uri *url.URL, certs []tls.Certificate, invitations chan protocol.SessionInvitation, timeout time.Duration) *dynamicClient {
  27. c := &dynamicClient{
  28. pooladdr: uri,
  29. certs: certs,
  30. timeout: timeout,
  31. }
  32. c.commonClient = newCommonClient(invitations, c.serve, fmt.Sprintf("dynamicClient@%p", c))
  33. return c
  34. }
  35. func (c *dynamicClient) serve(ctx context.Context) error {
  36. uri := *c.pooladdr
  37. // Trim off the `dynamic+` prefix
  38. uri.Scheme = uri.Scheme[8:]
  39. l.Debugln(c, "looking up dynamic relays")
  40. req, err := http.NewRequestWithContext(ctx, "GET", uri.String(), nil)
  41. if err != nil {
  42. l.Debugln(c, "failed to lookup dynamic relays", err)
  43. return err
  44. }
  45. data, err := http.DefaultClient.Do(req)
  46. if err != nil {
  47. l.Debugln(c, "failed to lookup dynamic relays", err)
  48. return err
  49. }
  50. var ann dynamicAnnouncement
  51. err = json.NewDecoder(data.Body).Decode(&ann)
  52. data.Body.Close()
  53. if err != nil {
  54. l.Debugln(c, "failed to lookup dynamic relays", err)
  55. return err
  56. }
  57. var addrs []string
  58. for _, relayAnn := range ann.Relays {
  59. ruri, err := url.Parse(relayAnn.URL)
  60. if err != nil {
  61. l.Debugln(c, "failed to parse dynamic relay address", relayAnn.URL, err)
  62. continue
  63. }
  64. l.Debugln(c, "found", ruri)
  65. addrs = append(addrs, ruri.String())
  66. }
  67. for _, addr := range relayAddressesOrder(ctx, addrs) {
  68. select {
  69. case <-ctx.Done():
  70. l.Debugln(c, "stopping")
  71. return nil
  72. default:
  73. ruri, err := url.Parse(addr)
  74. if err != nil {
  75. l.Debugln(c, "skipping relay", addr, err)
  76. continue
  77. }
  78. client := newStaticClient(ruri, c.certs, c.invitations, c.timeout)
  79. c.mut.Lock()
  80. c.client = client
  81. c.mut.Unlock()
  82. err = c.client.Serve(ctx)
  83. l.Debugf("Disconnected from %s://%s: %v", c.client.URI().Scheme, c.client.URI().Host, err)
  84. c.mut.Lock()
  85. c.client = nil
  86. c.mut.Unlock()
  87. }
  88. }
  89. l.Debugln(c, "could not find a connectable relay")
  90. return errors.New("could not find a connectable relay")
  91. }
  92. func (c *dynamicClient) Error() error {
  93. c.mut.RLock()
  94. defer c.mut.RUnlock()
  95. if c.client == nil {
  96. return c.commonClient.Error()
  97. }
  98. return c.client.Error()
  99. }
  100. func (c *dynamicClient) String() string {
  101. return fmt.Sprintf("DynamicClient:%p:%s@%s", c, c.URI(), c.pooladdr)
  102. }
  103. func (c *dynamicClient) URI() *url.URL {
  104. c.mut.RLock()
  105. defer c.mut.RUnlock()
  106. if c.client == nil {
  107. return nil
  108. }
  109. return c.client.URI()
  110. }
  111. // This is the announcement received from the relay server;
  112. // {"relays": [{"url": "relay://10.20.30.40:5060"}, ...]}
  113. type dynamicAnnouncement struct {
  114. Relays []struct {
  115. URL string
  116. }
  117. }
  118. // relayAddressesOrder checks the latency to each relay, rounds latency down to
  119. // the closest 50ms, and puts them in buckets of 50ms latency ranges. Then
  120. // shuffles each bucket, and returns all addresses starting with the ones from
  121. // the lowest latency bucket, ending with the highest latency bucket.
  122. func relayAddressesOrder(ctx context.Context, input []string) []string {
  123. buckets := make(map[int][]string)
  124. for _, relay := range input {
  125. latency, err := osutil.GetLatencyForURL(ctx, relay)
  126. if err != nil {
  127. latency = time.Hour
  128. }
  129. id := int(latency/time.Millisecond) / 50
  130. buckets[id] = append(buckets[id], relay)
  131. select {
  132. case <-ctx.Done():
  133. return nil
  134. default:
  135. }
  136. }
  137. var ids []int
  138. for id, bucket := range buckets {
  139. rand.Shuffle(bucket)
  140. ids = append(ids, id)
  141. }
  142. slices.Sort(ids)
  143. addresses := make([]string, 0, len(input))
  144. for _, id := range ids {
  145. addresses = append(addresses, buckets[id]...)
  146. }
  147. return addresses
  148. }