docker_api_build_test.go 6.9 KB

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