dynamic.go 4.0 KB

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