run_linux_test.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. package container // import "github.com/docker/docker/integration/container"
  2. import (
  3. "bytes"
  4. "context"
  5. "io"
  6. "os"
  7. "path/filepath"
  8. "strings"
  9. "testing"
  10. "time"
  11. "github.com/docker/docker/api/types"
  12. containertypes "github.com/docker/docker/api/types/container"
  13. "github.com/docker/docker/api/types/versions"
  14. "github.com/docker/docker/integration/internal/container"
  15. net "github.com/docker/docker/integration/internal/network"
  16. "github.com/docker/docker/pkg/system"
  17. "golang.org/x/sys/unix"
  18. "gotest.tools/v3/assert"
  19. is "gotest.tools/v3/assert/cmp"
  20. "gotest.tools/v3/poll"
  21. "gotest.tools/v3/skip"
  22. )
  23. func TestNISDomainname(t *testing.T) {
  24. // Older versions of the daemon would concatenate hostname and domainname,
  25. // so hostname "foobar" and domainname "baz.cyphar.com" would produce
  26. // `foobar.baz.cyphar.com` as hostname.
  27. skip.If(t, versions.LessThan(testEnv.DaemonAPIVersion(), "1.40"), "skip test from new feature")
  28. skip.If(t, testEnv.DaemonInfo.OSType != "linux")
  29. // Rootless supports custom Hostname but doesn't support custom Domainname
  30. // OCI runtime create failed: container_linux.go:349: starting container process caused "process_linux.go:449: container init caused \
  31. // "write sysctl key kernel.domainname: open /proc/sys/kernel/domainname: permission denied\"": unknown.
  32. skip.If(t, testEnv.IsRootless, "rootless mode doesn't support setting Domainname (TODO: https://github.com/moby/moby/issues/40632)")
  33. defer setupTest(t)()
  34. client := testEnv.APIClient()
  35. ctx := context.Background()
  36. const (
  37. hostname = "foobar"
  38. domainname = "baz.cyphar.com"
  39. )
  40. cID := container.Run(ctx, t, client, func(c *container.TestContainerConfig) {
  41. c.Config.Hostname = hostname
  42. c.Config.Domainname = domainname
  43. })
  44. poll.WaitOn(t, container.IsInState(ctx, client, cID, "running"), poll.WithDelay(100*time.Millisecond))
  45. inspect, err := client.ContainerInspect(ctx, cID)
  46. assert.NilError(t, err)
  47. assert.Check(t, is.Equal(hostname, inspect.Config.Hostname))
  48. assert.Check(t, is.Equal(domainname, inspect.Config.Domainname))
  49. // Check hostname.
  50. res, err := container.Exec(ctx, client, cID,
  51. []string{"cat", "/proc/sys/kernel/hostname"})
  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(hostname, strings.TrimSpace(res.Stdout())))
  56. // Check domainname.
  57. res, err = container.Exec(ctx, client, cID,
  58. []string{"cat", "/proc/sys/kernel/domainname"})
  59. assert.NilError(t, err)
  60. assert.Assert(t, is.Len(res.Stderr(), 0))
  61. assert.Equal(t, 0, res.ExitCode)
  62. assert.Check(t, is.Equal(domainname, strings.TrimSpace(res.Stdout())))
  63. }
  64. func TestHostnameDnsResolution(t *testing.T) {
  65. skip.If(t, testEnv.DaemonInfo.OSType != "linux")
  66. defer setupTest(t)()
  67. client := testEnv.APIClient()
  68. ctx := context.Background()
  69. const (
  70. hostname = "foobar"
  71. )
  72. // using user defined network as we want to use internal DNS
  73. netName := "foobar-net"
  74. net.CreateNoError(context.Background(), t, client, netName, net.WithDriver("bridge"))
  75. cID := container.Run(ctx, t, client, func(c *container.TestContainerConfig) {
  76. c.Config.Hostname = hostname
  77. c.HostConfig.NetworkMode = containertypes.NetworkMode(netName)
  78. })
  79. poll.WaitOn(t, container.IsInState(ctx, client, cID, "running"), poll.WithDelay(100*time.Millisecond))
  80. inspect, err := client.ContainerInspect(ctx, cID)
  81. assert.NilError(t, err)
  82. assert.Check(t, is.Equal(hostname, inspect.Config.Hostname))
  83. // Clear hosts file so ping will use DNS for hostname resolution
  84. res, err := container.Exec(ctx, client, cID,
  85. []string{"sh", "-c", "echo 127.0.0.1 localhost | tee /etc/hosts && ping -c 1 foobar"})
  86. assert.NilError(t, err)
  87. assert.Check(t, is.Equal("", res.Stderr()))
  88. assert.Equal(t, 0, res.ExitCode)
  89. }
  90. func TestUnprivilegedPortsAndPing(t *testing.T) {
  91. skip.If(t, testEnv.DaemonInfo.OSType != "linux")
  92. skip.If(t, testEnv.IsRootless, "rootless mode doesn't support setting net.ipv4.ping_group_range and net.ipv4.ip_unprivileged_port_start")
  93. defer setupTest(t)()
  94. client := testEnv.APIClient()
  95. ctx := context.Background()
  96. cID := container.Run(ctx, t, client, func(c *container.TestContainerConfig) {
  97. c.Config.User = "1000:1000"
  98. })
  99. poll.WaitOn(t, container.IsInState(ctx, client, cID, "running"), poll.WithDelay(100*time.Millisecond))
  100. // Check net.ipv4.ping_group_range.
  101. res, err := container.Exec(ctx, client, cID, []string{"cat", "/proc/sys/net/ipv4/ping_group_range"})
  102. assert.NilError(t, err)
  103. assert.Assert(t, is.Len(res.Stderr(), 0))
  104. assert.Equal(t, 0, res.ExitCode)
  105. assert.Equal(t, `0 2147483647`, strings.TrimSpace(res.Stdout()))
  106. // Check net.ipv4.ip_unprivileged_port_start.
  107. res, err = container.Exec(ctx, client, cID, []string{"cat", "/proc/sys/net/ipv4/ip_unprivileged_port_start"})
  108. assert.NilError(t, err)
  109. assert.Assert(t, is.Len(res.Stderr(), 0))
  110. assert.Equal(t, 0, res.ExitCode)
  111. assert.Equal(t, "0", strings.TrimSpace(res.Stdout()))
  112. }
  113. func TestPrivilegedHostDevices(t *testing.T) {
  114. // Host devices are linux only. Also it creates host devices,
  115. // so needs to be same host.
  116. skip.If(t, testEnv.IsRemoteDaemon)
  117. skip.If(t, testEnv.DaemonInfo.OSType != "linux")
  118. defer setupTest(t)()
  119. client := testEnv.APIClient()
  120. ctx := context.Background()
  121. const (
  122. devTest = "/dev/test"
  123. devRootOnlyTest = "/dev/root-only/test"
  124. )
  125. // Create Null devices.
  126. if err := system.Mknod(devTest, unix.S_IFCHR|0600, int(system.Mkdev(1, 3))); err != nil {
  127. t.Fatal(err)
  128. }
  129. defer os.Remove(devTest)
  130. if err := os.Mkdir(filepath.Dir(devRootOnlyTest), 0700); err != nil {
  131. t.Fatal(err)
  132. }
  133. defer os.RemoveAll(filepath.Dir(devRootOnlyTest))
  134. if err := system.Mknod(devRootOnlyTest, unix.S_IFCHR|0600, int(system.Mkdev(1, 3))); err != nil {
  135. t.Fatal(err)
  136. }
  137. defer os.Remove(devRootOnlyTest)
  138. cID := container.Run(ctx, t, client, container.WithPrivileged(true))
  139. poll.WaitOn(t, container.IsInState(ctx, client, cID, "running"), poll.WithDelay(100*time.Millisecond))
  140. // Check test device.
  141. res, err := container.Exec(ctx, client, cID, []string{"ls", devTest})
  142. assert.NilError(t, err)
  143. assert.Equal(t, 0, res.ExitCode)
  144. assert.Check(t, is.Equal(strings.TrimSpace(res.Stdout()), devTest))
  145. // Check root-only test device.
  146. res, err = container.Exec(ctx, client, cID, []string{"ls", devRootOnlyTest})
  147. assert.NilError(t, err)
  148. if testEnv.IsRootless() {
  149. assert.Equal(t, 1, res.ExitCode)
  150. assert.Check(t, is.Contains(res.Stderr(), "No such file or directory"))
  151. } else {
  152. assert.Equal(t, 0, res.ExitCode)
  153. assert.Check(t, is.Equal(strings.TrimSpace(res.Stdout()), devRootOnlyTest))
  154. }
  155. }
  156. func TestRunConsoleSize(t *testing.T) {
  157. skip.If(t, testEnv.DaemonInfo.OSType != "linux")
  158. skip.If(t, versions.LessThan(testEnv.DaemonAPIVersion(), "1.42"), "skip test from new feature")
  159. defer setupTest(t)()
  160. client := testEnv.APIClient()
  161. ctx := context.Background()
  162. cID := container.Run(ctx, t, client,
  163. container.WithTty(true),
  164. container.WithImage("busybox"),
  165. container.WithCmd("stty", "size"),
  166. container.WithConsoleSize(57, 123),
  167. )
  168. poll.WaitOn(t, container.IsStopped(ctx, client, cID), poll.WithDelay(100*time.Millisecond))
  169. out, err := client.ContainerLogs(ctx, cID, types.ContainerLogsOptions{ShowStdout: true})
  170. assert.NilError(t, err)
  171. defer out.Close()
  172. var b bytes.Buffer
  173. _, err = io.Copy(&b, out)
  174. assert.NilError(t, err)
  175. assert.Equal(t, strings.TrimSpace(b.String()), "123 57")
  176. }