docker_api_build_test.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386
  1. package main
  2. import (
  3. "archive/tar"
  4. "bytes"
  5. "encoding/json"
  6. "fmt"
  7. "io"
  8. "io/ioutil"
  9. "net/http"
  10. "regexp"
  11. "strings"
  12. "github.com/docker/docker/api/types"
  13. "github.com/docker/docker/integration-cli/checker"
  14. "github.com/docker/docker/integration-cli/cli/build/fakecontext"
  15. "github.com/docker/docker/integration-cli/cli/build/fakegit"
  16. "github.com/docker/docker/integration-cli/cli/build/fakestorage"
  17. "github.com/docker/docker/integration-cli/request"
  18. "github.com/docker/docker/pkg/testutil"
  19. "github.com/go-check/check"
  20. "github.com/stretchr/testify/assert"
  21. "github.com/stretchr/testify/require"
  22. "golang.org/x/net/context"
  23. )
  24. func (s *DockerSuite) TestBuildAPIDockerFileRemote(c *check.C) {
  25. testRequires(c, NotUserNamespace)
  26. var testD string
  27. if testEnv.DaemonPlatform() == "windows" {
  28. testD = `FROM busybox
  29. RUN find / -name ba*
  30. RUN find /tmp/`
  31. } else {
  32. // -xdev is required because sysfs can cause EPERM
  33. testD = `FROM busybox
  34. RUN find / -xdev -name ba*
  35. RUN find /tmp/`
  36. }
  37. server := fakestorage.New(c, "", fakecontext.WithFiles(map[string]string{"testD": testD}))
  38. defer server.Close()
  39. res, body, err := request.Post("/build?dockerfile=baz&remote="+server.URL()+"/testD", request.JSON)
  40. c.Assert(err, checker.IsNil)
  41. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  42. buf, err := testutil.ReadBody(body)
  43. c.Assert(err, checker.IsNil)
  44. // Make sure Dockerfile exists.
  45. // Make sure 'baz' doesn't exist ANYWHERE despite being mentioned in the URL
  46. out := string(buf)
  47. c.Assert(out, checker.Contains, "RUN find /tmp")
  48. c.Assert(out, checker.Not(checker.Contains), "baz")
  49. }
  50. func (s *DockerSuite) TestBuildAPIRemoteTarballContext(c *check.C) {
  51. buffer := new(bytes.Buffer)
  52. tw := tar.NewWriter(buffer)
  53. defer tw.Close()
  54. dockerfile := []byte("FROM busybox")
  55. err := tw.WriteHeader(&tar.Header{
  56. Name: "Dockerfile",
  57. Size: int64(len(dockerfile)),
  58. })
  59. // failed to write tar file header
  60. c.Assert(err, checker.IsNil)
  61. _, err = tw.Write(dockerfile)
  62. // failed to write tar file content
  63. c.Assert(err, checker.IsNil)
  64. // failed to close tar archive
  65. c.Assert(tw.Close(), checker.IsNil)
  66. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  67. "testT.tar": buffer,
  68. }))
  69. defer server.Close()
  70. res, b, err := request.Post("/build?remote="+server.URL()+"/testT.tar", request.ContentType("application/tar"))
  71. c.Assert(err, checker.IsNil)
  72. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  73. b.Close()
  74. }
  75. func (s *DockerSuite) TestBuildAPIRemoteTarballContextWithCustomDockerfile(c *check.C) {
  76. buffer := new(bytes.Buffer)
  77. tw := tar.NewWriter(buffer)
  78. defer tw.Close()
  79. dockerfile := []byte(`FROM busybox
  80. RUN echo 'wrong'`)
  81. err := tw.WriteHeader(&tar.Header{
  82. Name: "Dockerfile",
  83. Size: int64(len(dockerfile)),
  84. })
  85. // failed to write tar file header
  86. c.Assert(err, checker.IsNil)
  87. _, err = tw.Write(dockerfile)
  88. // failed to write tar file content
  89. c.Assert(err, checker.IsNil)
  90. custom := []byte(`FROM busybox
  91. RUN echo 'right'
  92. `)
  93. err = tw.WriteHeader(&tar.Header{
  94. Name: "custom",
  95. Size: int64(len(custom)),
  96. })
  97. // failed to write tar file header
  98. c.Assert(err, checker.IsNil)
  99. _, err = tw.Write(custom)
  100. // failed to write tar file content
  101. c.Assert(err, checker.IsNil)
  102. // failed to close tar archive
  103. c.Assert(tw.Close(), checker.IsNil)
  104. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  105. "testT.tar": buffer,
  106. }))
  107. defer server.Close()
  108. url := "/build?dockerfile=custom&remote=" + server.URL() + "/testT.tar"
  109. res, body, err := request.Post(url, request.ContentType("application/tar"))
  110. c.Assert(err, checker.IsNil)
  111. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  112. defer body.Close()
  113. content, err := testutil.ReadBody(body)
  114. c.Assert(err, checker.IsNil)
  115. // Build used the wrong dockerfile.
  116. c.Assert(string(content), checker.Not(checker.Contains), "wrong")
  117. }
  118. func (s *DockerSuite) TestBuildAPILowerDockerfile(c *check.C) {
  119. git := fakegit.New(c, "repo", map[string]string{
  120. "dockerfile": `FROM busybox
  121. RUN echo from dockerfile`,
  122. }, false)
  123. defer git.Close()
  124. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  125. c.Assert(err, checker.IsNil)
  126. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  127. buf, err := testutil.ReadBody(body)
  128. c.Assert(err, checker.IsNil)
  129. out := string(buf)
  130. c.Assert(out, checker.Contains, "from dockerfile")
  131. }
  132. func (s *DockerSuite) TestBuildAPIBuildGitWithF(c *check.C) {
  133. git := fakegit.New(c, "repo", map[string]string{
  134. "baz": `FROM busybox
  135. RUN echo from baz`,
  136. "Dockerfile": `FROM busybox
  137. RUN echo from Dockerfile`,
  138. }, false)
  139. defer git.Close()
  140. // Make sure it tries to 'dockerfile' query param value
  141. res, body, err := request.Post("/build?dockerfile=baz&remote="+git.RepoURL, request.JSON)
  142. c.Assert(err, checker.IsNil)
  143. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  144. buf, err := testutil.ReadBody(body)
  145. c.Assert(err, checker.IsNil)
  146. out := string(buf)
  147. c.Assert(out, checker.Contains, "from baz")
  148. }
  149. func (s *DockerSuite) TestBuildAPIDoubleDockerfile(c *check.C) {
  150. testRequires(c, UnixCli) // dockerfile overwrites Dockerfile on Windows
  151. git := fakegit.New(c, "repo", map[string]string{
  152. "Dockerfile": `FROM busybox
  153. RUN echo from Dockerfile`,
  154. "dockerfile": `FROM busybox
  155. RUN echo from dockerfile`,
  156. }, false)
  157. defer git.Close()
  158. // Make sure it tries to 'dockerfile' query param value
  159. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  160. c.Assert(err, checker.IsNil)
  161. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  162. buf, err := testutil.ReadBody(body)
  163. c.Assert(err, checker.IsNil)
  164. out := string(buf)
  165. c.Assert(out, checker.Contains, "from Dockerfile")
  166. }
  167. func (s *DockerSuite) TestBuildAPIUnnormalizedTarPaths(c *check.C) {
  168. // Make sure that build context tars with entries of the form
  169. // x/./y don't cause caching false positives.
  170. buildFromTarContext := func(fileContents []byte) string {
  171. buffer := new(bytes.Buffer)
  172. tw := tar.NewWriter(buffer)
  173. defer tw.Close()
  174. dockerfile := []byte(`FROM busybox
  175. COPY dir /dir/`)
  176. err := tw.WriteHeader(&tar.Header{
  177. Name: "Dockerfile",
  178. Size: int64(len(dockerfile)),
  179. })
  180. //failed to write tar file header
  181. c.Assert(err, checker.IsNil)
  182. _, err = tw.Write(dockerfile)
  183. // failed to write Dockerfile in tar file content
  184. c.Assert(err, checker.IsNil)
  185. err = tw.WriteHeader(&tar.Header{
  186. Name: "dir/./file",
  187. Size: int64(len(fileContents)),
  188. })
  189. //failed to write tar file header
  190. c.Assert(err, checker.IsNil)
  191. _, err = tw.Write(fileContents)
  192. // failed to write file contents in tar file content
  193. c.Assert(err, checker.IsNil)
  194. // failed to close tar archive
  195. c.Assert(tw.Close(), checker.IsNil)
  196. res, body, err := request.Post("/build", request.RawContent(ioutil.NopCloser(buffer)), request.ContentType("application/x-tar"))
  197. c.Assert(err, checker.IsNil)
  198. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  199. out, err := testutil.ReadBody(body)
  200. c.Assert(err, checker.IsNil)
  201. lines := strings.Split(string(out), "\n")
  202. c.Assert(len(lines), checker.GreaterThan, 1)
  203. c.Assert(lines[len(lines)-2], checker.Matches, ".*Successfully built [0-9a-f]{12}.*")
  204. re := regexp.MustCompile("Successfully built ([0-9a-f]{12})")
  205. matches := re.FindStringSubmatch(lines[len(lines)-2])
  206. return matches[1]
  207. }
  208. imageA := buildFromTarContext([]byte("abc"))
  209. imageB := buildFromTarContext([]byte("def"))
  210. c.Assert(imageA, checker.Not(checker.Equals), imageB)
  211. }
  212. func (s *DockerSuite) TestBuildOnBuildWithCopy(c *check.C) {
  213. dockerfile := `
  214. FROM ` + minimalBaseImage() + ` as onbuildbase
  215. ONBUILD COPY file /file
  216. FROM onbuildbase
  217. `
  218. ctx := fakecontext.New(c, "",
  219. fakecontext.WithDockerfile(dockerfile),
  220. fakecontext.WithFile("file", "some content"),
  221. )
  222. defer ctx.Close()
  223. res, body, err := request.Post(
  224. "/build",
  225. request.RawContent(ctx.AsTarReader(c)),
  226. request.ContentType("application/x-tar"))
  227. c.Assert(err, checker.IsNil)
  228. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  229. out, err := testutil.ReadBody(body)
  230. c.Assert(err, checker.IsNil)
  231. c.Assert(string(out), checker.Contains, "Successfully built")
  232. }
  233. func (s *DockerSuite) TestBuildOnBuildCache(c *check.C) {
  234. build := func(dockerfile string) []byte {
  235. ctx := fakecontext.New(c, "",
  236. fakecontext.WithDockerfile(dockerfile),
  237. )
  238. defer ctx.Close()
  239. res, body, err := request.Post(
  240. "/build",
  241. request.RawContent(ctx.AsTarReader(c)),
  242. request.ContentType("application/x-tar"))
  243. require.NoError(c, err)
  244. assert.Equal(c, http.StatusOK, res.StatusCode)
  245. out, err := testutil.ReadBody(body)
  246. require.NoError(c, err)
  247. assert.Contains(c, string(out), "Successfully built")
  248. return out
  249. }
  250. dockerfile := `
  251. FROM ` + minimalBaseImage() + ` as onbuildbase
  252. ENV something=bar
  253. ONBUILD ENV foo=bar
  254. `
  255. build(dockerfile)
  256. dockerfile += "FROM onbuildbase"
  257. out := build(dockerfile)
  258. imageIDs := getImageIDsFromBuild(c, out)
  259. assert.Len(c, imageIDs, 2)
  260. parentID, childID := imageIDs[0], imageIDs[1]
  261. client, err := request.NewClient()
  262. require.NoError(c, err)
  263. // check parentID is correct
  264. image, _, err := client.ImageInspectWithRaw(context.Background(), childID)
  265. require.NoError(c, err)
  266. assert.Equal(c, parentID, image.Parent)
  267. }
  268. func (s *DockerRegistrySuite) TestBuildCopyFromForcePull(c *check.C) {
  269. client, err := request.NewClient()
  270. require.NoError(c, err)
  271. repoName := fmt.Sprintf("%v/dockercli/busybox", privateRegistryURL)
  272. // tag the image to upload it to the private registry
  273. err = client.ImageTag(context.TODO(), "busybox", repoName)
  274. assert.Nil(c, err)
  275. // push the image to the registry
  276. rc, err := client.ImagePush(context.TODO(), repoName, types.ImagePushOptions{RegistryAuth: "{}"})
  277. assert.Nil(c, err)
  278. _, err = io.Copy(ioutil.Discard, rc)
  279. assert.Nil(c, err)
  280. dockerfile := fmt.Sprintf(`
  281. FROM %s AS foo
  282. RUN touch abc
  283. FROM %s
  284. COPY --from=foo /abc /
  285. `, repoName, repoName)
  286. ctx := fakecontext.New(c, "",
  287. fakecontext.WithDockerfile(dockerfile),
  288. )
  289. defer ctx.Close()
  290. res, body, err := request.Post(
  291. "/build?pull=1",
  292. request.RawContent(ctx.AsTarReader(c)),
  293. request.ContentType("application/x-tar"))
  294. require.NoError(c, err)
  295. assert.Equal(c, http.StatusOK, res.StatusCode)
  296. out, err := testutil.ReadBody(body)
  297. require.NoError(c, err)
  298. assert.Contains(c, string(out), "Successfully built")
  299. }
  300. type buildLine struct {
  301. Stream string
  302. Aux struct {
  303. ID string
  304. }
  305. }
  306. func getImageIDsFromBuild(c *check.C, output []byte) []string {
  307. ids := []string{}
  308. for _, line := range bytes.Split(output, []byte("\n")) {
  309. if len(line) == 0 {
  310. continue
  311. }
  312. entry := buildLine{}
  313. require.NoError(c, json.Unmarshal(line, &entry))
  314. if entry.Aux.ID != "" {
  315. ids = append(ids, entry.Aux.ID)
  316. }
  317. }
  318. return ids
  319. }