client_test.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. package plugins // import "github.com/docker/docker/pkg/plugins"
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "io"
  8. "net/http"
  9. "net/http/httptest"
  10. "net/url"
  11. "os"
  12. "strings"
  13. "testing"
  14. "time"
  15. "github.com/docker/docker/pkg/plugins/transport"
  16. "github.com/docker/go-connections/tlsconfig"
  17. "gotest.tools/v3/assert"
  18. is "gotest.tools/v3/assert/cmp"
  19. )
  20. func setupRemotePluginServer(t *testing.T) (mux *http.ServeMux, addr string) {
  21. t.Helper()
  22. mux = http.NewServeMux()
  23. server := httptest.NewServer(mux)
  24. t.Logf("started remote plugin server listening on: %s", server.URL)
  25. t.Cleanup(func() {
  26. server.Close()
  27. })
  28. return mux, server.URL
  29. }
  30. func TestFailedConnection(t *testing.T) {
  31. t.Parallel()
  32. c, _ := NewClient("tcp://127.0.0.1:1", &tlsconfig.Options{InsecureSkipVerify: true})
  33. _, err := c.callWithRetry("Service.Method", nil, false)
  34. if err == nil {
  35. t.Fatal("Unexpected successful connection")
  36. }
  37. }
  38. func TestFailOnce(t *testing.T) {
  39. t.Parallel()
  40. mux, addr := setupRemotePluginServer(t)
  41. failed := false
  42. mux.HandleFunc("/Test.FailOnce", func(w http.ResponseWriter, r *http.Request) {
  43. if !failed {
  44. failed = true
  45. panic("Plugin not ready (intentional panic for test)")
  46. }
  47. })
  48. c, _ := NewClient(addr, &tlsconfig.Options{InsecureSkipVerify: true})
  49. b := strings.NewReader("body")
  50. _, err := c.callWithRetry("Test.FailOnce", b, true)
  51. if err != nil {
  52. t.Fatal(err)
  53. }
  54. }
  55. func TestEchoInputOutput(t *testing.T) {
  56. t.Parallel()
  57. mux, addr := setupRemotePluginServer(t)
  58. m := Manifest{[]string{"VolumeDriver", "NetworkDriver"}}
  59. mux.HandleFunc("/Test.Echo", func(w http.ResponseWriter, r *http.Request) {
  60. if r.Method != http.MethodPost {
  61. t.Fatalf("Expected POST, got %s\n", r.Method)
  62. }
  63. header := w.Header()
  64. header.Set("Content-Type", transport.VersionMimetype)
  65. io.Copy(w, r.Body)
  66. })
  67. c, _ := NewClient(addr, &tlsconfig.Options{InsecureSkipVerify: true})
  68. var output Manifest
  69. err := c.Call("Test.Echo", m, &output)
  70. if err != nil {
  71. t.Fatal(err)
  72. }
  73. assert.Check(t, is.DeepEqual(m, output))
  74. err = c.Call("Test.Echo", nil, nil)
  75. if err != nil {
  76. t.Fatal(err)
  77. }
  78. }
  79. func TestBackoff(t *testing.T) {
  80. t.Parallel()
  81. cases := []struct {
  82. retries int
  83. expTimeOff time.Duration
  84. }{
  85. {expTimeOff: time.Duration(1)},
  86. {retries: 1, expTimeOff: time.Duration(2)},
  87. {retries: 2, expTimeOff: time.Duration(4)},
  88. {retries: 4, expTimeOff: time.Duration(16)},
  89. {retries: 6, expTimeOff: time.Duration(30)},
  90. {retries: 10, expTimeOff: time.Duration(30)},
  91. }
  92. for _, tc := range cases {
  93. tc := tc
  94. t.Run(fmt.Sprintf("retries: %v", tc.retries), func(t *testing.T) {
  95. s := tc.expTimeOff * time.Second
  96. if d := backoff(tc.retries); d != s {
  97. t.Fatalf("Retry %v, expected %v, was %v\n", tc.retries, s, d)
  98. }
  99. })
  100. }
  101. }
  102. func TestAbortRetry(t *testing.T) {
  103. t.Parallel()
  104. cases := []struct {
  105. timeOff time.Duration
  106. expAbort bool
  107. }{
  108. {timeOff: time.Duration(1)},
  109. {timeOff: time.Duration(2)},
  110. {timeOff: time.Duration(10)},
  111. {timeOff: time.Duration(30), expAbort: true},
  112. {timeOff: time.Duration(40), expAbort: true},
  113. }
  114. for _, tc := range cases {
  115. tc := tc
  116. t.Run(fmt.Sprintf("duration: %v", tc.timeOff), func(t *testing.T) {
  117. s := tc.timeOff * time.Second
  118. if a := abort(time.Now(), s, 0); a != tc.expAbort {
  119. t.Fatalf("Duration %v, expected %v, was %v\n", tc.timeOff, s, a)
  120. }
  121. })
  122. }
  123. }
  124. func TestClientScheme(t *testing.T) {
  125. t.Parallel()
  126. cases := map[string]string{
  127. "tcp://127.0.0.1:8080": "http",
  128. "unix:///usr/local/plugins/foo": "http",
  129. "http://127.0.0.1:8080": "http",
  130. "https://127.0.0.1:8080": "https",
  131. }
  132. for addr, scheme := range cases {
  133. u, err := url.Parse(addr)
  134. if err != nil {
  135. t.Error(err)
  136. }
  137. s := httpScheme(u)
  138. if s != scheme {
  139. t.Fatalf("URL scheme mismatch, expected %s, got %s", scheme, s)
  140. }
  141. }
  142. }
  143. func TestNewClientWithTimeout(t *testing.T) {
  144. t.Parallel()
  145. mux, addr := setupRemotePluginServer(t)
  146. m := Manifest{[]string{"VolumeDriver", "NetworkDriver"}}
  147. mux.HandleFunc("/Test.Echo", func(w http.ResponseWriter, r *http.Request) {
  148. time.Sleep(20 * time.Millisecond)
  149. io.Copy(w, r.Body)
  150. })
  151. timeout := 10 * time.Millisecond
  152. c, _ := NewClientWithTimeout(addr, &tlsconfig.Options{InsecureSkipVerify: true}, timeout)
  153. var output Manifest
  154. err := c.CallWithOptions("Test.Echo", m, &output, func(opts *RequestOpts) { opts.testTimeOut = 1 })
  155. assert.ErrorType(t, err, os.IsTimeout)
  156. }
  157. func TestClientStream(t *testing.T) {
  158. t.Parallel()
  159. mux, addr := setupRemotePluginServer(t)
  160. m := Manifest{[]string{"VolumeDriver", "NetworkDriver"}}
  161. var output Manifest
  162. mux.HandleFunc("/Test.Echo", func(w http.ResponseWriter, r *http.Request) {
  163. if r.Method != http.MethodPost {
  164. t.Fatalf("Expected POST, got %s", r.Method)
  165. }
  166. header := w.Header()
  167. header.Set("Content-Type", transport.VersionMimetype)
  168. io.Copy(w, r.Body)
  169. })
  170. c, _ := NewClient(addr, &tlsconfig.Options{InsecureSkipVerify: true})
  171. body, err := c.Stream("Test.Echo", m)
  172. if err != nil {
  173. t.Fatal(err)
  174. }
  175. defer body.Close()
  176. if err := json.NewDecoder(body).Decode(&output); err != nil {
  177. t.Fatalf("Test.Echo: error reading plugin resp: %v", err)
  178. }
  179. assert.Check(t, is.DeepEqual(m, output))
  180. }
  181. func TestClientSendFile(t *testing.T) {
  182. t.Parallel()
  183. mux, addr := setupRemotePluginServer(t)
  184. m := Manifest{[]string{"VolumeDriver", "NetworkDriver"}}
  185. var output Manifest
  186. var buf bytes.Buffer
  187. if err := json.NewEncoder(&buf).Encode(m); err != nil {
  188. t.Fatal(err)
  189. }
  190. mux.HandleFunc("/Test.Echo", func(w http.ResponseWriter, r *http.Request) {
  191. if r.Method != http.MethodPost {
  192. t.Fatalf("Expected POST, got %s\n", r.Method)
  193. }
  194. header := w.Header()
  195. header.Set("Content-Type", transport.VersionMimetype)
  196. io.Copy(w, r.Body)
  197. })
  198. c, _ := NewClient(addr, &tlsconfig.Options{InsecureSkipVerify: true})
  199. if err := c.SendFile("Test.Echo", &buf, &output); err != nil {
  200. t.Fatal(err)
  201. }
  202. assert.Check(t, is.DeepEqual(m, output))
  203. }
  204. func TestClientWithRequestTimeout(t *testing.T) {
  205. t.Parallel()
  206. type timeoutError interface {
  207. Timeout() bool
  208. }
  209. unblock := make(chan struct{})
  210. testHandler := func(w http.ResponseWriter, r *http.Request) {
  211. select {
  212. case <-unblock:
  213. case <-r.Context().Done():
  214. }
  215. w.WriteHeader(http.StatusOK)
  216. }
  217. srv := httptest.NewServer(http.HandlerFunc(testHandler))
  218. defer func() {
  219. close(unblock)
  220. srv.Close()
  221. }()
  222. client := &Client{http: srv.Client(), requestFactory: &testRequestWrapper{srv}}
  223. errCh := make(chan error, 1)
  224. go func() {
  225. _, err := client.callWithRetry("/Plugin.Hello", nil, false, WithRequestTimeout(time.Millisecond))
  226. errCh <- err
  227. }()
  228. timer := time.NewTimer(5 * time.Second)
  229. defer timer.Stop()
  230. select {
  231. case err := <-errCh:
  232. var tErr timeoutError
  233. if assert.Check(t, errors.As(err, &tErr), "want timeout error, got %T", err) {
  234. assert.Check(t, tErr.Timeout())
  235. }
  236. case <-timer.C:
  237. t.Fatal("client request did not time out in time")
  238. }
  239. }
  240. type testRequestWrapper struct {
  241. *httptest.Server
  242. }
  243. func (w *testRequestWrapper) NewRequest(path string, data io.Reader) (*http.Request, error) {
  244. req, err := http.NewRequest(http.MethodPost, path, data)
  245. if err != nil {
  246. return nil, err
  247. }
  248. u, err := url.Parse(w.Server.URL)
  249. if err != nil {
  250. return nil, err
  251. }
  252. req.URL = u
  253. return req, nil
  254. }