update_linux_test.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. package container // import "github.com/docker/docker/integration/container"
  2. import (
  3. "context"
  4. "strconv"
  5. "strings"
  6. "testing"
  7. "time"
  8. "github.com/docker/docker/api/types"
  9. containertypes "github.com/docker/docker/api/types/container"
  10. "github.com/docker/docker/integration/internal/container"
  11. "gotest.tools/assert"
  12. is "gotest.tools/assert/cmp"
  13. "gotest.tools/poll"
  14. "gotest.tools/skip"
  15. )
  16. func TestUpdateMemory(t *testing.T) {
  17. skip.If(t, testEnv.DaemonInfo.OSType == "windows")
  18. skip.If(t, !testEnv.DaemonInfo.MemoryLimit)
  19. skip.If(t, !testEnv.DaemonInfo.SwapLimit)
  20. defer setupTest(t)()
  21. client := testEnv.APIClient()
  22. ctx := context.Background()
  23. cID := container.Run(t, ctx, client, func(c *container.TestContainerConfig) {
  24. c.HostConfig.Resources = containertypes.Resources{
  25. Memory: 200 * 1024 * 1024,
  26. }
  27. })
  28. poll.WaitOn(t, container.IsInState(ctx, client, cID, "running"), poll.WithDelay(100*time.Millisecond))
  29. const (
  30. setMemory int64 = 314572800
  31. setMemorySwap int64 = 524288000
  32. )
  33. _, err := client.ContainerUpdate(ctx, cID, containertypes.UpdateConfig{
  34. Resources: containertypes.Resources{
  35. Memory: setMemory,
  36. MemorySwap: setMemorySwap,
  37. },
  38. })
  39. assert.NilError(t, err)
  40. inspect, err := client.ContainerInspect(ctx, cID)
  41. assert.NilError(t, err)
  42. assert.Check(t, is.Equal(setMemory, inspect.HostConfig.Memory))
  43. assert.Check(t, is.Equal(setMemorySwap, inspect.HostConfig.MemorySwap))
  44. res, err := container.Exec(ctx, client, cID,
  45. []string{"cat", "/sys/fs/cgroup/memory/memory.limit_in_bytes"})
  46. assert.NilError(t, err)
  47. assert.Assert(t, is.Len(res.Stderr(), 0))
  48. assert.Equal(t, 0, res.ExitCode)
  49. assert.Check(t, is.Equal(strconv.FormatInt(setMemory, 10), strings.TrimSpace(res.Stdout())))
  50. res, err = container.Exec(ctx, client, cID,
  51. []string{"cat", "/sys/fs/cgroup/memory/memory.memsw.limit_in_bytes"})
  52. assert.NilError(t, err)
  53. assert.Assert(t, is.Len(res.Stderr(), 0))
  54. assert.Equal(t, 0, res.ExitCode)
  55. assert.Check(t, is.Equal(strconv.FormatInt(setMemorySwap, 10), strings.TrimSpace(res.Stdout())))
  56. }
  57. func TestUpdateCPUQuota(t *testing.T) {
  58. defer setupTest(t)()
  59. client := testEnv.APIClient()
  60. ctx := context.Background()
  61. cID := container.Run(t, ctx, client)
  62. for _, test := range []struct {
  63. desc string
  64. update int64
  65. }{
  66. {desc: "some random value", update: 15000},
  67. {desc: "a higher value", update: 20000},
  68. {desc: "a lower value", update: 10000},
  69. {desc: "unset value", update: -1},
  70. } {
  71. _, err := client.ContainerUpdate(ctx, cID, containertypes.UpdateConfig{
  72. Resources: containertypes.Resources{
  73. CPUQuota: test.update,
  74. },
  75. })
  76. assert.NilError(t, err)
  77. inspect, err := client.ContainerInspect(ctx, cID)
  78. assert.NilError(t, err)
  79. assert.Check(t, is.Equal(test.update, inspect.HostConfig.CPUQuota))
  80. res, err := container.Exec(ctx, client, cID,
  81. []string{"/bin/cat", "/sys/fs/cgroup/cpu/cpu.cfs_quota_us"})
  82. assert.NilError(t, err)
  83. assert.Assert(t, is.Len(res.Stderr(), 0))
  84. assert.Equal(t, 0, res.ExitCode)
  85. assert.Check(t, is.Equal(strconv.FormatInt(test.update, 10), strings.TrimSpace(res.Stdout())))
  86. }
  87. }
  88. func TestUpdatePidsLimit(t *testing.T) {
  89. skip.If(t, testEnv.DaemonInfo.OSType == "windows")
  90. skip.If(t, !testEnv.DaemonInfo.PidsLimit)
  91. defer setupTest(t)()
  92. client := testEnv.APIClient()
  93. ctx := context.Background()
  94. cID := container.Run(t, ctx, client)
  95. intPtr := func(i int64) *int64 {
  96. return &i
  97. }
  98. for _, test := range []struct {
  99. desc string
  100. update *int64
  101. expect int64
  102. expectCg string
  103. }{
  104. {desc: "update from none", update: intPtr(32), expect: 32, expectCg: "32"},
  105. {desc: "no change", update: nil, expectCg: "32"},
  106. {desc: "update lower", update: intPtr(16), expect: 16, expectCg: "16"},
  107. {desc: "unset limit", update: intPtr(0), expect: 0, expectCg: "max"},
  108. } {
  109. var before types.ContainerJSON
  110. if test.update == nil {
  111. var err error
  112. before, err = client.ContainerInspect(ctx, cID)
  113. assert.NilError(t, err)
  114. }
  115. t.Run(test.desc, func(t *testing.T) {
  116. _, err := client.ContainerUpdate(ctx, cID, containertypes.UpdateConfig{
  117. Resources: containertypes.Resources{
  118. PidsLimit: test.update,
  119. },
  120. })
  121. assert.NilError(t, err)
  122. inspect, err := client.ContainerInspect(ctx, cID)
  123. assert.NilError(t, err)
  124. assert.Assert(t, inspect.HostConfig.Resources.PidsLimit != nil)
  125. if test.update == nil {
  126. assert.Assert(t, before.HostConfig.Resources.PidsLimit != nil)
  127. assert.Equal(t, *before.HostConfig.Resources.PidsLimit, *inspect.HostConfig.Resources.PidsLimit)
  128. } else {
  129. assert.Equal(t, *inspect.HostConfig.Resources.PidsLimit, test.expect)
  130. }
  131. ctx, cancel := context.WithTimeout(ctx, 60*time.Second)
  132. defer cancel()
  133. res, err := container.Exec(ctx, client, cID, []string{"cat", "/sys/fs/cgroup/pids/pids.max"})
  134. assert.NilError(t, err)
  135. assert.Assert(t, is.Len(res.Stderr(), 0))
  136. out := strings.TrimSpace(res.Stdout())
  137. assert.Equal(t, out, test.expectCg)
  138. })
  139. }
  140. }