kill_test.go 6.2 KB

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