docker_api_build_test.go 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580
  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/go-check/check"
  19. "github.com/moby/buildkit/session"
  20. "github.com/moby/buildkit/session/filesync"
  21. "github.com/stretchr/testify/assert"
  22. "github.com/stretchr/testify/require"
  23. "golang.org/x/net/context"
  24. "golang.org/x/sync/errgroup"
  25. )
  26. func (s *DockerSuite) TestBuildAPIDockerFileRemote(c *check.C) {
  27. // E2E: Requires built httpserver.
  28. testRequires(c, NotUserNamespace, NotE2E)
  29. var testD string
  30. if testEnv.DaemonPlatform() == "windows" {
  31. testD = `FROM busybox
  32. RUN find / -name ba*
  33. RUN find /tmp/`
  34. } else {
  35. // -xdev is required because sysfs can cause EPERM
  36. testD = `FROM busybox
  37. RUN find / -xdev -name ba*
  38. RUN find /tmp/`
  39. }
  40. server := fakestorage.New(c, "", fakecontext.WithFiles(map[string]string{"testD": testD}))
  41. defer server.Close()
  42. res, body, err := request.Post("/build?dockerfile=baz&remote="+server.URL()+"/testD", request.JSON)
  43. c.Assert(err, checker.IsNil)
  44. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  45. buf, err := request.ReadBody(body)
  46. c.Assert(err, checker.IsNil)
  47. // Make sure Dockerfile exists.
  48. // Make sure 'baz' doesn't exist ANYWHERE despite being mentioned in the URL
  49. out := string(buf)
  50. c.Assert(out, checker.Contains, "RUN find /tmp")
  51. c.Assert(out, checker.Not(checker.Contains), "baz")
  52. }
  53. func (s *DockerSuite) TestBuildAPIRemoteTarballContext(c *check.C) {
  54. // E2E: Requires built httpserver.
  55. testRequires(c, NotE2E)
  56. buffer := new(bytes.Buffer)
  57. tw := tar.NewWriter(buffer)
  58. defer tw.Close()
  59. dockerfile := []byte("FROM busybox")
  60. err := tw.WriteHeader(&tar.Header{
  61. Name: "Dockerfile",
  62. Size: int64(len(dockerfile)),
  63. })
  64. // failed to write tar file header
  65. c.Assert(err, checker.IsNil)
  66. _, err = tw.Write(dockerfile)
  67. // failed to write tar file content
  68. c.Assert(err, checker.IsNil)
  69. // failed to close tar archive
  70. c.Assert(tw.Close(), checker.IsNil)
  71. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  72. "testT.tar": buffer,
  73. }))
  74. defer server.Close()
  75. res, b, err := request.Post("/build?remote="+server.URL()+"/testT.tar", request.ContentType("application/tar"))
  76. c.Assert(err, checker.IsNil)
  77. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  78. b.Close()
  79. }
  80. func (s *DockerSuite) TestBuildAPIRemoteTarballContextWithCustomDockerfile(c *check.C) {
  81. // E2E: Requires built httpserver.
  82. testRequires(c, NotE2E)
  83. buffer := new(bytes.Buffer)
  84. tw := tar.NewWriter(buffer)
  85. defer tw.Close()
  86. dockerfile := []byte(`FROM busybox
  87. RUN echo 'wrong'`)
  88. err := tw.WriteHeader(&tar.Header{
  89. Name: "Dockerfile",
  90. Size: int64(len(dockerfile)),
  91. })
  92. // failed to write tar file header
  93. c.Assert(err, checker.IsNil)
  94. _, err = tw.Write(dockerfile)
  95. // failed to write tar file content
  96. c.Assert(err, checker.IsNil)
  97. custom := []byte(`FROM busybox
  98. RUN echo 'right'
  99. `)
  100. err = tw.WriteHeader(&tar.Header{
  101. Name: "custom",
  102. Size: int64(len(custom)),
  103. })
  104. // failed to write tar file header
  105. c.Assert(err, checker.IsNil)
  106. _, err = tw.Write(custom)
  107. // failed to write tar file content
  108. c.Assert(err, checker.IsNil)
  109. // failed to close tar archive
  110. c.Assert(tw.Close(), checker.IsNil)
  111. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  112. "testT.tar": buffer,
  113. }))
  114. defer server.Close()
  115. url := "/build?dockerfile=custom&remote=" + server.URL() + "/testT.tar"
  116. res, body, err := request.Post(url, request.ContentType("application/tar"))
  117. c.Assert(err, checker.IsNil)
  118. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  119. defer body.Close()
  120. content, err := request.ReadBody(body)
  121. c.Assert(err, checker.IsNil)
  122. // Build used the wrong dockerfile.
  123. c.Assert(string(content), checker.Not(checker.Contains), "wrong")
  124. }
  125. func (s *DockerSuite) TestBuildAPILowerDockerfile(c *check.C) {
  126. // E2E: Requires built httpserver.
  127. testRequires(c, NotE2E)
  128. git := fakegit.New(c, "repo", map[string]string{
  129. "dockerfile": `FROM busybox
  130. RUN echo from dockerfile`,
  131. }, false)
  132. defer git.Close()
  133. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  134. c.Assert(err, checker.IsNil)
  135. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  136. buf, err := request.ReadBody(body)
  137. c.Assert(err, checker.IsNil)
  138. out := string(buf)
  139. c.Assert(out, checker.Contains, "from dockerfile")
  140. }
  141. func (s *DockerSuite) TestBuildAPIBuildGitWithF(c *check.C) {
  142. // E2E: Test requires go and contrib source.
  143. testRequires(c, NotE2E)
  144. git := fakegit.New(c, "repo", map[string]string{
  145. "baz": `FROM busybox
  146. RUN echo from baz`,
  147. "Dockerfile": `FROM busybox
  148. RUN echo from Dockerfile`,
  149. }, false)
  150. defer git.Close()
  151. // Make sure it tries to 'dockerfile' query param value
  152. res, body, err := request.Post("/build?dockerfile=baz&remote="+git.RepoURL, request.JSON)
  153. c.Assert(err, checker.IsNil)
  154. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  155. buf, err := request.ReadBody(body)
  156. c.Assert(err, checker.IsNil)
  157. out := string(buf)
  158. c.Assert(out, checker.Contains, "from baz")
  159. }
  160. func (s *DockerSuite) TestBuildAPIDoubleDockerfile(c *check.C) {
  161. // E2E: Requires built httpserver.
  162. testRequires(c, UnixCli, NotE2E) // dockerfile overwrites Dockerfile on Windows
  163. git := fakegit.New(c, "repo", map[string]string{
  164. "Dockerfile": `FROM busybox
  165. RUN echo from Dockerfile`,
  166. "dockerfile": `FROM busybox
  167. RUN echo from dockerfile`,
  168. }, false)
  169. defer git.Close()
  170. // Make sure it tries to 'dockerfile' query param value
  171. res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
  172. c.Assert(err, checker.IsNil)
  173. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  174. buf, err := request.ReadBody(body)
  175. c.Assert(err, checker.IsNil)
  176. out := string(buf)
  177. c.Assert(out, checker.Contains, "from Dockerfile")
  178. }
  179. func (s *DockerSuite) TestBuildAPIUnnormalizedTarPaths(c *check.C) {
  180. // Make sure that build context tars with entries of the form
  181. // x/./y don't cause caching false positives.
  182. buildFromTarContext := func(fileContents []byte) string {
  183. buffer := new(bytes.Buffer)
  184. tw := tar.NewWriter(buffer)
  185. defer tw.Close()
  186. dockerfile := []byte(`FROM busybox
  187. COPY dir /dir/`)
  188. err := tw.WriteHeader(&tar.Header{
  189. Name: "Dockerfile",
  190. Size: int64(len(dockerfile)),
  191. })
  192. //failed to write tar file header
  193. c.Assert(err, checker.IsNil)
  194. _, err = tw.Write(dockerfile)
  195. // failed to write Dockerfile in tar file content
  196. c.Assert(err, checker.IsNil)
  197. err = tw.WriteHeader(&tar.Header{
  198. Name: "dir/./file",
  199. Size: int64(len(fileContents)),
  200. })
  201. //failed to write tar file header
  202. c.Assert(err, checker.IsNil)
  203. _, err = tw.Write(fileContents)
  204. // failed to write file contents in tar file content
  205. c.Assert(err, checker.IsNil)
  206. // failed to close tar archive
  207. c.Assert(tw.Close(), checker.IsNil)
  208. res, body, err := request.Post("/build", request.RawContent(ioutil.NopCloser(buffer)), request.ContentType("application/x-tar"))
  209. c.Assert(err, checker.IsNil)
  210. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  211. out, err := request.ReadBody(body)
  212. c.Assert(err, checker.IsNil)
  213. lines := strings.Split(string(out), "\n")
  214. c.Assert(len(lines), checker.GreaterThan, 1)
  215. c.Assert(lines[len(lines)-2], checker.Matches, ".*Successfully built [0-9a-f]{12}.*")
  216. re := regexp.MustCompile("Successfully built ([0-9a-f]{12})")
  217. matches := re.FindStringSubmatch(lines[len(lines)-2])
  218. return matches[1]
  219. }
  220. imageA := buildFromTarContext([]byte("abc"))
  221. imageB := buildFromTarContext([]byte("def"))
  222. c.Assert(imageA, checker.Not(checker.Equals), imageB)
  223. }
  224. func (s *DockerSuite) TestBuildOnBuildWithCopy(c *check.C) {
  225. dockerfile := `
  226. FROM ` + minimalBaseImage() + ` as onbuildbase
  227. ONBUILD COPY file /file
  228. FROM onbuildbase
  229. `
  230. ctx := fakecontext.New(c, "",
  231. fakecontext.WithDockerfile(dockerfile),
  232. fakecontext.WithFile("file", "some content"),
  233. )
  234. defer ctx.Close()
  235. res, body, err := request.Post(
  236. "/build",
  237. request.RawContent(ctx.AsTarReader(c)),
  238. request.ContentType("application/x-tar"))
  239. c.Assert(err, checker.IsNil)
  240. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  241. out, err := request.ReadBody(body)
  242. c.Assert(err, checker.IsNil)
  243. c.Assert(string(out), checker.Contains, "Successfully built")
  244. }
  245. func (s *DockerSuite) TestBuildOnBuildCache(c *check.C) {
  246. build := func(dockerfile string) []byte {
  247. ctx := fakecontext.New(c, "",
  248. fakecontext.WithDockerfile(dockerfile),
  249. )
  250. defer ctx.Close()
  251. res, body, err := request.Post(
  252. "/build",
  253. request.RawContent(ctx.AsTarReader(c)),
  254. request.ContentType("application/x-tar"))
  255. require.NoError(c, err)
  256. assert.Equal(c, http.StatusOK, res.StatusCode)
  257. out, err := request.ReadBody(body)
  258. require.NoError(c, err)
  259. assert.Contains(c, string(out), "Successfully built")
  260. return out
  261. }
  262. dockerfile := `
  263. FROM ` + minimalBaseImage() + ` as onbuildbase
  264. ENV something=bar
  265. ONBUILD ENV foo=bar
  266. `
  267. build(dockerfile)
  268. dockerfile += "FROM onbuildbase"
  269. out := build(dockerfile)
  270. imageIDs := getImageIDsFromBuild(c, out)
  271. assert.Len(c, imageIDs, 2)
  272. parentID, childID := imageIDs[0], imageIDs[1]
  273. client, err := request.NewClient()
  274. require.NoError(c, err)
  275. // check parentID is correct
  276. image, _, err := client.ImageInspectWithRaw(context.Background(), childID)
  277. require.NoError(c, err)
  278. assert.Equal(c, parentID, image.Parent)
  279. }
  280. func (s *DockerRegistrySuite) TestBuildCopyFromForcePull(c *check.C) {
  281. client, err := request.NewClient()
  282. require.NoError(c, err)
  283. repoName := fmt.Sprintf("%v/dockercli/busybox", privateRegistryURL)
  284. // tag the image to upload it to the private registry
  285. err = client.ImageTag(context.TODO(), "busybox", repoName)
  286. assert.Nil(c, err)
  287. // push the image to the registry
  288. rc, err := client.ImagePush(context.TODO(), repoName, types.ImagePushOptions{RegistryAuth: "{}"})
  289. assert.Nil(c, err)
  290. _, err = io.Copy(ioutil.Discard, rc)
  291. assert.Nil(c, err)
  292. dockerfile := fmt.Sprintf(`
  293. FROM %s AS foo
  294. RUN touch abc
  295. FROM %s
  296. COPY --from=foo /abc /
  297. `, repoName, repoName)
  298. ctx := fakecontext.New(c, "",
  299. fakecontext.WithDockerfile(dockerfile),
  300. )
  301. defer ctx.Close()
  302. res, body, err := request.Post(
  303. "/build?pull=1",
  304. request.RawContent(ctx.AsTarReader(c)),
  305. request.ContentType("application/x-tar"))
  306. require.NoError(c, err)
  307. assert.Equal(c, http.StatusOK, res.StatusCode)
  308. out, err := request.ReadBody(body)
  309. require.NoError(c, err)
  310. assert.Contains(c, string(out), "Successfully built")
  311. }
  312. func (s *DockerSuite) TestBuildAddRemoteNoDecompress(c *check.C) {
  313. // E2E: Requires built httpserver.
  314. testRequires(c, NotE2E)
  315. buffer := new(bytes.Buffer)
  316. tw := tar.NewWriter(buffer)
  317. dt := []byte("contents")
  318. err := tw.WriteHeader(&tar.Header{
  319. Name: "foo",
  320. Size: int64(len(dt)),
  321. Mode: 0600,
  322. Typeflag: tar.TypeReg,
  323. })
  324. require.NoError(c, err)
  325. _, err = tw.Write(dt)
  326. require.NoError(c, err)
  327. err = tw.Close()
  328. require.NoError(c, err)
  329. server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
  330. "test.tar": buffer,
  331. }))
  332. defer server.Close()
  333. dockerfile := fmt.Sprintf(`
  334. FROM busybox
  335. ADD %s/test.tar /
  336. RUN [ -f test.tar ]
  337. `, server.URL())
  338. ctx := fakecontext.New(c, "",
  339. fakecontext.WithDockerfile(dockerfile),
  340. )
  341. defer ctx.Close()
  342. res, body, err := request.Post(
  343. "/build",
  344. request.RawContent(ctx.AsTarReader(c)),
  345. request.ContentType("application/x-tar"))
  346. require.NoError(c, err)
  347. assert.Equal(c, http.StatusOK, res.StatusCode)
  348. out, err := request.ReadBody(body)
  349. require.NoError(c, err)
  350. assert.Contains(c, string(out), "Successfully built")
  351. }
  352. func (s *DockerSuite) TestBuildChownOnCopy(c *check.C) {
  353. testRequires(c, DaemonIsLinux)
  354. dockerfile := `FROM busybox
  355. RUN echo 'test1:x:1001:1001::/bin:/bin/false' >> /etc/passwd
  356. RUN echo 'test1:x:1001:' >> /etc/group
  357. RUN echo 'test2:x:1002:' >> /etc/group
  358. COPY --chown=test1:1002 . /new_dir
  359. RUN ls -l /
  360. RUN [ $(ls -l / | grep new_dir | awk '{print $3":"$4}') = 'test1:test2' ]
  361. RUN [ $(ls -nl / | grep new_dir | awk '{print $3":"$4}') = '1001:1002' ]
  362. `
  363. ctx := fakecontext.New(c, "",
  364. fakecontext.WithDockerfile(dockerfile),
  365. fakecontext.WithFile("test_file1", "some test content"),
  366. )
  367. defer ctx.Close()
  368. res, body, err := request.Post(
  369. "/build",
  370. request.RawContent(ctx.AsTarReader(c)),
  371. request.ContentType("application/x-tar"))
  372. c.Assert(err, checker.IsNil)
  373. c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
  374. out, err := request.ReadBody(body)
  375. require.NoError(c, err)
  376. assert.Contains(c, string(out), "Successfully built")
  377. }
  378. func (s *DockerSuite) TestBuildWithSession(c *check.C) {
  379. testRequires(c, ExperimentalDaemon)
  380. dockerfile := `
  381. FROM busybox
  382. COPY file /
  383. RUN cat /file
  384. `
  385. fctx := fakecontext.New(c, "",
  386. fakecontext.WithFile("file", "some content"),
  387. )
  388. defer fctx.Close()
  389. out := testBuildWithSession(c, fctx.Dir, dockerfile)
  390. assert.Contains(c, out, "some content")
  391. fctx.Add("second", "contentcontent")
  392. dockerfile += `
  393. COPY second /
  394. RUN cat /second
  395. `
  396. out = testBuildWithSession(c, fctx.Dir, dockerfile)
  397. assert.Equal(c, strings.Count(out, "Using cache"), 2)
  398. assert.Contains(c, out, "contentcontent")
  399. client, err := request.NewClient()
  400. require.NoError(c, err)
  401. du, err := client.DiskUsage(context.TODO())
  402. assert.Nil(c, err)
  403. assert.True(c, du.BuilderSize > 10)
  404. out = testBuildWithSession(c, fctx.Dir, dockerfile)
  405. assert.Equal(c, strings.Count(out, "Using cache"), 4)
  406. du2, err := client.DiskUsage(context.TODO())
  407. assert.Nil(c, err)
  408. assert.Equal(c, du.BuilderSize, du2.BuilderSize)
  409. // rebuild with regular tar, confirm cache still applies
  410. fctx.Add("Dockerfile", dockerfile)
  411. res, body, err := request.Post(
  412. "/build",
  413. request.RawContent(fctx.AsTarReader(c)),
  414. request.ContentType("application/x-tar"))
  415. require.NoError(c, err)
  416. assert.Equal(c, http.StatusOK, res.StatusCode)
  417. outBytes, err := request.ReadBody(body)
  418. require.NoError(c, err)
  419. assert.Contains(c, string(outBytes), "Successfully built")
  420. assert.Equal(c, strings.Count(string(outBytes), "Using cache"), 4)
  421. _, err = client.BuildCachePrune(context.TODO())
  422. assert.Nil(c, err)
  423. du, err = client.DiskUsage(context.TODO())
  424. assert.Nil(c, err)
  425. assert.Equal(c, du.BuilderSize, int64(0))
  426. }
  427. func testBuildWithSession(c *check.C, dir, dockerfile string) (outStr string) {
  428. client, err := request.NewClient()
  429. require.NoError(c, err)
  430. sess, err := session.NewSession("foo1", "foo")
  431. assert.Nil(c, err)
  432. fsProvider := filesync.NewFSSyncProvider(dir, nil)
  433. sess.Allow(fsProvider)
  434. g, ctx := errgroup.WithContext(context.Background())
  435. g.Go(func() error {
  436. return sess.Run(ctx, client.DialSession)
  437. })
  438. g.Go(func() error {
  439. res, body, err := request.Post("/build?remote=client-session&session="+sess.UUID(), func(req *http.Request) error {
  440. req.Body = ioutil.NopCloser(strings.NewReader(dockerfile))
  441. return nil
  442. })
  443. if err != nil {
  444. return err
  445. }
  446. assert.Equal(c, res.StatusCode, http.StatusOK)
  447. out, err := request.ReadBody(body)
  448. require.NoError(c, err)
  449. assert.Contains(c, string(out), "Successfully built")
  450. sess.Close()
  451. outStr = string(out)
  452. return nil
  453. })
  454. err = g.Wait()
  455. assert.Nil(c, err)
  456. return
  457. }
  458. type buildLine struct {
  459. Stream string
  460. Aux struct {
  461. ID string
  462. }
  463. }
  464. func getImageIDsFromBuild(c *check.C, output []byte) []string {
  465. ids := []string{}
  466. for _, line := range bytes.Split(output, []byte("\n")) {
  467. if len(line) == 0 {
  468. continue
  469. }
  470. entry := buildLine{}
  471. require.NoError(c, json.Unmarshal(line, &entry))
  472. if entry.Aux.ID != "" {
  473. ids = append(ids, entry.Aux.ID)
  474. }
  475. }
  476. return ids
  477. }