docker_cli_update_unix_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. // +build !windows
  2. package main
  3. import (
  4. "context"
  5. "encoding/json"
  6. "fmt"
  7. "os/exec"
  8. "strings"
  9. "time"
  10. "github.com/docker/docker/api/types"
  11. "github.com/docker/docker/client"
  12. "github.com/docker/docker/integration-cli/checker"
  13. "github.com/docker/docker/integration-cli/request"
  14. "github.com/go-check/check"
  15. "github.com/kr/pty"
  16. )
  17. func (s *DockerSuite) TestUpdateRunningContainer(c *check.C) {
  18. testRequires(c, DaemonIsLinux)
  19. testRequires(c, memoryLimitSupport)
  20. name := "test-update-container"
  21. dockerCmd(c, "run", "-d", "--name", name, "-m", "300M", "busybox", "top")
  22. dockerCmd(c, "update", "-m", "500M", name)
  23. c.Assert(inspectField(c, name, "HostConfig.Memory"), checker.Equals, "524288000")
  24. file := "/sys/fs/cgroup/memory/memory.limit_in_bytes"
  25. out, _ := dockerCmd(c, "exec", name, "cat", file)
  26. c.Assert(strings.TrimSpace(out), checker.Equals, "524288000")
  27. }
  28. func (s *DockerSuite) TestUpdateRunningContainerWithRestart(c *check.C) {
  29. testRequires(c, DaemonIsLinux)
  30. testRequires(c, memoryLimitSupport)
  31. name := "test-update-container"
  32. dockerCmd(c, "run", "-d", "--name", name, "-m", "300M", "busybox", "top")
  33. dockerCmd(c, "update", "-m", "500M", name)
  34. dockerCmd(c, "restart", name)
  35. c.Assert(inspectField(c, name, "HostConfig.Memory"), checker.Equals, "524288000")
  36. file := "/sys/fs/cgroup/memory/memory.limit_in_bytes"
  37. out, _ := dockerCmd(c, "exec", name, "cat", file)
  38. c.Assert(strings.TrimSpace(out), checker.Equals, "524288000")
  39. }
  40. func (s *DockerSuite) TestUpdateStoppedContainer(c *check.C) {
  41. testRequires(c, DaemonIsLinux)
  42. testRequires(c, memoryLimitSupport)
  43. name := "test-update-container"
  44. file := "/sys/fs/cgroup/memory/memory.limit_in_bytes"
  45. dockerCmd(c, "run", "--name", name, "-m", "300M", "busybox", "cat", file)
  46. dockerCmd(c, "update", "-m", "500M", name)
  47. c.Assert(inspectField(c, name, "HostConfig.Memory"), checker.Equals, "524288000")
  48. out, _ := dockerCmd(c, "start", "-a", name)
  49. c.Assert(strings.TrimSpace(out), checker.Equals, "524288000")
  50. }
  51. func (s *DockerSuite) TestUpdatePausedContainer(c *check.C) {
  52. testRequires(c, DaemonIsLinux)
  53. testRequires(c, cpuShare)
  54. name := "test-update-container"
  55. dockerCmd(c, "run", "-d", "--name", name, "--cpu-shares", "1000", "busybox", "top")
  56. dockerCmd(c, "pause", name)
  57. dockerCmd(c, "update", "--cpu-shares", "500", name)
  58. c.Assert(inspectField(c, name, "HostConfig.CPUShares"), checker.Equals, "500")
  59. dockerCmd(c, "unpause", name)
  60. file := "/sys/fs/cgroup/cpu/cpu.shares"
  61. out, _ := dockerCmd(c, "exec", name, "cat", file)
  62. c.Assert(strings.TrimSpace(out), checker.Equals, "500")
  63. }
  64. func (s *DockerSuite) TestUpdateWithUntouchedFields(c *check.C) {
  65. testRequires(c, DaemonIsLinux)
  66. testRequires(c, memoryLimitSupport)
  67. testRequires(c, cpuShare)
  68. name := "test-update-container"
  69. dockerCmd(c, "run", "-d", "--name", name, "-m", "300M", "--cpu-shares", "800", "busybox", "top")
  70. dockerCmd(c, "update", "-m", "500M", name)
  71. // Update memory and not touch cpus, `cpuset.cpus` should still have the old value
  72. out := inspectField(c, name, "HostConfig.CPUShares")
  73. c.Assert(out, check.Equals, "800")
  74. file := "/sys/fs/cgroup/cpu/cpu.shares"
  75. out, _ = dockerCmd(c, "exec", name, "cat", file)
  76. c.Assert(strings.TrimSpace(out), checker.Equals, "800")
  77. }
  78. func (s *DockerSuite) TestUpdateContainerInvalidValue(c *check.C) {
  79. testRequires(c, DaemonIsLinux)
  80. testRequires(c, memoryLimitSupport)
  81. name := "test-update-container"
  82. dockerCmd(c, "run", "-d", "--name", name, "-m", "300M", "busybox", "true")
  83. out, _, err := dockerCmdWithError("update", "-m", "2M", name)
  84. c.Assert(err, check.NotNil)
  85. expected := "Minimum memory limit allowed is 4MB"
  86. c.Assert(out, checker.Contains, expected)
  87. }
  88. func (s *DockerSuite) TestUpdateContainerWithoutFlags(c *check.C) {
  89. testRequires(c, DaemonIsLinux)
  90. testRequires(c, memoryLimitSupport)
  91. name := "test-update-container"
  92. dockerCmd(c, "run", "-d", "--name", name, "-m", "300M", "busybox", "true")
  93. _, _, err := dockerCmdWithError("update", name)
  94. c.Assert(err, check.NotNil)
  95. }
  96. func (s *DockerSuite) TestUpdateKernelMemory(c *check.C) {
  97. testRequires(c, DaemonIsLinux, kernelMemorySupport)
  98. name := "test-update-container"
  99. dockerCmd(c, "run", "-d", "--name", name, "--kernel-memory", "50M", "busybox", "top")
  100. dockerCmd(c, "update", "--kernel-memory", "100M", name)
  101. c.Assert(inspectField(c, name, "HostConfig.KernelMemory"), checker.Equals, "104857600")
  102. file := "/sys/fs/cgroup/memory/memory.kmem.limit_in_bytes"
  103. out, _ := dockerCmd(c, "exec", name, "cat", file)
  104. c.Assert(strings.TrimSpace(out), checker.Equals, "104857600")
  105. }
  106. func (s *DockerSuite) TestUpdateKernelMemoryUninitialized(c *check.C) {
  107. testRequires(c, DaemonIsLinux, kernelMemorySupport)
  108. isNewKernel := CheckKernelVersion(4, 6, 0)
  109. name := "test-update-container"
  110. dockerCmd(c, "run", "-d", "--name", name, "busybox", "top")
  111. _, _, err := dockerCmdWithError("update", "--kernel-memory", "100M", name)
  112. // Update kernel memory to a running container without kernel memory initialized
  113. // is not allowed before kernel version 4.6.
  114. if !isNewKernel {
  115. c.Assert(err, check.NotNil)
  116. } else {
  117. c.Assert(err, check.IsNil)
  118. }
  119. dockerCmd(c, "pause", name)
  120. _, _, err = dockerCmdWithError("update", "--kernel-memory", "200M", name)
  121. if !isNewKernel {
  122. c.Assert(err, check.NotNil)
  123. } else {
  124. c.Assert(err, check.IsNil)
  125. }
  126. dockerCmd(c, "unpause", name)
  127. dockerCmd(c, "stop", name)
  128. dockerCmd(c, "update", "--kernel-memory", "300M", name)
  129. dockerCmd(c, "start", name)
  130. c.Assert(inspectField(c, name, "HostConfig.KernelMemory"), checker.Equals, "314572800")
  131. file := "/sys/fs/cgroup/memory/memory.kmem.limit_in_bytes"
  132. out, _ := dockerCmd(c, "exec", name, "cat", file)
  133. c.Assert(strings.TrimSpace(out), checker.Equals, "314572800")
  134. }
  135. func (s *DockerSuite) TestUpdateSwapMemoryOnly(c *check.C) {
  136. testRequires(c, DaemonIsLinux)
  137. testRequires(c, memoryLimitSupport)
  138. testRequires(c, swapMemorySupport)
  139. name := "test-update-container"
  140. dockerCmd(c, "run", "-d", "--name", name, "--memory", "300M", "--memory-swap", "500M", "busybox", "top")
  141. dockerCmd(c, "update", "--memory-swap", "600M", name)
  142. c.Assert(inspectField(c, name, "HostConfig.MemorySwap"), checker.Equals, "629145600")
  143. file := "/sys/fs/cgroup/memory/memory.memsw.limit_in_bytes"
  144. out, _ := dockerCmd(c, "exec", name, "cat", file)
  145. c.Assert(strings.TrimSpace(out), checker.Equals, "629145600")
  146. }
  147. func (s *DockerSuite) TestUpdateInvalidSwapMemory(c *check.C) {
  148. testRequires(c, DaemonIsLinux)
  149. testRequires(c, memoryLimitSupport)
  150. testRequires(c, swapMemorySupport)
  151. name := "test-update-container"
  152. dockerCmd(c, "run", "-d", "--name", name, "--memory", "300M", "--memory-swap", "500M", "busybox", "top")
  153. _, _, err := dockerCmdWithError("update", "--memory-swap", "200M", name)
  154. // Update invalid swap memory should fail.
  155. // This will pass docker config validation, but failed at kernel validation
  156. c.Assert(err, check.NotNil)
  157. // Update invalid swap memory with failure should not change HostConfig
  158. c.Assert(inspectField(c, name, "HostConfig.Memory"), checker.Equals, "314572800")
  159. c.Assert(inspectField(c, name, "HostConfig.MemorySwap"), checker.Equals, "524288000")
  160. dockerCmd(c, "update", "--memory-swap", "600M", name)
  161. c.Assert(inspectField(c, name, "HostConfig.MemorySwap"), checker.Equals, "629145600")
  162. file := "/sys/fs/cgroup/memory/memory.memsw.limit_in_bytes"
  163. out, _ := dockerCmd(c, "exec", name, "cat", file)
  164. c.Assert(strings.TrimSpace(out), checker.Equals, "629145600")
  165. }
  166. func (s *DockerSuite) TestUpdateStats(c *check.C) {
  167. testRequires(c, DaemonIsLinux)
  168. testRequires(c, memoryLimitSupport)
  169. testRequires(c, cpuCfsQuota)
  170. name := "foo"
  171. dockerCmd(c, "run", "-d", "-ti", "--name", name, "-m", "500m", "busybox")
  172. c.Assert(waitRun(name), checker.IsNil)
  173. getMemLimit := func(id string) uint64 {
  174. resp, body, err := request.Get(fmt.Sprintf("/containers/%s/stats?stream=false", id))
  175. c.Assert(err, checker.IsNil)
  176. c.Assert(resp.Header.Get("Content-Type"), checker.Equals, "application/json")
  177. var v *types.Stats
  178. err = json.NewDecoder(body).Decode(&v)
  179. c.Assert(err, checker.IsNil)
  180. body.Close()
  181. return v.MemoryStats.Limit
  182. }
  183. preMemLimit := getMemLimit(name)
  184. dockerCmd(c, "update", "--cpu-quota", "2000", name)
  185. curMemLimit := getMemLimit(name)
  186. c.Assert(preMemLimit, checker.Equals, curMemLimit)
  187. }
  188. func (s *DockerSuite) TestUpdateMemoryWithSwapMemory(c *check.C) {
  189. testRequires(c, DaemonIsLinux)
  190. testRequires(c, memoryLimitSupport)
  191. testRequires(c, swapMemorySupport)
  192. name := "test-update-container"
  193. dockerCmd(c, "run", "-d", "--name", name, "--memory", "300M", "busybox", "top")
  194. out, _, err := dockerCmdWithError("update", "--memory", "800M", name)
  195. c.Assert(err, checker.NotNil)
  196. c.Assert(out, checker.Contains, "Memory limit should be smaller than already set memoryswap limit")
  197. dockerCmd(c, "update", "--memory", "800M", "--memory-swap", "1000M", name)
  198. }
  199. func (s *DockerSuite) TestUpdateNotAffectMonitorRestartPolicy(c *check.C) {
  200. testRequires(c, DaemonIsLinux, cpuShare)
  201. out, _ := dockerCmd(c, "run", "-tid", "--restart=always", "busybox", "sh")
  202. id := strings.TrimSpace(string(out))
  203. dockerCmd(c, "update", "--cpu-shares", "512", id)
  204. cpty, tty, err := pty.Open()
  205. c.Assert(err, checker.IsNil)
  206. defer cpty.Close()
  207. cmd := exec.Command(dockerBinary, "attach", id)
  208. cmd.Stdin = tty
  209. c.Assert(cmd.Start(), checker.IsNil)
  210. defer cmd.Process.Kill()
  211. _, err = cpty.Write([]byte("exit\n"))
  212. c.Assert(err, checker.IsNil)
  213. c.Assert(cmd.Wait(), checker.IsNil)
  214. // container should restart again and keep running
  215. err = waitInspect(id, "{{.RestartCount}}", "1", 30*time.Second)
  216. c.Assert(err, checker.IsNil)
  217. c.Assert(waitRun(id), checker.IsNil)
  218. }
  219. func (s *DockerSuite) TestUpdateWithNanoCPUs(c *check.C) {
  220. testRequires(c, cpuCfsQuota, cpuCfsPeriod)
  221. file1 := "/sys/fs/cgroup/cpu/cpu.cfs_quota_us"
  222. file2 := "/sys/fs/cgroup/cpu/cpu.cfs_period_us"
  223. out, _ := dockerCmd(c, "run", "-d", "--cpus", "0.5", "--name", "top", "busybox", "top")
  224. c.Assert(strings.TrimSpace(out), checker.Not(checker.Equals), "")
  225. out, _ = dockerCmd(c, "exec", "top", "sh", "-c", fmt.Sprintf("cat %s && cat %s", file1, file2))
  226. c.Assert(strings.TrimSpace(out), checker.Equals, "50000\n100000")
  227. clt, err := client.NewEnvClient()
  228. c.Assert(err, checker.IsNil)
  229. inspect, err := clt.ContainerInspect(context.Background(), "top")
  230. c.Assert(err, checker.IsNil)
  231. c.Assert(inspect.HostConfig.NanoCPUs, checker.Equals, int64(500000000))
  232. out = inspectField(c, "top", "HostConfig.CpuQuota")
  233. c.Assert(out, checker.Equals, "0", check.Commentf("CPU CFS quota should be 0"))
  234. out = inspectField(c, "top", "HostConfig.CpuPeriod")
  235. c.Assert(out, checker.Equals, "0", check.Commentf("CPU CFS period should be 0"))
  236. out, _, err = dockerCmdWithError("update", "--cpu-quota", "80000", "top")
  237. c.Assert(err, checker.NotNil)
  238. c.Assert(out, checker.Contains, "Conflicting options: CPU Quota cannot be updated as NanoCPUs has already been set")
  239. out, _ = dockerCmd(c, "update", "--cpus", "0.8", "top")
  240. inspect, err = clt.ContainerInspect(context.Background(), "top")
  241. c.Assert(err, checker.IsNil)
  242. c.Assert(inspect.HostConfig.NanoCPUs, checker.Equals, int64(800000000))
  243. out = inspectField(c, "top", "HostConfig.CpuQuota")
  244. c.Assert(out, checker.Equals, "0", check.Commentf("CPU CFS quota should be 0"))
  245. out = inspectField(c, "top", "HostConfig.CpuPeriod")
  246. c.Assert(out, checker.Equals, "0", check.Commentf("CPU CFS period should be 0"))
  247. out, _ = dockerCmd(c, "exec", "top", "sh", "-c", fmt.Sprintf("cat %s && cat %s", file1, file2))
  248. c.Assert(strings.TrimSpace(out), checker.Equals, "80000\n100000")
  249. }