update_linux_test.go 5.5 KB

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