internals_test.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. package dockerfile // import "github.com/docker/docker/builder/dockerfile"
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "runtime"
  7. "testing"
  8. "github.com/docker/docker/api/types"
  9. "github.com/docker/docker/api/types/backend"
  10. "github.com/docker/docker/api/types/container"
  11. "github.com/docker/docker/builder"
  12. "github.com/docker/docker/builder/remotecontext"
  13. "github.com/docker/docker/image"
  14. "github.com/docker/docker/layer"
  15. "github.com/docker/docker/pkg/archive"
  16. "github.com/docker/go-connections/nat"
  17. "github.com/opencontainers/go-digest"
  18. "gotest.tools/v3/assert"
  19. is "gotest.tools/v3/assert/cmp"
  20. "gotest.tools/v3/skip"
  21. )
  22. func TestEmptyDockerfile(t *testing.T) {
  23. contextDir, cleanup := createTestTempDir(t, "", "builder-dockerfile-test")
  24. defer cleanup()
  25. createTestTempFile(t, contextDir, builder.DefaultDockerfileName, "", 0o777)
  26. readAndCheckDockerfile(t, "emptyDockerfile", contextDir, "", "the Dockerfile (Dockerfile) cannot be empty")
  27. }
  28. func TestSymlinkDockerfile(t *testing.T) {
  29. contextDir, cleanup := createTestTempDir(t, "", "builder-dockerfile-test")
  30. defer cleanup()
  31. createTestSymlink(t, contextDir, builder.DefaultDockerfileName, "/etc/passwd")
  32. // The reason the error is "Cannot locate specified Dockerfile" is because
  33. // in the builder, the symlink is resolved within the context, therefore
  34. // Dockerfile -> /etc/passwd becomes etc/passwd from the context which is
  35. // a nonexistent file.
  36. expectedError := fmt.Sprintf("Cannot locate specified Dockerfile: %s", builder.DefaultDockerfileName)
  37. readAndCheckDockerfile(t, "symlinkDockerfile", contextDir, builder.DefaultDockerfileName, expectedError)
  38. }
  39. func TestDockerfileOutsideTheBuildContext(t *testing.T) {
  40. contextDir, cleanup := createTestTempDir(t, "", "builder-dockerfile-test")
  41. defer cleanup()
  42. expectedError := "path outside the build context: ../../Dockerfile ()"
  43. if runtime.GOOS == "windows" {
  44. expectedError = "failed to resolve scoped path ../../Dockerfile ()"
  45. }
  46. readAndCheckDockerfile(t, "DockerfileOutsideTheBuildContext", contextDir, "../../Dockerfile", expectedError)
  47. }
  48. func TestNonExistingDockerfile(t *testing.T) {
  49. contextDir, cleanup := createTestTempDir(t, "", "builder-dockerfile-test")
  50. defer cleanup()
  51. expectedError := "Cannot locate specified Dockerfile: Dockerfile"
  52. readAndCheckDockerfile(t, "NonExistingDockerfile", contextDir, "Dockerfile", expectedError)
  53. }
  54. func readAndCheckDockerfile(t *testing.T, testName, contextDir, dockerfilePath, expectedError string) {
  55. if runtime.GOOS != "windows" {
  56. skip.If(t, os.Getuid() != 0, "skipping test that requires root")
  57. }
  58. tarStream, err := archive.Tar(contextDir, archive.Uncompressed)
  59. assert.NilError(t, err)
  60. defer func() {
  61. if err = tarStream.Close(); err != nil {
  62. t.Fatalf("Error when closing tar stream: %s", err)
  63. }
  64. }()
  65. if dockerfilePath == "" { // handled in BuildWithContext
  66. dockerfilePath = builder.DefaultDockerfileName
  67. }
  68. config := backend.BuildConfig{
  69. Options: &types.ImageBuildOptions{Dockerfile: dockerfilePath},
  70. Source: tarStream,
  71. }
  72. _, _, err = remotecontext.Detect(config)
  73. assert.Check(t, is.ErrorContains(err, expectedError))
  74. }
  75. func TestCopyRunConfig(t *testing.T) {
  76. defaultEnv := []string{"foo=1"}
  77. defaultCmd := []string{"old"}
  78. testcases := []struct {
  79. doc string
  80. modifiers []runConfigModifier
  81. expected *container.Config
  82. }{
  83. {
  84. doc: "Set the command",
  85. modifiers: []runConfigModifier{withCmd([]string{"new"})},
  86. expected: &container.Config{
  87. Cmd: []string{"new"},
  88. Env: defaultEnv,
  89. },
  90. },
  91. {
  92. doc: "Set the command to a comment",
  93. modifiers: []runConfigModifier{withCmdComment("comment", runtime.GOOS)},
  94. expected: &container.Config{
  95. Cmd: append(defaultShellForOS(runtime.GOOS), "#(nop) ", "comment"),
  96. Env: defaultEnv,
  97. },
  98. },
  99. {
  100. doc: "Set the command and env",
  101. modifiers: []runConfigModifier{
  102. withCmd([]string{"new"}),
  103. withEnv([]string{"one", "two"}),
  104. },
  105. expected: &container.Config{
  106. Cmd: []string{"new"},
  107. Env: []string{"one", "two"},
  108. },
  109. },
  110. }
  111. for _, testcase := range testcases {
  112. runConfig := &container.Config{
  113. Cmd: defaultCmd,
  114. Env: defaultEnv,
  115. }
  116. runConfigCopy := copyRunConfig(runConfig, testcase.modifiers...)
  117. assert.Check(t, is.DeepEqual(testcase.expected, runConfigCopy), testcase.doc)
  118. // Assert the original was not modified
  119. assert.Check(t, runConfig != runConfigCopy, testcase.doc)
  120. }
  121. }
  122. func fullMutableRunConfig() *container.Config {
  123. return &container.Config{
  124. Cmd: []string{"command", "arg1"},
  125. Env: []string{"env1=foo", "env2=bar"},
  126. ExposedPorts: nat.PortSet{
  127. "1000/tcp": {},
  128. "1001/tcp": {},
  129. },
  130. Volumes: map[string]struct{}{
  131. "one": {},
  132. "two": {},
  133. },
  134. Entrypoint: []string{"entry", "arg1"},
  135. OnBuild: []string{"first", "next"},
  136. Labels: map[string]string{
  137. "label1": "value1",
  138. "label2": "value2",
  139. },
  140. Shell: []string{"shell", "-c"},
  141. }
  142. }
  143. func TestDeepCopyRunConfig(t *testing.T) {
  144. runConfig := fullMutableRunConfig()
  145. copy := copyRunConfig(runConfig)
  146. assert.Check(t, is.DeepEqual(fullMutableRunConfig(), copy))
  147. copy.Cmd[1] = "arg2"
  148. copy.Env[1] = "env2=new"
  149. copy.ExposedPorts["10002"] = struct{}{}
  150. copy.Volumes["three"] = struct{}{}
  151. copy.Entrypoint[1] = "arg2"
  152. copy.OnBuild[0] = "start"
  153. copy.Labels["label3"] = "value3"
  154. copy.Shell[0] = "sh"
  155. assert.Check(t, is.DeepEqual(fullMutableRunConfig(), runConfig))
  156. }
  157. type MockRWLayer struct{}
  158. func (l *MockRWLayer) Release() error { return nil }
  159. func (l *MockRWLayer) Root() string { return "" }
  160. func (l *MockRWLayer) Commit() (builder.ROLayer, error) {
  161. return &MockROLayer{
  162. diffID: layer.DiffID(digest.Digest("sha256:1234")),
  163. }, nil
  164. }
  165. type MockROLayer struct {
  166. diffID layer.DiffID
  167. }
  168. func (l *MockROLayer) ContentStoreDigest() digest.Digest { return "" }
  169. func (l *MockROLayer) Release() error { return nil }
  170. func (l *MockROLayer) NewRWLayer() (builder.RWLayer, error) { return nil, nil }
  171. func (l *MockROLayer) DiffID() layer.DiffID { return l.diffID }
  172. func getMockBuildBackend() builder.Backend {
  173. return &MockBackend{}
  174. }
  175. func TestExportImage(t *testing.T) {
  176. ds := newDispatchState(NewBuildArgs(map[string]*string{}))
  177. layer := &MockRWLayer{}
  178. parentImage := &image.Image{
  179. V1Image: image.V1Image{
  180. OS: "linux",
  181. Architecture: "arm64",
  182. Variant: "v8",
  183. },
  184. }
  185. runConfig := &container.Config{}
  186. b := &Builder{
  187. imageSources: getMockImageSource(nil, nil, nil),
  188. docker: getMockBuildBackend(),
  189. }
  190. err := b.exportImage(context.TODO(), ds, layer, parentImage, runConfig)
  191. assert.NilError(t, err)
  192. }