moby/integration/container/overlayfs_linux_test.go
Brian Goff e8dc902781 Wire up tests to support otel tracing
Integration tests will now configure clients to propagate traces as well
as create spans for all tests.

Some extra changes were needed (or desired for trace propagation) in the
test helpers to pass through tracing spans via context.

Signed-off-by: Brian Goff <cpuguy83@gmail.com>
2023-09-07 18:38:22 +00:00

109 lines
3 KiB
Go

package container
import (
"io"
"strings"
"testing"
"github.com/docker/docker/api/types"
"github.com/docker/docker/integration/internal/container"
"github.com/docker/docker/pkg/archive"
"github.com/docker/docker/pkg/dmesg"
"gotest.tools/v3/assert"
"gotest.tools/v3/skip"
)
func TestNoOverlayfsWarningsAboutUndefinedBehaviors(t *testing.T) {
skip.If(t, testEnv.DaemonInfo.OSType != "linux", "overlayfs is only available on linux")
skip.If(t, testEnv.IsRemoteDaemon(), "local daemon is needed for kernel log access")
skip.If(t, testEnv.IsRootless(), "root is needed for reading kernel log")
ctx := setupTest(t)
client := testEnv.APIClient()
cID := container.Run(ctx, t, client, container.WithCmd("sh", "-c", `while true; do echo $RANDOM >>/file; sleep 0.1; done`))
testCases := []struct {
name string
operation func(t *testing.T) error
}{
{name: "diff", operation: func(*testing.T) error {
_, err := client.ContainerDiff(ctx, cID)
return err
}},
{name: "export", operation: func(*testing.T) error {
rc, err := client.ContainerExport(ctx, cID)
if err == nil {
defer rc.Close()
_, err = io.Copy(io.Discard, rc)
}
return err
}},
{name: "cp to container", operation: func(t *testing.T) error {
archive, err := archive.Generate("new-file", "hello-world")
assert.NilError(t, err, "failed to create a temporary archive")
return client.CopyToContainer(ctx, cID, "/", archive, types.CopyToContainerOptions{})
}},
{name: "cp from container", operation: func(*testing.T) error {
rc, _, err := client.CopyFromContainer(ctx, cID, "/file")
if err == nil {
defer rc.Close()
_, err = io.Copy(io.Discard, rc)
}
return err
}},
}
for _, tc := range testCases {
tc := tc
t.Run(tc.name, func(t *testing.T) {
prev := dmesgLines(256)
err := tc.operation(t)
assert.NilError(t, err)
after := dmesgLines(2048)
diff := diffDmesg(prev, after)
for _, line := range diff {
overlayfs := strings.Contains(line, "overlayfs: ")
lowerDirInUse := strings.Contains(line, "lowerdir is in-use as ")
upperDirInUse := strings.Contains(line, "upperdir is in-use as ")
workDirInuse := strings.Contains(line, "workdir is in-use as ")
undefinedBehavior := strings.Contains(line, "will result in undefined behavior")
if overlayfs && (lowerDirInUse || upperDirInUse || workDirInuse) && undefinedBehavior {
t.Errorf("%s caused overlayfs kernel warning: %s", tc.name, line)
}
}
})
}
}
func dmesgLines(bytes int) []string {
data := dmesg.Dmesg(bytes)
return strings.Split(strings.TrimSpace(string(data)), "\n")
}
func diffDmesg(prev, next []string) []string {
// All lines have a timestamp, so just take the last one from the previous
// log and find it in the new log.
lastPrev := prev[len(prev)-1]
for idx := len(next) - 1; idx >= 0; idx-- {
line := next[idx]
if line == lastPrev {
nextIdx := idx + 1
if nextIdx < len(next) {
return next[nextIdx:]
} else {
// Found at the last position, log is the same.
return nil
}
}
}
return next
}