kill_test.go 6.0 KB

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