docker_cli_attach_test.go 3.5 KB

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