container_exec_test.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. package client // import "github.com/docker/docker/client"
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/json"
  6. "fmt"
  7. "io"
  8. "net/http"
  9. "strings"
  10. "testing"
  11. "github.com/docker/docker/api/types"
  12. "github.com/docker/docker/errdefs"
  13. "gotest.tools/v3/assert"
  14. is "gotest.tools/v3/assert/cmp"
  15. )
  16. func TestContainerExecCreateError(t *testing.T) {
  17. client := &Client{
  18. client: newMockClient(errorMock(http.StatusInternalServerError, "Server error")),
  19. }
  20. _, err := client.ContainerExecCreate(context.Background(), "container_id", types.ExecConfig{})
  21. assert.Check(t, is.ErrorType(err, errdefs.IsSystem))
  22. }
  23. func TestContainerExecCreate(t *testing.T) {
  24. expectedURL := "/containers/container_id/exec"
  25. client := &Client{
  26. client: newMockClient(func(req *http.Request) (*http.Response, error) {
  27. if !strings.HasPrefix(req.URL.Path, expectedURL) {
  28. return nil, fmt.Errorf("expected URL '%s', got '%s'", expectedURL, req.URL)
  29. }
  30. if req.Method != http.MethodPost {
  31. return nil, fmt.Errorf("expected POST method, got %s", req.Method)
  32. }
  33. // FIXME validate the content is the given ExecConfig ?
  34. if err := req.ParseForm(); err != nil {
  35. return nil, err
  36. }
  37. execConfig := &types.ExecConfig{}
  38. if err := json.NewDecoder(req.Body).Decode(execConfig); err != nil {
  39. return nil, err
  40. }
  41. if execConfig.User != "user" {
  42. return nil, fmt.Errorf("expected an execConfig with User == 'user', got %v", execConfig)
  43. }
  44. b, err := json.Marshal(types.IDResponse{
  45. ID: "exec_id",
  46. })
  47. if err != nil {
  48. return nil, err
  49. }
  50. return &http.Response{
  51. StatusCode: http.StatusOK,
  52. Body: io.NopCloser(bytes.NewReader(b)),
  53. }, nil
  54. }),
  55. }
  56. r, err := client.ContainerExecCreate(context.Background(), "container_id", types.ExecConfig{
  57. User: "user",
  58. })
  59. if err != nil {
  60. t.Fatal(err)
  61. }
  62. if r.ID != "exec_id" {
  63. t.Fatalf("expected `exec_id`, got %s", r.ID)
  64. }
  65. }
  66. func TestContainerExecStartError(t *testing.T) {
  67. client := &Client{
  68. client: newMockClient(errorMock(http.StatusInternalServerError, "Server error")),
  69. }
  70. err := client.ContainerExecStart(context.Background(), "nothing", types.ExecStartCheck{})
  71. assert.Check(t, is.ErrorType(err, errdefs.IsSystem))
  72. }
  73. func TestContainerExecStart(t *testing.T) {
  74. expectedURL := "/exec/exec_id/start"
  75. client := &Client{
  76. client: newMockClient(func(req *http.Request) (*http.Response, error) {
  77. if !strings.HasPrefix(req.URL.Path, expectedURL) {
  78. return nil, fmt.Errorf("Expected URL '%s', got '%s'", expectedURL, req.URL)
  79. }
  80. if err := req.ParseForm(); err != nil {
  81. return nil, err
  82. }
  83. execStartCheck := &types.ExecStartCheck{}
  84. if err := json.NewDecoder(req.Body).Decode(execStartCheck); err != nil {
  85. return nil, err
  86. }
  87. if execStartCheck.Tty || !execStartCheck.Detach {
  88. return nil, fmt.Errorf("expected execStartCheck{Detach:true,Tty:false}, got %v", execStartCheck)
  89. }
  90. return &http.Response{
  91. StatusCode: http.StatusOK,
  92. Body: io.NopCloser(bytes.NewReader([]byte(""))),
  93. }, nil
  94. }),
  95. }
  96. err := client.ContainerExecStart(context.Background(), "exec_id", types.ExecStartCheck{
  97. Detach: true,
  98. Tty: false,
  99. })
  100. if err != nil {
  101. t.Fatal(err)
  102. }
  103. }
  104. func TestContainerExecInspectError(t *testing.T) {
  105. client := &Client{
  106. client: newMockClient(errorMock(http.StatusInternalServerError, "Server error")),
  107. }
  108. _, err := client.ContainerExecInspect(context.Background(), "nothing")
  109. assert.Check(t, is.ErrorType(err, errdefs.IsSystem))
  110. }
  111. func TestContainerExecInspect(t *testing.T) {
  112. expectedURL := "/exec/exec_id/json"
  113. client := &Client{
  114. client: newMockClient(func(req *http.Request) (*http.Response, error) {
  115. if !strings.HasPrefix(req.URL.Path, expectedURL) {
  116. return nil, fmt.Errorf("Expected URL '%s', got '%s'", expectedURL, req.URL)
  117. }
  118. b, err := json.Marshal(types.ContainerExecInspect{
  119. ExecID: "exec_id",
  120. ContainerID: "container_id",
  121. })
  122. if err != nil {
  123. return nil, err
  124. }
  125. return &http.Response{
  126. StatusCode: http.StatusOK,
  127. Body: io.NopCloser(bytes.NewReader(b)),
  128. }, nil
  129. }),
  130. }
  131. inspect, err := client.ContainerExecInspect(context.Background(), "exec_id")
  132. if err != nil {
  133. t.Fatal(err)
  134. }
  135. if inspect.ExecID != "exec_id" {
  136. t.Fatalf("expected ExecID to be `exec_id`, got %s", inspect.ExecID)
  137. }
  138. if inspect.ContainerID != "container_id" {
  139. t.Fatalf("expected ContainerID `container_id`, got %s", inspect.ContainerID)
  140. }
  141. }