
Because runc will delete a container after a successful checkpoint we need to handle a NotFound error from runc on delete. There is also a race between SIGKILL'ing the shim and it actually exiting to unmount the tasks rootfs, we need to loop and wait for the task to actually be reaped before trying to delete the rootfs+bundle. Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
214 lines
4.8 KiB
Go
214 lines
4.8 KiB
Go
package containerd
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"flag"
|
|
"fmt"
|
|
"io/ioutil"
|
|
golog "log"
|
|
"os"
|
|
"os/exec"
|
|
"runtime"
|
|
"syscall"
|
|
"testing"
|
|
"time"
|
|
|
|
"google.golang.org/grpc/grpclog"
|
|
|
|
"github.com/containerd/containerd/log"
|
|
"github.com/containerd/containerd/namespaces"
|
|
"github.com/containerd/containerd/testutil"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
address string
|
|
noDaemon bool
|
|
noCriu bool
|
|
supportsCriu bool
|
|
)
|
|
|
|
func init() {
|
|
// Discard grpc logs so that they don't mess with our stdio
|
|
grpclog.SetLogger(golog.New(ioutil.Discard, "", golog.LstdFlags))
|
|
|
|
flag.StringVar(&address, "address", defaultAddress, "The address to the containerd socket for use in the tests")
|
|
flag.BoolVar(&noDaemon, "no-daemon", false, "Do not start a dedicated daemon for the tests")
|
|
flag.BoolVar(&noCriu, "no-criu", false, "Do not run the checkpoint tests")
|
|
flag.Parse()
|
|
}
|
|
|
|
func testContext() (context.Context, context.CancelFunc) {
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
ctx = namespaces.WithNamespace(ctx, "testing")
|
|
return ctx, cancel
|
|
}
|
|
|
|
func TestMain(m *testing.M) {
|
|
if testing.Short() {
|
|
os.Exit(m.Run())
|
|
}
|
|
testutil.RequiresRootM()
|
|
// check if criu is installed on the system
|
|
_, err := exec.LookPath("criu")
|
|
supportsCriu = err == nil && !noCriu
|
|
|
|
var (
|
|
cmd *exec.Cmd
|
|
buf = bytes.NewBuffer(nil)
|
|
ctx, cancel = testContext()
|
|
)
|
|
defer cancel()
|
|
|
|
if !noDaemon {
|
|
os.RemoveAll(defaultRoot)
|
|
|
|
// setup a new containerd daemon if !testing.Short
|
|
cmd = exec.Command("containerd",
|
|
"--root", defaultRoot,
|
|
"--address", address,
|
|
"--log-level", "debug",
|
|
)
|
|
cmd.Stdout = buf
|
|
cmd.Stderr = buf
|
|
if err := cmd.Start(); err != nil {
|
|
cmd.Wait()
|
|
fmt.Fprintf(os.Stderr, "%s: %s", err, buf.String())
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
client, err := waitForDaemonStart(ctx, address)
|
|
if err != nil {
|
|
if cmd.Process != nil {
|
|
cmd.Process.Kill()
|
|
}
|
|
cmd.Wait()
|
|
fmt.Fprintf(os.Stderr, "%s: %s", err, buf.String())
|
|
os.Exit(1)
|
|
}
|
|
|
|
// print out the version in information
|
|
version, err := client.Version(ctx)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error getting version: %s", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
// allow comparison with containerd under test
|
|
log.G(ctx).WithFields(logrus.Fields{
|
|
"version": version.Version,
|
|
"revision": version.Revision,
|
|
}).Info("running tests against containerd")
|
|
|
|
// pull a seed image
|
|
if runtime.GOOS != "windows" { // TODO: remove once pull is supported on windows
|
|
if _, err = client.Pull(ctx, testImage, WithPullUnpack); err != nil {
|
|
cmd.Process.Signal(syscall.SIGTERM)
|
|
cmd.Wait()
|
|
fmt.Fprintf(os.Stderr, "%s: %s", err, buf.String())
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
if err := platformTestSetup(client); err != nil {
|
|
fmt.Fprintln(os.Stderr, "platform test setup failed", err)
|
|
os.Exit(1)
|
|
}
|
|
|
|
if err := client.Close(); err != nil {
|
|
fmt.Fprintln(os.Stderr, "failed to close client", err)
|
|
}
|
|
|
|
// run the test
|
|
status := m.Run()
|
|
|
|
if !noDaemon {
|
|
// tear down the daemon and resources created
|
|
if err := cmd.Process.Signal(syscall.SIGTERM); err != nil {
|
|
if err := cmd.Process.Kill(); err != nil {
|
|
fmt.Fprintln(os.Stderr, "failed to signal containerd", err)
|
|
}
|
|
}
|
|
if err := cmd.Wait(); err != nil {
|
|
if _, ok := err.(*exec.ExitError); !ok {
|
|
fmt.Fprintln(os.Stderr, "failed to wait for containerd", err)
|
|
}
|
|
}
|
|
if err := os.RemoveAll(defaultRoot); err != nil {
|
|
fmt.Fprintln(os.Stderr, "failed to remove test root dir", err)
|
|
os.Exit(1)
|
|
}
|
|
// only print containerd logs if the test failed
|
|
if status != 0 {
|
|
fmt.Fprintln(os.Stderr, buf.String())
|
|
}
|
|
}
|
|
os.Exit(status)
|
|
}
|
|
|
|
func waitForDaemonStart(ctx context.Context, address string) (*Client, error) {
|
|
var (
|
|
client *Client
|
|
serving bool
|
|
err error
|
|
)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
if client == nil {
|
|
client, err = New(address)
|
|
}
|
|
if err == nil {
|
|
serving, err = client.IsServing(ctx)
|
|
if serving {
|
|
return client, nil
|
|
}
|
|
}
|
|
time.Sleep(100 * time.Millisecond)
|
|
}
|
|
return nil, fmt.Errorf("containerd did not start within 2s: %v", err)
|
|
}
|
|
|
|
func newClient(t testing.TB, address string, opts ...ClientOpt) (*Client, error) {
|
|
if testing.Short() {
|
|
t.Skip()
|
|
}
|
|
// testutil.RequiresRoot(t) is not needed here (already called in TestMain)
|
|
return New(address, opts...)
|
|
}
|
|
|
|
func TestNewClient(t *testing.T) {
|
|
client, err := newClient(t, address)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if client == nil {
|
|
t.Fatal("New() returned nil client")
|
|
}
|
|
if err := client.Close(); err != nil {
|
|
t.Errorf("client closed returned errror %v", err)
|
|
}
|
|
}
|
|
|
|
func TestImagePull(t *testing.T) {
|
|
if runtime.GOOS == "windows" {
|
|
// TODO: remove once Windows has a snapshotter
|
|
t.Skip("Windows does not have a snapshotter yet")
|
|
}
|
|
|
|
client, err := newClient(t, address)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer client.Close()
|
|
|
|
ctx, cancel := testContext()
|
|
defer cancel()
|
|
_, err = client.Pull(ctx, testImage)
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
}
|