docker_api_build_test.go 7.8 KB

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