docker_api_build_test.go 14 KB

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