2017-06-21 15:13:44 -04:00
|
|
|
package image
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2017-09-08 13:45:19 -04:00
|
|
|
"runtime"
|
2017-06-21 15:13:44 -04:00
|
|
|
"sort"
|
2017-09-08 13:45:19 -04:00
|
|
|
"syscall"
|
2017-06-21 15:13:44 -04:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/docker/cli/cli/command"
|
2017-08-21 16:30:09 -04:00
|
|
|
"github.com/docker/cli/internal/test"
|
2017-06-21 15:13:44 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
|
|
|
"github.com/docker/docker/pkg/archive"
|
2017-09-08 13:45:19 -04:00
|
|
|
"github.com/gotestyourself/gotestyourself/fs"
|
|
|
|
"github.com/gotestyourself/gotestyourself/skip"
|
2017-06-21 15:13:44 -04:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"golang.org/x/net/context"
|
|
|
|
)
|
|
|
|
|
2017-09-08 13:45:19 -04:00
|
|
|
func TestRunBuildResetsUidAndGidInContext(t *testing.T) {
|
|
|
|
skip.IfCondition(t, runtime.GOOS == "windows", "uid and gid not relevant on windows")
|
|
|
|
dest := fs.NewDir(t, "test-build-context-dest")
|
|
|
|
defer dest.Remove()
|
|
|
|
|
|
|
|
fakeImageBuild := func(_ context.Context, context io.Reader, options types.ImageBuildOptions) (types.ImageBuildResponse, error) {
|
|
|
|
assert.NoError(t, archive.Untar(context, dest.Path(), nil))
|
|
|
|
|
|
|
|
body := new(bytes.Buffer)
|
|
|
|
return types.ImageBuildResponse{Body: ioutil.NopCloser(body)}, nil
|
|
|
|
}
|
|
|
|
cli := test.NewFakeCli(&fakeClient{imageBuildFunc: fakeImageBuild})
|
|
|
|
|
|
|
|
dir := fs.NewDir(t, "test-build-context",
|
|
|
|
fs.WithFile("foo", "some content", fs.AsUser(65534, 65534)),
|
|
|
|
fs.WithFile("Dockerfile", `
|
|
|
|
FROM alpine:3.6
|
|
|
|
COPY foo bar /
|
|
|
|
`),
|
|
|
|
)
|
|
|
|
defer dir.Remove()
|
|
|
|
|
|
|
|
options := newBuildOptions()
|
|
|
|
options.context = dir.Path()
|
|
|
|
|
|
|
|
err := runBuild(cli, options)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
files, err := ioutil.ReadDir(dest.Path())
|
|
|
|
require.NoError(t, err)
|
|
|
|
for _, fileInfo := range files {
|
|
|
|
assert.Equal(t, uint32(0), fileInfo.Sys().(*syscall.Stat_t).Uid)
|
|
|
|
assert.Equal(t, uint32(0), fileInfo.Sys().(*syscall.Stat_t).Gid)
|
|
|
|
}
|
|
|
|
}
|
2017-06-21 15:13:44 -04:00
|
|
|
func TestRunBuildDockerfileFromStdinWithCompress(t *testing.T) {
|
|
|
|
dest, err := ioutil.TempDir("", "test-build-compress-dest")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.RemoveAll(dest)
|
|
|
|
|
|
|
|
var dockerfileName string
|
|
|
|
fakeImageBuild := func(_ context.Context, context io.Reader, options types.ImageBuildOptions) (types.ImageBuildResponse, error) {
|
|
|
|
buffer := new(bytes.Buffer)
|
|
|
|
tee := io.TeeReader(context, buffer)
|
|
|
|
|
|
|
|
assert.NoError(t, archive.Untar(tee, dest, nil))
|
|
|
|
dockerfileName = options.Dockerfile
|
|
|
|
|
|
|
|
header := buffer.Bytes()[:10]
|
|
|
|
assert.Equal(t, archive.Gzip, archive.DetectCompression(header))
|
|
|
|
|
|
|
|
body := new(bytes.Buffer)
|
|
|
|
return types.ImageBuildResponse{Body: ioutil.NopCloser(body)}, nil
|
|
|
|
}
|
|
|
|
|
2017-07-05 14:43:39 -04:00
|
|
|
cli := test.NewFakeCli(&fakeClient{imageBuildFunc: fakeImageBuild})
|
2017-06-21 15:13:44 -04:00
|
|
|
dockerfile := bytes.NewBufferString(`
|
|
|
|
FROM alpine:3.6
|
|
|
|
COPY foo /
|
|
|
|
`)
|
|
|
|
cli.SetIn(command.NewInStream(ioutil.NopCloser(dockerfile)))
|
|
|
|
|
|
|
|
dir, err := ioutil.TempDir("", "test-build-compress")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
|
|
|
|
ioutil.WriteFile(filepath.Join(dir, "foo"), []byte("some content"), 0644)
|
|
|
|
|
|
|
|
options := newBuildOptions()
|
|
|
|
options.compress = true
|
|
|
|
options.dockerfileName = "-"
|
|
|
|
options.context = dir
|
|
|
|
|
|
|
|
err = runBuild(cli, options)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
files, err := ioutil.ReadDir(dest)
|
|
|
|
require.NoError(t, err)
|
|
|
|
actual := []string{}
|
|
|
|
for _, fileInfo := range files {
|
|
|
|
actual = append(actual, fileInfo.Name())
|
|
|
|
}
|
|
|
|
sort.Strings(actual)
|
|
|
|
assert.Equal(t, []string{dockerfileName, ".dockerignore", "foo"}, actual)
|
|
|
|
}
|
2017-07-27 05:30:18 -04:00
|
|
|
|
|
|
|
// TestRunBuildFromLocalGitHubDirNonExistingRepo tests that build contexts
|
|
|
|
// starting with `github.com/` are special-cased, and the build command attempts
|
|
|
|
// to clone the remote repo.
|
|
|
|
func TestRunBuildFromGitHubSpecialCase(t *testing.T) {
|
2017-08-16 12:19:53 -04:00
|
|
|
cmd := NewBuildCommand(test.NewFakeCli(nil))
|
2017-07-27 05:30:18 -04:00
|
|
|
cmd.SetArgs([]string{"github.com/docker/no-such-repository"})
|
|
|
|
cmd.SetOutput(ioutil.Discard)
|
|
|
|
err := cmd.Execute()
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Contains(t, err.Error(), "unable to prepare context: unable to 'git clone'")
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestRunBuildFromLocalGitHubDirNonExistingRepo tests that a local directory
|
|
|
|
// starting with `github.com` takes precedence over the `github.com` special
|
|
|
|
// case.
|
|
|
|
func TestRunBuildFromLocalGitHubDir(t *testing.T) {
|
|
|
|
tmpDir, err := ioutil.TempDir("", "docker-build-from-local-dir-")
|
|
|
|
require.NoError(t, err)
|
|
|
|
defer os.RemoveAll(tmpDir)
|
|
|
|
|
|
|
|
buildDir := filepath.Join(tmpDir, "github.com", "docker", "no-such-repository")
|
|
|
|
err = os.MkdirAll(buildDir, 0777)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = ioutil.WriteFile(filepath.Join(buildDir, "Dockerfile"), []byte("FROM busybox\n"), 0644)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
client := test.NewFakeCli(&fakeClient{})
|
|
|
|
cmd := NewBuildCommand(client)
|
|
|
|
cmd.SetArgs([]string{buildDir})
|
|
|
|
cmd.SetOutput(ioutil.Discard)
|
|
|
|
err = cmd.Execute()
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|