kill_test.go 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. package container // import "github.com/docker/docker/integration/container"
  2. import (
  3. "context"
  4. "testing"
  5. "time"
  6. "github.com/docker/docker/client"
  7. "github.com/docker/docker/integration/internal/container"
  8. "github.com/docker/docker/testutil/request"
  9. "gotest.tools/v3/assert"
  10. is "gotest.tools/v3/assert/cmp"
  11. "gotest.tools/v3/poll"
  12. "gotest.tools/v3/skip"
  13. )
  14. func TestKillContainerInvalidSignal(t *testing.T) {
  15. defer setupTest(t)()
  16. client := testEnv.APIClient()
  17. ctx := context.Background()
  18. id := container.Run(ctx, t, client)
  19. err := client.ContainerKill(ctx, id, "0")
  20. assert.Error(t, err, "Error response from daemon: Invalid signal: 0")
  21. poll.WaitOn(t, container.IsInState(ctx, client, id, "running"), poll.WithDelay(100*time.Millisecond))
  22. err = client.ContainerKill(ctx, id, "SIG42")
  23. assert.Error(t, err, "Error response from daemon: Invalid signal: SIG42")
  24. poll.WaitOn(t, container.IsInState(ctx, client, id, "running"), poll.WithDelay(100*time.Millisecond))
  25. }
  26. func TestKillContainer(t *testing.T) {
  27. defer setupTest(t)()
  28. client := testEnv.APIClient()
  29. testCases := []struct {
  30. doc string
  31. signal string
  32. status string
  33. skipOs string
  34. }{
  35. {
  36. doc: "no signal",
  37. signal: "",
  38. status: "exited",
  39. skipOs: "",
  40. },
  41. {
  42. doc: "non killing signal",
  43. signal: "SIGWINCH",
  44. status: "running",
  45. skipOs: "windows",
  46. },
  47. {
  48. doc: "killing signal",
  49. signal: "SIGTERM",
  50. status: "exited",
  51. skipOs: "",
  52. },
  53. }
  54. for _, tc := range testCases {
  55. tc := tc
  56. t.Run(tc.doc, func(t *testing.T) {
  57. skip.If(t, testEnv.OSType == tc.skipOs, "Windows does not support SIGWINCH")
  58. ctx := context.Background()
  59. id := container.Run(ctx, t, client)
  60. err := client.ContainerKill(ctx, id, tc.signal)
  61. assert.NilError(t, err)
  62. poll.WaitOn(t, container.IsInState(ctx, client, id, tc.status), poll.WithDelay(100*time.Millisecond))
  63. })
  64. }
  65. }
  66. func TestKillWithStopSignalAndRestartPolicies(t *testing.T) {
  67. skip.If(t, testEnv.OSType == "windows", "Windows only supports 1.25 or later")
  68. defer setupTest(t)()
  69. client := testEnv.APIClient()
  70. testCases := []struct {
  71. doc string
  72. stopsignal string
  73. status string
  74. }{
  75. {
  76. doc: "same-signal-disables-restart-policy",
  77. stopsignal: "TERM",
  78. status: "exited",
  79. },
  80. {
  81. doc: "different-signal-keep-restart-policy",
  82. stopsignal: "CONT",
  83. status: "running",
  84. },
  85. }
  86. for _, tc := range testCases {
  87. tc := tc
  88. t.Run(tc.doc, func(t *testing.T) {
  89. ctx := context.Background()
  90. id := container.Run(ctx, t, client,
  91. container.WithRestartPolicy("always"),
  92. func(c *container.TestContainerConfig) {
  93. c.Config.StopSignal = tc.stopsignal
  94. })
  95. err := client.ContainerKill(ctx, id, "TERM")
  96. assert.NilError(t, err)
  97. poll.WaitOn(t, container.IsInState(ctx, client, id, tc.status), poll.WithDelay(100*time.Millisecond))
  98. })
  99. }
  100. }
  101. func TestKillStoppedContainer(t *testing.T) {
  102. skip.If(t, testEnv.OSType == "windows", "Windows only supports 1.25 or later")
  103. defer setupTest(t)()
  104. ctx := context.Background()
  105. client := testEnv.APIClient()
  106. id := container.Create(ctx, t, client)
  107. err := client.ContainerKill(ctx, id, "SIGKILL")
  108. assert.Assert(t, is.ErrorContains(err, ""))
  109. assert.Assert(t, is.Contains(err.Error(), "is not running"))
  110. }
  111. func TestKillStoppedContainerAPIPre120(t *testing.T) {
  112. skip.If(t, testEnv.OSType == "windows", "Windows only supports 1.25 or later")
  113. defer setupTest(t)()
  114. ctx := context.Background()
  115. client := request.NewAPIClient(t, client.WithVersion("1.19"))
  116. id := container.Create(ctx, t, client)
  117. err := client.ContainerKill(ctx, id, "SIGKILL")
  118. assert.NilError(t, err)
  119. }
  120. func TestKillDifferentUserContainer(t *testing.T) {
  121. // TODO Windows: Windows does not yet support -u (Feb 2016).
  122. skip.If(t, testEnv.OSType == "windows", "User containers (container.Config.User) are not yet supported on %q platform", testEnv.OSType)
  123. defer setupTest(t)()
  124. ctx := context.Background()
  125. client := request.NewAPIClient(t, client.WithVersion("1.19"))
  126. id := container.Run(ctx, t, client, func(c *container.TestContainerConfig) {
  127. c.Config.User = "daemon"
  128. })
  129. poll.WaitOn(t, container.IsInState(ctx, client, id, "running"), poll.WithDelay(100*time.Millisecond))
  130. err := client.ContainerKill(ctx, id, "SIGKILL")
  131. assert.NilError(t, err)
  132. poll.WaitOn(t, container.IsInState(ctx, client, id, "exited"), poll.WithDelay(100*time.Millisecond))
  133. }
  134. func TestInspectOomKilledTrue(t *testing.T) {
  135. skip.If(t, testEnv.DaemonInfo.OSType == "windows")
  136. skip.If(t, testEnv.DaemonInfo.CgroupDriver == "none")
  137. skip.If(t, !testEnv.DaemonInfo.MemoryLimit || !testEnv.DaemonInfo.SwapLimit)
  138. skip.If(t, testEnv.DaemonInfo.CgroupVersion == "2", "FIXME: flaky on cgroup v2 (https://github.com/moby/moby/issues/41929)")
  139. defer setupTest(t)()
  140. ctx := context.Background()
  141. client := testEnv.APIClient()
  142. cID := container.Run(ctx, t, client, container.WithCmd("sh", "-c", "x=a; while true; do x=$x$x$x$x; done"), func(c *container.TestContainerConfig) {
  143. c.HostConfig.Resources.Memory = 32 * 1024 * 1024
  144. })
  145. poll.WaitOn(t, container.IsInState(ctx, client, cID, "exited"), poll.WithDelay(100*time.Millisecond))
  146. inspect, err := client.ContainerInspect(ctx, cID)
  147. assert.NilError(t, err)
  148. assert.Check(t, is.Equal(true, inspect.State.OOMKilled))
  149. }
  150. func TestInspectOomKilledFalse(t *testing.T) {
  151. skip.If(t, testEnv.DaemonInfo.OSType == "windows" || !testEnv.DaemonInfo.MemoryLimit || !testEnv.DaemonInfo.SwapLimit)
  152. defer setupTest(t)()
  153. ctx := context.Background()
  154. client := testEnv.APIClient()
  155. cID := container.Run(ctx, t, client, container.WithCmd("sh", "-c", "echo hello world"))
  156. poll.WaitOn(t, container.IsInState(ctx, client, cID, "exited"), poll.WithDelay(100*time.Millisecond))
  157. inspect, err := client.ContainerInspect(ctx, cID)
  158. assert.NilError(t, err)
  159. assert.Check(t, is.Equal(false, inspect.State.OOMKilled))
  160. }