docker_cli_attach_test.go 3.6 KB

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