docker_api_build_test.go 7.1 KB

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