kill_test.go 5.6 KB

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