dynamic.go 3.9 KB

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