1
0

dynamic.go 4.9 KB

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