docker_api_build_test.go 6.7 KB

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