123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666 |
- package main
- import (
- "archive/tar"
- "bytes"
- "encoding/json"
- "fmt"
- "io"
- "io/ioutil"
- "net/http"
- "regexp"
- "strings"
- "github.com/docker/docker/api/types"
- "github.com/docker/docker/integration-cli/checker"
- "github.com/docker/docker/integration-cli/cli/build/fakecontext"
- "github.com/docker/docker/integration-cli/cli/build/fakegit"
- "github.com/docker/docker/integration-cli/cli/build/fakestorage"
- "github.com/docker/docker/integration-cli/request"
- "github.com/go-check/check"
- "github.com/moby/buildkit/session"
- "github.com/moby/buildkit/session/filesync"
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
- "golang.org/x/net/context"
- "golang.org/x/sync/errgroup"
- )
- func (s *DockerSuite) TestBuildAPIDockerFileRemote(c *check.C) {
- testRequires(c, NotUserNamespace)
- var testD string
- if testEnv.DaemonPlatform() == "windows" {
- testD = `FROM busybox
- RUN find / -name ba*
- RUN find /tmp/`
- } else {
- // -xdev is required because sysfs can cause EPERM
- testD = `FROM busybox
- RUN find / -xdev -name ba*
- RUN find /tmp/`
- }
- server := fakestorage.New(c, "", fakecontext.WithFiles(map[string]string{"testD": testD}))
- defer server.Close()
- res, body, err := request.Post("/build?dockerfile=baz&remote="+server.URL()+"/testD", request.JSON)
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- buf, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- // Make sure Dockerfile exists.
- // Make sure 'baz' doesn't exist ANYWHERE despite being mentioned in the URL
- out := string(buf)
- c.Assert(out, checker.Contains, "RUN find /tmp")
- c.Assert(out, checker.Not(checker.Contains), "baz")
- }
- func (s *DockerSuite) TestBuildAPIRemoteTarballContext(c *check.C) {
- buffer := new(bytes.Buffer)
- tw := tar.NewWriter(buffer)
- defer tw.Close()
- dockerfile := []byte("FROM busybox")
- err := tw.WriteHeader(&tar.Header{
- Name: "Dockerfile",
- Size: int64(len(dockerfile)),
- })
- // failed to write tar file header
- c.Assert(err, checker.IsNil)
- _, err = tw.Write(dockerfile)
- // failed to write tar file content
- c.Assert(err, checker.IsNil)
- // failed to close tar archive
- c.Assert(tw.Close(), checker.IsNil)
- server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
- "testT.tar": buffer,
- }))
- defer server.Close()
- res, b, err := request.Post("/build?remote="+server.URL()+"/testT.tar", request.ContentType("application/tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- b.Close()
- }
- func (s *DockerSuite) TestBuildAPIRemoteTarballContextWithCustomDockerfile(c *check.C) {
- buffer := new(bytes.Buffer)
- tw := tar.NewWriter(buffer)
- defer tw.Close()
- dockerfile := []byte(`FROM busybox
- RUN echo 'wrong'`)
- err := tw.WriteHeader(&tar.Header{
- Name: "Dockerfile",
- Size: int64(len(dockerfile)),
- })
- // failed to write tar file header
- c.Assert(err, checker.IsNil)
- _, err = tw.Write(dockerfile)
- // failed to write tar file content
- c.Assert(err, checker.IsNil)
- custom := []byte(`FROM busybox
- RUN echo 'right'
- `)
- err = tw.WriteHeader(&tar.Header{
- Name: "custom",
- Size: int64(len(custom)),
- })
- // failed to write tar file header
- c.Assert(err, checker.IsNil)
- _, err = tw.Write(custom)
- // failed to write tar file content
- c.Assert(err, checker.IsNil)
- // failed to close tar archive
- c.Assert(tw.Close(), checker.IsNil)
- server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
- "testT.tar": buffer,
- }))
- defer server.Close()
- url := "/build?dockerfile=custom&remote=" + server.URL() + "/testT.tar"
- res, body, err := request.Post(url, request.ContentType("application/tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- defer body.Close()
- content, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- // Build used the wrong dockerfile.
- c.Assert(string(content), checker.Not(checker.Contains), "wrong")
- }
- func (s *DockerSuite) TestBuildAPILowerDockerfile(c *check.C) {
- git := fakegit.New(c, "repo", map[string]string{
- "dockerfile": `FROM busybox
- RUN echo from dockerfile`,
- }, false)
- defer git.Close()
- res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- buf, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- out := string(buf)
- c.Assert(out, checker.Contains, "from dockerfile")
- }
- func (s *DockerSuite) TestBuildAPIBuildGitWithF(c *check.C) {
- git := fakegit.New(c, "repo", map[string]string{
- "baz": `FROM busybox
- RUN echo from baz`,
- "Dockerfile": `FROM busybox
- RUN echo from Dockerfile`,
- }, false)
- defer git.Close()
- // Make sure it tries to 'dockerfile' query param value
- res, body, err := request.Post("/build?dockerfile=baz&remote="+git.RepoURL, request.JSON)
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- buf, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- out := string(buf)
- c.Assert(out, checker.Contains, "from baz")
- }
- func (s *DockerSuite) TestBuildAPIDoubleDockerfile(c *check.C) {
- testRequires(c, UnixCli) // dockerfile overwrites Dockerfile on Windows
- git := fakegit.New(c, "repo", map[string]string{
- "Dockerfile": `FROM busybox
- RUN echo from Dockerfile`,
- "dockerfile": `FROM busybox
- RUN echo from dockerfile`,
- }, false)
- defer git.Close()
- // Make sure it tries to 'dockerfile' query param value
- res, body, err := request.Post("/build?remote="+git.RepoURL, request.JSON)
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- buf, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- out := string(buf)
- c.Assert(out, checker.Contains, "from Dockerfile")
- }
- func (s *DockerSuite) TestBuildAPIUnnormalizedTarPaths(c *check.C) {
- // Make sure that build context tars with entries of the form
- // x/./y don't cause caching false positives.
- buildFromTarContext := func(fileContents []byte) string {
- buffer := new(bytes.Buffer)
- tw := tar.NewWriter(buffer)
- defer tw.Close()
- dockerfile := []byte(`FROM busybox
- COPY dir /dir/`)
- err := tw.WriteHeader(&tar.Header{
- Name: "Dockerfile",
- Size: int64(len(dockerfile)),
- })
- //failed to write tar file header
- c.Assert(err, checker.IsNil)
- _, err = tw.Write(dockerfile)
- // failed to write Dockerfile in tar file content
- c.Assert(err, checker.IsNil)
- err = tw.WriteHeader(&tar.Header{
- Name: "dir/./file",
- Size: int64(len(fileContents)),
- })
- //failed to write tar file header
- c.Assert(err, checker.IsNil)
- _, err = tw.Write(fileContents)
- // failed to write file contents in tar file content
- c.Assert(err, checker.IsNil)
- // failed to close tar archive
- c.Assert(tw.Close(), checker.IsNil)
- res, body, err := request.Post("/build", request.RawContent(ioutil.NopCloser(buffer)), request.ContentType("application/x-tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- out, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- lines := strings.Split(string(out), "\n")
- c.Assert(len(lines), checker.GreaterThan, 1)
- c.Assert(lines[len(lines)-2], checker.Matches, ".*Successfully built [0-9a-f]{12}.*")
- re := regexp.MustCompile("Successfully built ([0-9a-f]{12})")
- matches := re.FindStringSubmatch(lines[len(lines)-2])
- return matches[1]
- }
- imageA := buildFromTarContext([]byte("abc"))
- imageB := buildFromTarContext([]byte("def"))
- c.Assert(imageA, checker.Not(checker.Equals), imageB)
- }
- func (s *DockerSuite) TestBuildOnBuildWithCopy(c *check.C) {
- dockerfile := `
- FROM ` + minimalBaseImage() + ` as onbuildbase
- ONBUILD COPY file /file
- FROM onbuildbase
- `
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("file", "some content"),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- out, err := request.ReadBody(body)
- c.Assert(err, checker.IsNil)
- c.Assert(string(out), checker.Contains, "Successfully built")
- }
- func (s *DockerSuite) TestBuildOnBuildCache(c *check.C) {
- build := func(dockerfile string) []byte {
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- return out
- }
- dockerfile := `
- FROM ` + minimalBaseImage() + ` as onbuildbase
- ENV something=bar
- ONBUILD ENV foo=bar
- `
- build(dockerfile)
- dockerfile += "FROM onbuildbase"
- out := build(dockerfile)
- imageIDs := getImageIDsFromBuild(c, out)
- assert.Len(c, imageIDs, 2)
- parentID, childID := imageIDs[0], imageIDs[1]
- client, err := request.NewClient()
- require.NoError(c, err)
- // check parentID is correct
- image, _, err := client.ImageInspectWithRaw(context.Background(), childID)
- require.NoError(c, err)
- assert.Equal(c, parentID, image.Parent)
- }
- func (s *DockerRegistrySuite) TestBuildCopyFromForcePull(c *check.C) {
- client, err := request.NewClient()
- require.NoError(c, err)
- repoName := fmt.Sprintf("%v/dockercli/busybox", privateRegistryURL)
- // tag the image to upload it to the private registry
- err = client.ImageTag(context.TODO(), "busybox", repoName)
- assert.Nil(c, err)
- // push the image to the registry
- rc, err := client.ImagePush(context.TODO(), repoName, types.ImagePushOptions{RegistryAuth: "{}"})
- assert.Nil(c, err)
- _, err = io.Copy(ioutil.Discard, rc)
- assert.Nil(c, err)
- dockerfile := fmt.Sprintf(`
- FROM %s AS foo
- RUN touch abc
- FROM %s
- COPY --from=foo /abc /
- `, repoName, repoName)
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build?pull=1",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- }
- func (s *DockerSuite) TestBuildAddRemoteNoDecompress(c *check.C) {
- buffer := new(bytes.Buffer)
- tw := tar.NewWriter(buffer)
- dt := []byte("contents")
- err := tw.WriteHeader(&tar.Header{
- Name: "foo",
- Size: int64(len(dt)),
- Mode: 0600,
- Typeflag: tar.TypeReg,
- })
- require.NoError(c, err)
- _, err = tw.Write(dt)
- require.NoError(c, err)
- err = tw.Close()
- require.NoError(c, err)
- server := fakestorage.New(c, "", fakecontext.WithBinaryFiles(map[string]*bytes.Buffer{
- "test.tar": buffer,
- }))
- defer server.Close()
- dockerfile := fmt.Sprintf(`
- FROM busybox
- ADD %s/test.tar /
- RUN [ -f test.tar ]
- `, server.URL())
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- }
- func (s *DockerSuite) TestBuildChownOnCopy(c *check.C) {
- testRequires(c, DaemonIsLinux)
- dockerfile := `FROM busybox
- RUN echo 'test1:x:1001:1001::/bin:/bin/false' >> /etc/passwd
- RUN echo 'test1:x:1001:' >> /etc/group
- RUN echo 'test2:x:1002:' >> /etc/group
- COPY --chown=test1:1002 . /new_dir
- RUN ls -l /
- RUN [ $(ls -l / | grep new_dir | awk '{print $3":"$4}') = 'test1:test2' ]
- RUN [ $(ls -nl / | grep new_dir | awk '{print $3":"$4}') = '1001:1002' ]
- `
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("test_file1", "some test content"),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- }
- func (s *DockerSuite) TestBuildCopyCacheOnFileChange(c *check.C) {
- dockerfile := `FROM busybox
- COPY file /file`
- ctx1 := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("file", "foo"))
- ctx2 := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("file", "bar"))
- var build = func(ctx *fakecontext.Fake) string {
- res, body, err := request.Post("/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- ids := getImageIDsFromBuild(c, out)
- return ids[len(ids)-1]
- }
- id1 := build(ctx1)
- id2 := build(ctx1)
- id3 := build(ctx2)
- if id1 != id2 {
- c.Fatal("didn't use the cache")
- }
- if id1 == id3 {
- c.Fatal("COPY With different source file should not share same cache")
- }
- }
- func (s *DockerSuite) TestBuildAddCacheOnFileChange(c *check.C) {
- dockerfile := `FROM busybox
- ADD file /file`
- ctx1 := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("file", "foo"))
- ctx2 := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- fakecontext.WithFile("file", "bar"))
- var build = func(ctx *fakecontext.Fake) string {
- res, body, err := request.Post("/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- ids := getImageIDsFromBuild(c, out)
- return ids[len(ids)-1]
- }
- id1 := build(ctx1)
- id2 := build(ctx1)
- id3 := build(ctx2)
- if id1 != id2 {
- c.Fatal("didn't use the cache")
- }
- if id1 == id3 {
- c.Fatal("COPY With different source file should not share same cache")
- }
- }
- func (s *DockerSuite) TestBuildWithSession(c *check.C) {
- testRequires(c, ExperimentalDaemon)
- dockerfile := `
- FROM busybox
- COPY file /
- RUN cat /file
- `
- fctx := fakecontext.New(c, "",
- fakecontext.WithFile("file", "some content"),
- )
- defer fctx.Close()
- out := testBuildWithSession(c, fctx.Dir, dockerfile)
- assert.Contains(c, out, "some content")
- fctx.Add("second", "contentcontent")
- dockerfile += `
- COPY second /
- RUN cat /second
- `
- out = testBuildWithSession(c, fctx.Dir, dockerfile)
- assert.Equal(c, strings.Count(out, "Using cache"), 2)
- assert.Contains(c, out, "contentcontent")
- client, err := request.NewClient()
- require.NoError(c, err)
- du, err := client.DiskUsage(context.TODO())
- assert.Nil(c, err)
- assert.True(c, du.BuilderSize > 10)
- out = testBuildWithSession(c, fctx.Dir, dockerfile)
- assert.Equal(c, strings.Count(out, "Using cache"), 4)
- du2, err := client.DiskUsage(context.TODO())
- assert.Nil(c, err)
- assert.Equal(c, du.BuilderSize, du2.BuilderSize)
- // rebuild with regular tar, confirm cache still applies
- fctx.Add("Dockerfile", dockerfile)
- res, body, err := request.Post(
- "/build",
- request.RawContent(fctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- require.NoError(c, err)
- assert.Equal(c, http.StatusOK, res.StatusCode)
- outBytes, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(outBytes), "Successfully built")
- assert.Equal(c, strings.Count(string(outBytes), "Using cache"), 4)
- _, err = client.BuildCachePrune(context.TODO())
- assert.Nil(c, err)
- du, err = client.DiskUsage(context.TODO())
- assert.Nil(c, err)
- assert.Equal(c, du.BuilderSize, int64(0))
- }
- func testBuildWithSession(c *check.C, dir, dockerfile string) (outStr string) {
- client, err := request.NewClient()
- require.NoError(c, err)
- sess, err := session.NewSession("foo1", "foo")
- assert.Nil(c, err)
- fsProvider := filesync.NewFSSyncProvider([]filesync.SyncedDir{
- {Dir: dir},
- })
- sess.Allow(fsProvider)
- g, ctx := errgroup.WithContext(context.Background())
- g.Go(func() error {
- return sess.Run(ctx, client.DialSession)
- })
- g.Go(func() error {
- res, body, err := request.Post("/build?remote=client-session&session="+sess.ID(), func(req *http.Request) error {
- req.Body = ioutil.NopCloser(strings.NewReader(dockerfile))
- return nil
- })
- if err != nil {
- return err
- }
- assert.Equal(c, res.StatusCode, http.StatusOK)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- sess.Close()
- outStr = string(out)
- return nil
- })
- err = g.Wait()
- assert.Nil(c, err)
- return
- }
- func (s *DockerSuite) TestBuildScratchCopy(c *check.C) {
- testRequires(c, DaemonIsLinux)
- dockerfile := `FROM scratch
- ADD Dockerfile /
- ENV foo bar`
- ctx := fakecontext.New(c, "",
- fakecontext.WithDockerfile(dockerfile),
- )
- defer ctx.Close()
- res, body, err := request.Post(
- "/build",
- request.RawContent(ctx.AsTarReader(c)),
- request.ContentType("application/x-tar"))
- c.Assert(err, checker.IsNil)
- c.Assert(res.StatusCode, checker.Equals, http.StatusOK)
- out, err := request.ReadBody(body)
- require.NoError(c, err)
- assert.Contains(c, string(out), "Successfully built")
- }
- type buildLine struct {
- Stream string
- Aux struct {
- ID string
- }
- }
- func getImageIDsFromBuild(c *check.C, output []byte) []string {
- ids := []string{}
- for _, line := range bytes.Split(output, []byte("\n")) {
- if len(line) == 0 {
- continue
- }
- entry := buildLine{}
- require.NoError(c, json.Unmarshal(line, &entry))
- if entry.Aux.ID != "" {
- ids = append(ids, entry.Aux.ID)
- }
- }
- return ids
- }
|