docker_api_build_test.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. package main
  2. import (
  3. "archive/tar"
  4. "bytes"
  5. "io/ioutil"
  6. "net/http"
  7. "regexp"
  8. "strings"
  9. "github.com/docker/docker/integration-cli/checker"
  10. "github.com/docker/docker/integration-cli/cli/build/fakecontext"
  11. "github.com/docker/docker/integration-cli/cli/build/fakestorage"
  12. "github.com/docker/docker/integration-cli/request"
  13. "github.com/docker/docker/pkg/testutil"
  14. "github.com/go-check/check"
  15. )
  16. func (s *DockerSuite) TestBuildAPIDockerFileRemote(c *check.C) {
  17. testRequires(c, NotUserNamespace)
  18. var testD string
  19. if testEnv.DaemonPlatform() == "windows" {
  20. testD = `FROM busybox
  21. COPY * /tmp/
  22. RUN find / -name ba*
  23. RUN find /tmp/`
  24. } else {
  25. // -xdev is required because sysfs can cause EPERM
  26. testD = `FROM busybox
  27. COPY * /tmp/
  28. RUN find / -xdev -name ba*
  29. RUN find /tmp/`
  30. }
  31. server := fakestorage.New(c, "", fakecontext.WithFiles(map[string]string{"testD": testD}))
  32. defer server.Close()
  33. res, body, err := request.Post("/build?dockerfile=baz&remote="+server.URL()+"/testD", request.JSON)
  34. c.Assert(err, checker.IsNil)
  35. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  36. buf, err := testutil.ReadBody(body)
  37. c.Assert(err, checker.IsNil)
  38. // Make sure Dockerfile exists.
  39. // Make sure 'baz' doesn't exist ANYWHERE despite being mentioned in the URL
  40. out := string(buf)
  41. c.Assert(out, checker.Contains, "/tmp/Dockerfile")
  42. c.Assert(out, checker.Not(checker.Contains), "baz")
  43. }
  44. func (s *DockerSuite) TestBuildAPIRemoteTarballContext(c *check.C) {
  45. buffer := new(bytes.Buffer)
  46. tw := tar.NewWriter(buffer)
  47. defer tw.Close()
  48. dockerfile := []byte("FROM busybox")
  49. err := tw.WriteHeader(&tar.Header{
  50. Name: "Dockerfile",
  51. Size: int64(len(dockerfile)),
  52. })
  53. // failed to write tar file header
  54. c.Assert(err, checker.IsNil)
  55. _, err = tw.Write(dockerfile)
  56. // failed to write tar file content
  57. c.Assert(err, checker.IsNil)
  58. // failed to close tar archive
  59. c.Assert(tw.Close(), checker.IsNil)
  60. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  61. "testT.tar": buffer,
  62. }))
  63. defer server.Close()
  64. res, b, err := request.Post("/build?remote="+server.URL()+"/testT.tar", request.ContentType("application/tar"))
  65. c.Assert(err, checker.IsNil)
  66. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  67. b.Close()
  68. }
  69. func (s *DockerSuite) TestBuildAPIRemoteTarballContextWithCustomDockerfile(c *check.C) {
  70. buffer := new(bytes.Buffer)
  71. tw := tar.NewWriter(buffer)
  72. defer tw.Close()
  73. dockerfile := []byte(`FROM busybox
  74. RUN echo 'wrong'`)
  75. err := tw.WriteHeader(&tar.Header{
  76. Name: "Dockerfile",
  77. Size: int64(len(dockerfile)),
  78. })
  79. // failed to write tar file header
  80. c.Assert(err, checker.IsNil)
  81. _, err = tw.Write(dockerfile)
  82. // failed to write tar file content
  83. c.Assert(err, checker.IsNil)
  84. custom := []byte(`FROM busybox
  85. RUN echo 'right'
  86. `)
  87. err = tw.WriteHeader(&tar.Header{
  88. Name: "custom",
  89. Size: int64(len(custom)),
  90. })
  91. // failed to write tar file header
  92. c.Assert(err, checker.IsNil)
  93. _, err = tw.Write(custom)
  94. // failed to write tar file content
  95. c.Assert(err, checker.IsNil)
  96. // failed to close tar archive
  97. c.Assert(tw.Close(), checker.IsNil)
  98. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  99. "testT.tar": buffer,
  100. }))
  101. defer server.Close()
  102. url := "/build?dockerfile=custom&remote=" + server.URL() + "/testT.tar"
  103. res, body, err := request.Post(url, request.ContentType("application/tar"))
  104. c.Assert(err, checker.IsNil)
  105. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  106. defer body.Close()
  107. content, err := testutil.ReadBody(body)
  108. c.Assert(err, checker.IsNil)
  109. // Build used the wrong dockerfile.
  110. c.Assert(string(content), checker.Not(checker.Contains), "wrong")
  111. }
  112. func (s *DockerSuite) TestBuildAPILowerDockerfile(c *check.C) {
  113. git := newFakeGit(c, "repo", map[string]string{
  114. "dockerfile": `FROM busybox
  115. RUN echo from dockerfile`,
  116. }, false)
  117. defer git.Close()
  118. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  119. c.Assert(err, checker.IsNil)
  120. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  121. buf, err := testutil.ReadBody(body)
  122. c.Assert(err, checker.IsNil)
  123. out := string(buf)
  124. c.Assert(out, checker.Contains, "from dockerfile")
  125. }
  126. func (s *DockerSuite) TestBuildAPIBuildGitWithF(c *check.C) {
  127. git := newFakeGit(c, "repo", map[string]string{
  128. "baz": `FROM busybox
  129. RUN echo from baz`,
  130. "Dockerfile": `FROM busybox
  131. RUN echo from Dockerfile`,
  132. }, false)
  133. defer git.Close()
  134. // Make sure it tries to 'dockerfile' query param value
  135. res, body, err := request.Post("/build?dockerfile=baz&remote="+git.RepoURL, request.JSON)
  136. c.Assert(err, checker.IsNil)
  137. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  138. buf, err := testutil.ReadBody(body)
  139. c.Assert(err, checker.IsNil)
  140. out := string(buf)
  141. c.Assert(out, checker.Contains, "from baz")
  142. }
  143. func (s *DockerSuite) TestBuildAPIDoubleDockerfile(c *check.C) {
  144. testRequires(c, UnixCli) // dockerfile overwrites Dockerfile on Windows
  145. git := newFakeGit(c, "repo", map[string]string{
  146. "Dockerfile": `FROM busybox
  147. RUN echo from Dockerfile`,
  148. "dockerfile": `FROM busybox
  149. RUN echo from dockerfile`,
  150. }, false)
  151. defer git.Close()
  152. // Make sure it tries to 'dockerfile' query param value
  153. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  154. c.Assert(err, checker.IsNil)
  155. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  156. buf, err := testutil.ReadBody(body)
  157. c.Assert(err, checker.IsNil)
  158. out := string(buf)
  159. c.Assert(out, checker.Contains, "from Dockerfile")
  160. }
  161. func (s *DockerSuite) TestBuildAPIUnnormalizedTarPaths(c *check.C) {
  162. // Make sure that build context tars with entries of the form
  163. // x/./y don't cause caching false positives.
  164. buildFromTarContext := func(fileContents []byte) string {
  165. buffer := new(bytes.Buffer)
  166. tw := tar.NewWriter(buffer)
  167. defer tw.Close()
  168. dockerfile := []byte(`FROM busybox
  169. COPY dir /dir/`)
  170. err := tw.WriteHeader(&tar.Header{
  171. Name: "Dockerfile",
  172. Size: int64(len(dockerfile)),
  173. })
  174. //failed to write tar file header
  175. c.Assert(err, checker.IsNil)
  176. _, err = tw.Write(dockerfile)
  177. // failed to write Dockerfile in tar file content
  178. c.Assert(err, checker.IsNil)
  179. err = tw.WriteHeader(&tar.Header{
  180. Name: "dir/./file",
  181. Size: int64(len(fileContents)),
  182. })
  183. //failed to write tar file header
  184. c.Assert(err, checker.IsNil)
  185. _, err = tw.Write(fileContents)
  186. // failed to write file contents in tar file content
  187. c.Assert(err, checker.IsNil)
  188. // failed to close tar archive
  189. c.Assert(tw.Close(), checker.IsNil)
  190. res, body, err := request.Post("/build", request.RawContent(ioutil.NopCloser(buffer)), request.ContentType("application/x-tar"))
  191. c.Assert(err, checker.IsNil)
  192. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  193. out, err := testutil.ReadBody(body)
  194. c.Assert(err, checker.IsNil)
  195. lines := strings.Split(string(out), "\n")
  196. c.Assert(len(lines), checker.GreaterThan, 1)
  197. c.Assert(lines[len(lines)-2], checker.Matches, ".*Successfully built [0-9a-f]{12}.*")
  198. re := regexp.MustCompile("Successfully built ([0-9a-f]{12})")
  199. matches := re.FindStringSubmatch(lines[len(lines)-2])
  200. return matches[1]
  201. }
  202. imageA := buildFromTarContext([]byte("abc"))
  203. imageB := buildFromTarContext([]byte("def"))
  204. c.Assert(imageA, checker.Not(checker.Equals), imageB)
  205. }