docker_cli_attach_test.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "os/exec"
  7. "strings"
  8. "sync"
  9. "time"
  10. "github.com/go-check/check"
  11. )
  12. const attachWait = 5 * time.Second
  13. func (s *DockerSuite) TestAttachMultipleAndRestart(c *check.C) {
  14. endGroup := &sync.WaitGroup{}
  15. startGroup := &sync.WaitGroup{}
  16. endGroup.Add(3)
  17. startGroup.Add(3)
  18. if err := waitForContainer("attacher", "-d", "busybox", "/bin/sh", "-c", "while true; do sleep 1; echo hello; done"); err != nil {
  19. c.Fatal(err)
  20. }
  21. startDone := make(chan struct{})
  22. endDone := make(chan struct{})
  23. go func() {
  24. endGroup.Wait()
  25. close(endDone)
  26. }()
  27. go func() {
  28. startGroup.Wait()
  29. close(startDone)
  30. }()
  31. for i := 0; i < 3; i++ {
  32. go func() {
  33. cmd := exec.Command(dockerBinary, "attach", "attacher")
  34. defer func() {
  35. cmd.Wait()
  36. endGroup.Done()
  37. }()
  38. out, err := cmd.StdoutPipe()
  39. if err != nil {
  40. c.Fatal(err)
  41. }
  42. if err := cmd.Start(); err != nil {
  43. c.Fatal(err)
  44. }
  45. buf := make([]byte, 1024)
  46. if _, err := out.Read(buf); err != nil && err != io.EOF {
  47. c.Fatal(err)
  48. }
  49. startGroup.Done()
  50. if !strings.Contains(string(buf), "hello") {
  51. c.Fatalf("unexpected output %s expected hello\n", string(buf))
  52. }
  53. }()
  54. }
  55. select {
  56. case <-startDone:
  57. case <-time.After(attachWait):
  58. c.Fatalf("Attaches did not initialize properly")
  59. }
  60. cmd := exec.Command(dockerBinary, "kill", "attacher")
  61. if _, err := runCommand(cmd); err != nil {
  62. c.Fatal(err)
  63. }
  64. select {
  65. case <-endDone:
  66. case <-time.After(attachWait):
  67. c.Fatalf("Attaches did not finish properly")
  68. }
  69. }
  70. func (s *DockerSuite) TestAttachTtyWithoutStdin(c *check.C) {
  71. cmd := exec.Command(dockerBinary, "run", "-d", "-ti", "busybox")
  72. out, _, err := runCommandWithOutput(cmd)
  73. if err != nil {
  74. c.Fatalf("failed to start container: %v (%v)", out, err)
  75. }
  76. id := strings.TrimSpace(out)
  77. if err := waitRun(id); err != nil {
  78. c.Fatal(err)
  79. }
  80. defer func() {
  81. cmd := exec.Command(dockerBinary, "kill", id)
  82. if out, _, err := runCommandWithOutput(cmd); err != nil {
  83. c.Fatalf("failed to kill container: %v (%v)", out, err)
  84. }
  85. }()
  86. done := make(chan error)
  87. go func() {
  88. defer close(done)
  89. cmd := exec.Command(dockerBinary, "attach", id)
  90. if _, err := cmd.StdinPipe(); err != nil {
  91. done <- err
  92. return
  93. }
  94. expected := "cannot enable tty mode"
  95. if out, _, err := runCommandWithOutput(cmd); err == nil {
  96. done <- fmt.Errorf("attach should have failed")
  97. return
  98. } else if !strings.Contains(out, expected) {
  99. done <- fmt.Errorf("attach failed with error %q: expected %q", out, expected)
  100. return
  101. }
  102. }()
  103. select {
  104. case err := <-done:
  105. c.Assert(err, check.IsNil)
  106. case <-time.After(attachWait):
  107. c.Fatal("attach is running but should have failed")
  108. }
  109. }
  110. func (s *DockerSuite) TestAttachDisconnect(c *check.C) {
  111. out, _ := dockerCmd(c, "run", "-di", "busybox", "/bin/cat")
  112. id := strings.TrimSpace(out)
  113. cmd := exec.Command(dockerBinary, "attach", id)
  114. stdin, err := cmd.StdinPipe()
  115. if err != nil {
  116. c.Fatal(err)
  117. }
  118. defer stdin.Close()
  119. stdout, err := cmd.StdoutPipe()
  120. if err != nil {
  121. c.Fatal(err)
  122. }
  123. defer stdout.Close()
  124. if err := cmd.Start(); err != nil {
  125. c.Fatal(err)
  126. }
  127. defer cmd.Process.Kill()
  128. if _, err := stdin.Write([]byte("hello\n")); err != nil {
  129. c.Fatal(err)
  130. }
  131. out, err = bufio.NewReader(stdout).ReadString('\n')
  132. if err != nil {
  133. c.Fatal(err)
  134. }
  135. if strings.TrimSpace(out) != "hello" {
  136. c.Fatalf("expected 'hello', got %q", out)
  137. }
  138. if err := stdin.Close(); err != nil {
  139. c.Fatal(err)
  140. }
  141. // Expect container to still be running after stdin is closed
  142. running, err := inspectField(id, "State.Running")
  143. if err != nil {
  144. c.Fatal(err)
  145. }
  146. if running != "true" {
  147. c.Fatal("expected container to still be running")
  148. }
  149. }