dynamic.go 3.8 KB

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