Remove shelling out to mount

Also remove the target from the Mount struct because it should not be
used at all.  The target can be variable and set by a caller, not by the
snapshot drivers.

Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
This commit is contained in:
Michael Crosby 2017-01-17 13:26:56 -08:00
parent c0fdda03cc
commit a7a6270f2a
9 changed files with 28 additions and 101 deletions

View file

@ -1,8 +1,6 @@
package containerd package containerd
import ( import (
"os"
"os/exec"
"strings" "strings"
"syscall" "syscall"
) )
@ -12,48 +10,23 @@ import (
type Mount struct { type Mount struct {
// Type specifies the host-specific of the mount. // Type specifies the host-specific of the mount.
Type string Type string
// Source specifies where to mount from. Depending on the host system, this // Source specifies where to mount from. Depending on the host system, this
// can be a source path or device. // can be a source path or device.
Source string Source string
// Target is the filesystem mount location.
Target string
// Options contains zero or more fstab-style mount options. Typically, // Options contains zero or more fstab-style mount options. Typically,
// these are platform specific. // these are platform specific.
Options []string Options []string
} }
// MountCommand converts the provided mount into a CLI arguments that can be used to mount the func (m *Mount) Mount(target string) error {
func MountCommand(m Mount) []string {
return []string{
"mount",
"-t", strings.ToLower(m.Type),
m.Source,
m.Target,
"-o", strings.Join(m.Options, ","),
}
}
func MountAll(mounts ...Mount) error {
for _, mount := range mounts {
cmd := exec.Command("mount", MountCommand(mount)[1:]...)
cmd.Stderr = os.Stderr
cmd.Stdout = os.Stdout
if err := cmd.Run(); err != nil {
return err
}
}
return nil
}
func MountFS(mounts []Mount, target string) error {
for _, m := range mounts {
flags, data := parseMountOptions(m.Options) flags, data := parseMountOptions(m.Options)
if err := syscall.Mount(m.Source, target, m.Type, uintptr(flags), data); err != nil { return syscall.Mount(m.Source, target, m.Type, uintptr(flags), data)
}
// MountAll mounts all the provided mounts to the provided target
func MountAll(mounts []Mount, target string) error {
for _, m := range mounts {
if err := m.Mount(target); err != nil {
return err return err
} }
} }

View file

@ -50,8 +50,6 @@ func (lm *Btrfs) Prepare(key, parent string) ([]containerd.Mount, error) {
{ {
Type: "btrfs", Type: "btrfs",
Source: lm.device, // device? Source: lm.device, // device?
Target: key,
// NOTE(stevvooe): While it would be nice to use to uuids for // NOTE(stevvooe): While it would be nice to use to uuids for
// mounts, they don't work reliably if the uuids are missing. // mounts, they don't work reliably if the uuids are missing.
Options: []string{fmt.Sprintf("subvolid=%d", info.ID)}, Options: []string{fmt.Sprintf("subvolid=%d", info.ID)},

View file

@ -27,11 +27,12 @@ func TestBtrfs(t *testing.T) {
} }
defer os.RemoveAll(root) defer os.RemoveAll(root)
target := filepath.Join(root, "test")
sm, err := NewBtrfs(device.deviceName, root) sm, err := NewBtrfs(device.deviceName, root)
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
mounts, err := sm.Prepare(filepath.Join(root, "test"), "") mounts, err := sm.Prepare(target, "")
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -48,16 +49,16 @@ func TestBtrfs(t *testing.T) {
} }
} }
if err := os.MkdirAll(mounts[0].Target, 0755); err != nil { if err := os.MkdirAll(target, 0755); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := containerd.MountAll(mounts...); err != nil { if err := containerd.MountAll(mounts, target); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer testutil.UnmountAll(t, mounts) defer testutil.Unmount(t, target)
// write in some data // write in some data
if err := ioutil.WriteFile(filepath.Join(mounts[0].Target, "foo"), []byte("content"), 0777); err != nil { if err := ioutil.WriteFile(filepath.Join(target, "foo"), []byte("content"), 0777); err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -78,26 +79,27 @@ func TestBtrfs(t *testing.T) {
} }
}() }()
mounts, err = sm.Prepare(filepath.Join(root, "test2"), filepath.Join(root, "snapshots/committed")) target = filepath.Join(root, "test2")
mounts, err = sm.Prepare(target, filepath.Join(root, "snapshots/committed"))
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := os.MkdirAll(filepath.Join(root, "test2"), 0755); err != nil { if err := os.MkdirAll(target, 0755); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := containerd.MountAll(mounts...); err != nil { if err := containerd.MountAll(mounts, target); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer testutil.UnmountAll(t, mounts) defer testutil.Unmount(t, target)
// TODO(stevvooe): Verify contents of "foo" // TODO(stevvooe): Verify contents of "foo"
if err := ioutil.WriteFile(filepath.Join(mounts[0].Target, "bar"), []byte("content"), 0777); err != nil { if err := ioutil.WriteFile(filepath.Join(target, "bar"), []byte("content"), 0777); err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := sm.Commit(filepath.Join(root, "snapshots/committed2"), filepath.Join(root, "test2")); err != nil { if err := sm.Commit(filepath.Join(root, "snapshots/committed2"), target); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer func() { defer func() {

View file

@ -197,7 +197,6 @@ func (lm *Manager) Prepare(dst, parent string) ([]containerd.Mount, error) {
{ {
Type: "overlay", Type: "overlay",
Source: "none", Source: "none",
Target: dst,
Options: opts, Options: opts,
}, },
}, nil }, nil

View file

@ -4,7 +4,6 @@ import (
"io/ioutil" "io/ioutil"
"os" "os"
"path/filepath" "path/filepath"
"strings"
"testing" "testing"
"github.com/docker/containerd" "github.com/docker/containerd"
@ -48,18 +47,10 @@ func TestSnapshotManagerBasic(t *testing.T) {
t.Fatal("expected mounts to have entries") t.Fatal("expected mounts to have entries")
} }
for _, mount := range mounts { if err := containerd.MountAll(mounts, preparing); err != nil {
if !strings.HasPrefix(mount.Target, preparing) {
t.Fatalf("expected mount target to be prefixed with tmpDir: %q does not startwith %q", mount.Target, preparing)
}
t.Log(containerd.MountCommand(mount))
}
if err := containerd.MountAll(mounts...); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer testutil.UnmountAll(t, mounts) defer testutil.Unmount(t, preparing)
if err := ioutil.WriteFile(filepath.Join(preparing, "foo"), []byte("foo\n"), 0777); err != nil { if err := ioutil.WriteFile(filepath.Join(preparing, "foo"), []byte("foo\n"), 0777); err != nil {
t.Fatal(err) t.Fatal(err)
@ -86,18 +77,10 @@ func TestSnapshotManagerBasic(t *testing.T) {
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := containerd.MountAll(mounts...); err != nil { if err := containerd.MountAll(mounts, next); err != nil {
t.Fatal(err) t.Fatal(err)
} }
defer testutil.UnmountAll(t, mounts) defer testutil.Unmount(t, next)
for _, mount := range mounts {
if !strings.HasPrefix(mount.Target, next) {
t.Fatalf("expected mount target to be prefixed with tmpDir: %q does not startwith %q", mount.Target, next)
}
t.Log(containerd.MountCommand(mount))
}
if err := ioutil.WriteFile(filepath.Join(next, "bar"), []byte("bar\n"), 0777); err != nil { if err := ioutil.WriteFile(filepath.Join(next, "bar"), []byte("bar\n"), 0777); err != nil {
t.Fatal(err) t.Fatal(err)

View file

@ -4,7 +4,6 @@ import (
"io/ioutil" "io/ioutil"
"os" "os"
"path/filepath" "path/filepath"
"strings"
"testing" "testing"
"github.com/docker/containerd" "github.com/docker/containerd"
@ -35,15 +34,7 @@ func TestSnapshotNaiveBasic(t *testing.T) {
t.Fatal(err) t.Fatal(err)
} }
for _, mount := range mounts { if err := containerd.MountAll(mounts, preparing); err != nil {
if !strings.HasPrefix(mount.Target, preparing) {
t.Fatalf("expected mount target to be prefixed with tmpDir: %q does not startwith %q", mount.Target, preparing)
}
t.Log(containerd.MountCommand(mount))
}
if err := containerd.MountAll(mounts...); err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -74,18 +65,10 @@ func TestSnapshotNaiveBasic(t *testing.T) {
if err != nil { if err != nil {
t.Fatal(err) t.Fatal(err)
} }
if err := containerd.MountAll(mounts...); err != nil { if err := containerd.MountAll(mounts, next); err != nil {
t.Fatal(err) t.Fatal(err)
} }
for _, mount := range mounts {
if !strings.HasPrefix(mount.Target, next) {
t.Fatalf("expected mount target to be prefixed with tmpDir: %q does not startwith %q", mount.Target, next)
}
t.Log(containerd.MountCommand(mount))
}
if err := ioutil.WriteFile(filepath.Join(next, "bar"), []byte("bar\n"), 0777); err != nil { if err := ioutil.WriteFile(filepath.Join(next, "bar"), []byte("bar\n"), 0777); err != nil {
t.Fatal(err) t.Fatal(err)
} }

View file

@ -163,7 +163,7 @@ func TestOverlayfsOverlayRead(t *testing.T) {
t.Error(err) t.Error(err)
return return
} }
if err := containerd.MountFS(mounts, dest); err != nil { if err := containerd.MountAll(mounts, dest); err != nil {
t.Error(err) t.Error(err)
return return
} }

BIN
snapshot/snapshot.test Executable file

Binary file not shown.

View file

@ -1,24 +1,13 @@
package testutil package testutil
import ( import (
"os/exec" "syscall"
"testing" "testing"
"github.com/docker/containerd"
) )
func UnmountAll(t *testing.T, mounts []containerd.Mount) {
for _, mount := range mounts {
Unmount(t, mount.Target)
}
}
func Unmount(t *testing.T, mountPoint string) { func Unmount(t *testing.T, mountPoint string) {
t.Log("unmount", mountPoint) t.Log("unmount", mountPoint)
umount := exec.Command("umount", mountPoint) if err := syscall.Unmount(mountPoint, 0); err != nil {
err := umount.Run()
if err != nil {
t.Error("Could not umount", mountPoint, err) t.Error("Could not umount", mountPoint, err)
} }
} }