mirror of
https://github.com/moby/moby.git
synced 2026-01-11 10:41:43 +00:00
365 lines
12 KiB
Go
365 lines
12 KiB
Go
package container
|
|
|
|
import (
|
|
"archive/tar"
|
|
"bytes"
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
|
|
cerrdefs "github.com/containerd/errdefs"
|
|
"github.com/moby/go-archive"
|
|
"github.com/moby/moby/api/types/build"
|
|
"github.com/moby/moby/api/types/jsonstream"
|
|
"github.com/moby/moby/client"
|
|
"github.com/moby/moby/client/pkg/jsonmessage"
|
|
"github.com/moby/moby/v2/integration/internal/container"
|
|
"github.com/moby/moby/v2/internal/testutil/fakecontext"
|
|
"gotest.tools/v3/assert"
|
|
is "gotest.tools/v3/assert/cmp"
|
|
"gotest.tools/v3/skip"
|
|
)
|
|
|
|
func TestCopyFromContainerPathDoesNotExist(t *testing.T) {
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
cid := container.Create(ctx, t, apiClient)
|
|
|
|
_, err := apiClient.CopyFromContainer(ctx, cid, client.CopyFromContainerOptions{SourcePath: "/dne"})
|
|
assert.Check(t, is.ErrorType(err, cerrdefs.IsNotFound))
|
|
assert.Check(t, is.ErrorContains(err, "Could not find the file /dne in container "+cid))
|
|
}
|
|
|
|
// TestCopyFromContainerPathIsNotDir tests that an error is returned when
|
|
// trying to create a directory on a path that's a file.
|
|
func TestCopyFromContainerPathIsNotDir(t *testing.T) {
|
|
skip.If(t, testEnv.UsingSnapshotter(), "FIXME: https://github.com/moby/moby/issues/47107")
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
cid := container.Create(ctx, t, apiClient)
|
|
|
|
// Pick a path that already exists as a file; on Linux "/etc/passwd"
|
|
// is expected to be there, so we pick that for convenience.
|
|
existingFile := "/etc/passwd/"
|
|
expected := []string{"not a directory"}
|
|
if testEnv.DaemonInfo.OSType == "windows" {
|
|
existingFile = "c:/windows/system32/drivers/etc/hosts/"
|
|
|
|
// Depending on the version of Windows, this produces a "ERROR_INVALID_NAME" (Windows < 2025),
|
|
// or a "ERROR_DIRECTORY" (Windows 2025); https://learn.microsoft.com/en-us/windows/win32/debug/system-error-codes--0-499-
|
|
expected = []string{
|
|
"The directory name is invalid.", // ERROR_DIRECTORY
|
|
"The filename, directory name, or volume label syntax is incorrect.", // ERROR_INVALID_NAME
|
|
}
|
|
}
|
|
_, err := apiClient.CopyFromContainer(ctx, cid, client.CopyFromContainerOptions{SourcePath: existingFile})
|
|
var found bool
|
|
for _, expErr := range expected {
|
|
if err != nil && strings.Contains(err.Error(), expErr) {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
assert.Check(t, found, "Expected error to be one of %v, but got %v", expected, err)
|
|
}
|
|
|
|
func TestCopyToContainerPathDoesNotExist(t *testing.T) {
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
cid := container.Create(ctx, t, apiClient)
|
|
|
|
_, err := apiClient.CopyToContainer(ctx, cid, client.CopyToContainerOptions{DestinationPath: "/dne", Content: bytes.NewReader([]byte(""))})
|
|
assert.Check(t, is.ErrorType(err, cerrdefs.IsNotFound))
|
|
assert.Check(t, is.ErrorContains(err, "Could not find the file /dne in container "+cid))
|
|
}
|
|
|
|
func TestCopyEmptyFile(t *testing.T) {
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
cid := container.Create(ctx, t, apiClient)
|
|
|
|
// empty content
|
|
dstDir, _ := makeEmptyArchive(t)
|
|
_, err := apiClient.CopyToContainer(ctx, cid, client.CopyToContainerOptions{DestinationPath: dstDir, Content: bytes.NewReader([]byte(""))})
|
|
assert.NilError(t, err)
|
|
|
|
// tar with empty file
|
|
dstDir, preparedArchive := makeEmptyArchive(t)
|
|
_, err = apiClient.CopyToContainer(ctx, cid, client.CopyToContainerOptions{DestinationPath: dstDir, Content: preparedArchive})
|
|
assert.NilError(t, err)
|
|
|
|
// tar with empty file archive mode
|
|
dstDir, preparedArchive = makeEmptyArchive(t)
|
|
_, err = apiClient.CopyToContainer(ctx, cid, client.CopyToContainerOptions{DestinationPath: dstDir, Content: preparedArchive, CopyUIDGID: true})
|
|
assert.NilError(t, err)
|
|
|
|
// copy from empty file
|
|
res, err := apiClient.CopyFromContainer(ctx, cid, client.CopyFromContainerOptions{SourcePath: dstDir})
|
|
rdr := res.Content
|
|
assert.NilError(t, err)
|
|
defer rdr.Close()
|
|
}
|
|
|
|
func TestCopyToContainerCopyUIDGID(t *testing.T) {
|
|
skip.If(t, testEnv.DaemonInfo.OSType == "windows")
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
imageID := makeTestImage(ctx, t)
|
|
|
|
tests := []struct {
|
|
doc string
|
|
user string
|
|
expected string
|
|
}{
|
|
{
|
|
doc: "image default",
|
|
expected: "2375:2376",
|
|
},
|
|
{
|
|
// Align with behavior of docker run, which treats a UID with
|
|
// empty groupname as default (0 (root)).
|
|
//
|
|
// docker run --rm --user "7777:" alpine id
|
|
// uid=7777 gid=0(root) groups=0(root)
|
|
doc: "trailing colon",
|
|
user: "7777:",
|
|
expected: "7777:0",
|
|
},
|
|
{
|
|
// Align with behavior of docker run, which treats a GID with
|
|
// empty username as default (0 (root)).
|
|
//
|
|
// docker run --rm --user ":7777" alpine id
|
|
// uid=0(root) gid=7777 groups=7777
|
|
doc: "leading colon",
|
|
user: ":7777",
|
|
expected: "0:7777",
|
|
},
|
|
{
|
|
doc: "known UID",
|
|
user: "2375",
|
|
expected: "2375:2376",
|
|
},
|
|
{
|
|
doc: "unknown UID",
|
|
user: "7777",
|
|
expected: "7777:0",
|
|
},
|
|
{
|
|
doc: "UID and GID",
|
|
user: "2375:2376",
|
|
expected: "2375:2376",
|
|
},
|
|
{
|
|
doc: "username and groupname",
|
|
user: "testuser:testgroup",
|
|
expected: "2375:2376",
|
|
},
|
|
{
|
|
doc: "username",
|
|
user: "testuser",
|
|
expected: "2375:2376",
|
|
},
|
|
{
|
|
doc: "username and GID",
|
|
user: "testuser:7777",
|
|
expected: "2375:7777",
|
|
},
|
|
{
|
|
doc: "UID and groupname",
|
|
user: "7777:testgroup",
|
|
expected: "7777:2376",
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.doc, func(t *testing.T) {
|
|
cID := container.Run(ctx, t, apiClient, container.WithImage(imageID), container.WithUser(tc.user))
|
|
defer container.Remove(ctx, t, apiClient, cID, client.ContainerRemoveOptions{Force: true})
|
|
|
|
// tar with empty file
|
|
dstDir, preparedArchive := makeEmptyArchive(t)
|
|
_, err := apiClient.CopyToContainer(ctx, cID, client.CopyToContainerOptions{DestinationPath: dstDir, Content: preparedArchive, CopyUIDGID: true})
|
|
assert.NilError(t, err)
|
|
|
|
res, err := container.Exec(ctx, apiClient, cID, []string{"stat", "-c", "%u:%g", "/empty-file.txt"})
|
|
assert.NilError(t, err)
|
|
assert.Equal(t, res.ExitCode, 0)
|
|
assert.Equal(t, strings.TrimSpace(res.Stdout()), tc.expected)
|
|
})
|
|
}
|
|
}
|
|
|
|
func makeTestImage(ctx context.Context, t *testing.T) (imageID string) {
|
|
t.Helper()
|
|
apiClient := testEnv.APIClient()
|
|
tmpDir := t.TempDir()
|
|
buildCtx := fakecontext.New(t, tmpDir, fakecontext.WithDockerfile(`
|
|
FROM busybox
|
|
RUN addgroup -g 2376 testgroup && adduser -D -u 2375 -G testgroup testuser
|
|
USER testuser:testgroup
|
|
`))
|
|
defer buildCtx.Close()
|
|
|
|
resp, err := apiClient.ImageBuild(ctx, buildCtx.AsTarReader(t), client.ImageBuildOptions{})
|
|
assert.NilError(t, err)
|
|
defer resp.Body.Close()
|
|
|
|
err = jsonmessage.DisplayJSONMessagesStream(resp.Body, io.Discard, 0, false, func(msg jsonstream.Message) {
|
|
var r build.Result
|
|
assert.NilError(t, json.Unmarshal(*msg.Aux, &r))
|
|
imageID = r.ID
|
|
})
|
|
assert.NilError(t, err)
|
|
assert.Assert(t, imageID != "")
|
|
return imageID
|
|
}
|
|
|
|
func makeEmptyArchive(t *testing.T) (string, io.ReadCloser) {
|
|
tmpDir := t.TempDir()
|
|
srcPath := filepath.Join(tmpDir, "empty-file.txt")
|
|
err := os.WriteFile(srcPath, []byte(""), 0o400)
|
|
assert.NilError(t, err)
|
|
|
|
// TODO(thaJeztah) Add utilities to the client to make steps below less complicated.
|
|
// Code below is taken from copyToContainer() in docker/cli.
|
|
srcInfo, err := archive.CopyInfoSourcePath(srcPath, false)
|
|
assert.NilError(t, err)
|
|
|
|
srcArchive, err := archive.TarResource(srcInfo)
|
|
assert.NilError(t, err)
|
|
t.Cleanup(func() {
|
|
srcArchive.Close()
|
|
})
|
|
|
|
ctrPath := "/empty-file.txt"
|
|
dstInfo := archive.CopyInfo{Path: ctrPath}
|
|
dstDir, preparedArchive, err := archive.PrepareArchiveCopy(srcArchive, srcInfo, dstInfo)
|
|
assert.NilError(t, err)
|
|
t.Cleanup(func() {
|
|
preparedArchive.Close()
|
|
})
|
|
return dstDir, preparedArchive
|
|
}
|
|
|
|
func TestCopyToContainerPathIsNotDir(t *testing.T) {
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
cid := container.Create(ctx, t, apiClient)
|
|
|
|
path := "/etc/passwd/"
|
|
if testEnv.DaemonInfo.OSType == "windows" {
|
|
path = "c:/windows/system32/drivers/etc/hosts/"
|
|
}
|
|
_, err := apiClient.CopyToContainer(ctx, cid, client.CopyToContainerOptions{DestinationPath: path})
|
|
assert.Check(t, is.ErrorContains(err, "not a directory"))
|
|
}
|
|
|
|
func TestCopyFromContainer(t *testing.T) {
|
|
skip.If(t, testEnv.DaemonInfo.OSType == "windows")
|
|
ctx := setupTest(t)
|
|
|
|
apiClient := testEnv.APIClient()
|
|
|
|
dir, err := os.MkdirTemp("", t.Name())
|
|
assert.NilError(t, err)
|
|
defer os.RemoveAll(dir)
|
|
|
|
buildCtx := fakecontext.New(t, dir, fakecontext.WithFile("foo", "hello"), fakecontext.WithFile("baz", "world"), fakecontext.WithDockerfile(`
|
|
FROM busybox
|
|
COPY foo /foo
|
|
COPY baz /bar/quux/baz
|
|
RUN ln -s notexist /bar/notarget && ln -s quux/baz /bar/filesymlink && ln -s quux /bar/dirsymlink && ln -s / /bar/root
|
|
CMD /fake
|
|
`))
|
|
defer buildCtx.Close()
|
|
|
|
resp, err := apiClient.ImageBuild(ctx, buildCtx.AsTarReader(t), client.ImageBuildOptions{})
|
|
assert.NilError(t, err)
|
|
defer resp.Body.Close()
|
|
|
|
var imageID string
|
|
err = jsonmessage.DisplayJSONMessagesStream(resp.Body, io.Discard, 0, false, func(msg jsonstream.Message) {
|
|
var r build.Result
|
|
assert.NilError(t, json.Unmarshal(*msg.Aux, &r))
|
|
imageID = r.ID
|
|
})
|
|
assert.NilError(t, err)
|
|
assert.Assert(t, imageID != "")
|
|
|
|
cid := container.Create(ctx, t, apiClient, container.WithImage(imageID))
|
|
|
|
for _, x := range []struct {
|
|
src string
|
|
expect map[string]string
|
|
}{
|
|
{"/", map[string]string{"/": "", "/foo": "hello", "/bar/quux/baz": "world", "/bar/filesymlink": "", "/bar/dirsymlink": "", "/bar/notarget": ""}},
|
|
{".", map[string]string{"./": "", "./foo": "hello", "./bar/quux/baz": "world", "./bar/filesymlink": "", "./bar/dirsymlink": "", "./bar/notarget": ""}},
|
|
{"/.", map[string]string{"./": "", "./foo": "hello", "./bar/quux/baz": "world", "./bar/filesymlink": "", "./bar/dirsymlink": "", "./bar/notarget": ""}},
|
|
{"./", map[string]string{"./": "", "./foo": "hello", "./bar/quux/baz": "world", "./bar/filesymlink": "", "./bar/dirsymlink": "", "./bar/notarget": ""}},
|
|
{"/./", map[string]string{"./": "", "./foo": "hello", "./bar/quux/baz": "world", "./bar/filesymlink": "", "./bar/dirsymlink": "", "./bar/notarget": ""}},
|
|
{"/bar/root", map[string]string{"root": ""}},
|
|
{"/bar/root/", map[string]string{"root/": "", "root/foo": "hello", "root/bar/quux/baz": "world", "root/bar/filesymlink": "", "root/bar/dirsymlink": "", "root/bar/notarget": ""}},
|
|
{"/bar/root/.", map[string]string{"./": "", "./foo": "hello", "./bar/quux/baz": "world", "./bar/filesymlink": "", "./bar/dirsymlink": "", "./bar/notarget": ""}},
|
|
|
|
{"bar/quux", map[string]string{"quux/": "", "quux/baz": "world"}},
|
|
{"bar/quux/", map[string]string{"quux/": "", "quux/baz": "world"}},
|
|
{"bar/quux/.", map[string]string{"./": "", "./baz": "world"}},
|
|
{"bar/quux/baz", map[string]string{"baz": "world"}},
|
|
|
|
{"bar/filesymlink", map[string]string{"filesymlink": ""}},
|
|
{"bar/dirsymlink", map[string]string{"dirsymlink": ""}},
|
|
{"bar/dirsymlink/", map[string]string{"dirsymlink/": "", "dirsymlink/baz": "world"}},
|
|
{"bar/dirsymlink/.", map[string]string{"./": "", "./baz": "world"}},
|
|
{"bar/notarget", map[string]string{"notarget": ""}},
|
|
} {
|
|
t.Run(x.src, func(t *testing.T) {
|
|
res, err := apiClient.CopyFromContainer(ctx, cid, client.CopyFromContainerOptions{SourcePath: x.src})
|
|
rdr := res.Content
|
|
assert.NilError(t, err)
|
|
defer rdr.Close()
|
|
|
|
found := make(map[string]bool, len(x.expect))
|
|
var numFound int
|
|
tr := tar.NewReader(rdr)
|
|
for numFound < len(x.expect) {
|
|
h, err := tr.Next()
|
|
if errors.Is(err, io.EOF) {
|
|
break
|
|
}
|
|
assert.NilError(t, err)
|
|
|
|
expected, exists := x.expect[h.Name]
|
|
if !exists {
|
|
// this archive will have extra stuff in it since we are copying from root
|
|
// and docker adds a bunch of stuff
|
|
continue
|
|
}
|
|
|
|
numFound++
|
|
found[h.Name] = true
|
|
|
|
buf, err := io.ReadAll(tr)
|
|
if err == nil {
|
|
assert.Check(t, is.Equal(string(buf), expected))
|
|
}
|
|
}
|
|
|
|
for f := range x.expect {
|
|
assert.Check(t, found[f], f+" not found in archive")
|
|
}
|
|
})
|
|
}
|
|
}
|