dynamic.go 4.6 KB

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