123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- package client // import "github.com/docker/docker/client"
- import (
- "bytes"
- "context"
- "errors"
- "fmt"
- "io"
- "math/rand"
- "net/http"
- "strings"
- "testing"
- "time"
- "github.com/docker/docker/api/types/container"
- "github.com/docker/docker/errdefs"
- "gotest.tools/v3/assert"
- is "gotest.tools/v3/assert/cmp"
- )
- // TestSetHostHeader should set fake host for local communications, set real host
- // for normal communications.
- func TestSetHostHeader(t *testing.T) {
- const testEndpoint = "/test"
- testCases := []struct {
- host string
- expectedHost string
- expectedURLHost string
- }{
- {
- host: "unix:///var/run/docker.sock",
- expectedHost: DummyHost,
- expectedURLHost: "/var/run/docker.sock",
- },
- {
- host: "npipe:////./pipe/docker_engine",
- expectedHost: DummyHost,
- expectedURLHost: "//./pipe/docker_engine",
- },
- {
- host: "tcp://0.0.0.0:4243",
- expectedHost: "",
- expectedURLHost: "0.0.0.0:4243",
- },
- {
- host: "tcp://localhost:4243",
- expectedHost: "",
- expectedURLHost: "localhost:4243",
- },
- }
- for _, tc := range testCases {
- tc := tc
- t.Run(tc.host, func(t *testing.T) {
- hostURL, err := ParseHostURL(tc.host)
- assert.Check(t, err)
- client := &Client{
- client: newMockClient(func(req *http.Request) (*http.Response, error) {
- if !strings.HasPrefix(req.URL.Path, testEndpoint) {
- return nil, fmt.Errorf("expected URL %q, got %q", testEndpoint, req.URL)
- }
- if req.Host != tc.expectedHost {
- return nil, fmt.Errorf("wxpected host %q, got %q", tc.expectedHost, req.Host)
- }
- if req.URL.Host != tc.expectedURLHost {
- return nil, fmt.Errorf("expected URL host %q, got %q", tc.expectedURLHost, req.URL.Host)
- }
- return &http.Response{
- StatusCode: http.StatusOK,
- Body: io.NopCloser(bytes.NewReader([]byte(""))),
- }, nil
- }),
- proto: hostURL.Scheme,
- addr: hostURL.Host,
- basePath: hostURL.Path,
- }
- _, err = client.sendRequest(context.Background(), http.MethodGet, testEndpoint, nil, nil, nil)
- assert.Check(t, err)
- })
- }
- }
- // TestPlainTextError tests the server returning an error in plain text for
- // backwards compatibility with API versions <1.24. All other tests use
- // errors returned as JSON
- func TestPlainTextError(t *testing.T) {
- client := &Client{
- client: newMockClient(plainTextErrorMock(http.StatusInternalServerError, "Server error")),
- }
- _, err := client.ContainerList(context.Background(), container.ListOptions{})
- assert.Check(t, is.ErrorType(err, errdefs.IsSystem))
- }
- func TestInfiniteError(t *testing.T) {
- infinitR := rand.New(rand.NewSource(42))
- client := &Client{
- client: newMockClient(func(req *http.Request) (*http.Response, error) {
- resp := &http.Response{
- StatusCode: http.StatusInternalServerError,
- Header: http.Header{},
- Body: io.NopCloser(infinitR),
- }
- return resp, nil
- }),
- }
- _, err := client.Ping(context.Background())
- assert.Check(t, is.ErrorContains(err, "request returned Internal Server Error"))
- }
- func TestCanceledContext(t *testing.T) {
- const testEndpoint = "/test"
- client := &Client{
- client: newMockClient(func(req *http.Request) (*http.Response, error) {
- assert.Check(t, is.ErrorType(req.Context().Err(), context.Canceled))
- return nil, context.Canceled
- }),
- }
- ctx, cancel := context.WithCancel(context.Background())
- cancel()
- _, err := client.sendRequest(ctx, http.MethodGet, testEndpoint, nil, nil, nil)
- assert.Check(t, is.ErrorType(err, errdefs.IsCancelled))
- assert.Check(t, errors.Is(err, context.Canceled))
- }
- func TestDeadlineExceededContext(t *testing.T) {
- const testEndpoint = "/test"
- client := &Client{
- client: newMockClient(func(req *http.Request) (*http.Response, error) {
- assert.Check(t, is.ErrorType(req.Context().Err(), context.DeadlineExceeded))
- return nil, context.DeadlineExceeded
- }),
- }
- ctx, cancel := context.WithDeadline(context.Background(), time.Now())
- defer cancel()
- <-ctx.Done()
- _, err := client.sendRequest(ctx, http.MethodGet, testEndpoint, nil, nil, nil)
- assert.Check(t, is.ErrorType(err, errdefs.IsDeadline))
- assert.Check(t, errors.Is(err, context.DeadlineExceeded))
- }
|