docker_api_build_test.go 7.2 KB

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