2018-02-05 16:05:59 -05:00
|
|
|
package local // import "github.com/docker/docker/volume/local"
|
2015-09-08 22:01:02 -04:00
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2016-08-17 20:36:17 -04:00
|
|
|
"path/filepath"
|
2016-08-01 23:02:51 -04:00
|
|
|
"reflect"
|
2016-02-29 11:57:30 -05:00
|
|
|
"runtime"
|
2016-02-11 21:48:16 -05:00
|
|
|
"strings"
|
2015-09-08 22:01:02 -04:00
|
|
|
"testing"
|
2016-02-11 21:48:16 -05:00
|
|
|
|
2017-05-19 18:06:46 -04:00
|
|
|
"github.com/docker/docker/pkg/idtools"
|
2020-03-13 19:38:24 -04:00
|
|
|
"github.com/moby/sys/mountinfo"
|
2020-02-07 08:39:24 -05:00
|
|
|
"gotest.tools/v3/skip"
|
2015-09-08 22:01:02 -04:00
|
|
|
)
|
|
|
|
|
2016-10-12 16:11:20 -04:00
|
|
|
func TestGetAddress(t *testing.T) {
|
|
|
|
cases := map[string]string{
|
|
|
|
"addr=11.11.11.1": "11.11.11.1",
|
|
|
|
" ": "",
|
|
|
|
"addr=": "",
|
|
|
|
"addr=2001:db8::68": "2001:db8::68",
|
|
|
|
}
|
|
|
|
for name, success := range cases {
|
|
|
|
v := getAddress(name)
|
|
|
|
if v != success {
|
|
|
|
t.Errorf("Test case failed for %s actual: %s expected : %s", name, v, success)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-09-08 22:01:02 -04:00
|
|
|
func TestRemove(t *testing.T) {
|
2018-04-20 05:59:08 -04:00
|
|
|
skip.If(t, runtime.GOOS == "windows", "FIXME: investigate why this test fails on CI")
|
2015-09-08 22:01:02 -04:00
|
|
|
rootDir, err := ioutil.TempDir("", "local-volume-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(rootDir)
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err := New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2015-09-08 22:01:02 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := r.Create("testing", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := r.Remove(vol); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err = r.Create("testing2", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := os.RemoveAll(vol.Path()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := r.Remove(vol); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := os.Stat(vol.Path()); err != nil && !os.IsNotExist(err) {
|
|
|
|
t.Fatal("volume dir not removed")
|
|
|
|
}
|
|
|
|
|
2015-09-23 16:29:14 -04:00
|
|
|
if l, _ := r.List(); len(l) != 0 {
|
2015-09-08 22:01:02 -04:00
|
|
|
t.Fatal("expected there to be no volumes")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInitializeWithVolumes(t *testing.T) {
|
|
|
|
rootDir, err := ioutil.TempDir("", "local-volume-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(rootDir)
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err := New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2015-09-08 22:01:02 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := r.Create("testing", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err = New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2015-09-08 22:01:02 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v, err := r.Get(vol.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if v.Path() != vol.Path() {
|
|
|
|
t.Fatal("expected to re-initialize root with existing volumes")
|
|
|
|
}
|
|
|
|
}
|
2015-10-20 13:09:48 -04:00
|
|
|
|
|
|
|
func TestCreate(t *testing.T) {
|
|
|
|
rootDir, err := ioutil.TempDir("", "local-volume-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(rootDir)
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err := New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2015-10-20 13:09:48 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := map[string]bool{
|
|
|
|
"name": true,
|
|
|
|
"name-with-dash": true,
|
|
|
|
"name_with_underscore": true,
|
|
|
|
"name/with/slash": false,
|
|
|
|
"name/with/../../slash": false,
|
|
|
|
"./name": false,
|
|
|
|
"../name": false,
|
|
|
|
"./": false,
|
|
|
|
"../": false,
|
|
|
|
"~": false,
|
|
|
|
".": false,
|
|
|
|
"..": false,
|
|
|
|
"...": false,
|
|
|
|
}
|
|
|
|
|
|
|
|
for name, success := range cases {
|
|
|
|
v, err := r.Create(name, nil)
|
|
|
|
if success {
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if v.Name() != name {
|
|
|
|
t.Fatalf("Expected volume with name %s, got %s", name, v.Name())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("Expected error creating volume with name %s, got nil", name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-17 20:36:17 -04:00
|
|
|
|
2019-08-05 11:49:50 -04:00
|
|
|
_, err = New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2016-08-17 20:36:17 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-10-20 13:09:48 -04:00
|
|
|
}
|
2016-01-04 18:00:49 -05:00
|
|
|
|
|
|
|
func TestValidateName(t *testing.T) {
|
|
|
|
r := &Root{}
|
|
|
|
names := map[string]bool{
|
2016-08-17 10:40:24 -04:00
|
|
|
"x": false,
|
2016-01-04 18:00:49 -05:00
|
|
|
"/testvol": false,
|
|
|
|
"thing.d": true,
|
|
|
|
"hello-world": true,
|
|
|
|
"./hello": false,
|
|
|
|
".hello": false,
|
|
|
|
}
|
|
|
|
|
|
|
|
for vol, expected := range names {
|
|
|
|
err := r.validateName(vol)
|
|
|
|
if expected && err != nil {
|
|
|
|
t.Fatalf("expected %s to be valid got %v", vol, err)
|
|
|
|
}
|
|
|
|
if !expected && err == nil {
|
|
|
|
t.Fatalf("expected %s to be invalid", vol)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-11 21:48:16 -05:00
|
|
|
|
|
|
|
func TestCreateWithOpts(t *testing.T) {
|
2018-04-20 05:59:08 -04:00
|
|
|
skip.If(t, runtime.GOOS == "windows")
|
2018-04-24 10:26:10 -04:00
|
|
|
skip.If(t, os.Getuid() != 0, "requires mounts")
|
2016-02-11 21:48:16 -05:00
|
|
|
rootDir, err := ioutil.TempDir("", "local-volume-test")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(rootDir)
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err := New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2016-02-11 21:48:16 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := r.Create("test", map[string]string{"invalidopt": "notsupported"}); err == nil {
|
|
|
|
t.Fatal("expected invalid opt to cause error")
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := r.Create("test", map[string]string{"device": "tmpfs", "type": "tmpfs", "o": "size=1m,uid=1000"})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
v := vol.(*localVolume)
|
|
|
|
|
2016-03-07 21:41:44 -05:00
|
|
|
dir, err := v.Mount("1234")
|
2016-02-11 21:48:16 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer func() {
|
2016-03-07 21:41:44 -05:00
|
|
|
if err := v.Unmount("1234"); err != nil {
|
2016-02-11 21:48:16 -05:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-03-13 19:38:24 -04:00
|
|
|
mountInfos, err := mountinfo.GetMounts(mountinfo.SingleEntryFilter(dir))
|
2016-02-11 21:48:16 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-03-08 18:27:09 -05:00
|
|
|
if len(mountInfos) != 1 {
|
|
|
|
t.Fatalf("expected 1 mount, found %d: %+v", len(mountInfos), mountInfos)
|
2016-02-11 21:48:16 -05:00
|
|
|
}
|
|
|
|
|
2018-03-08 18:27:09 -05:00
|
|
|
info := mountInfos[0]
|
|
|
|
t.Logf("%+v", info)
|
2020-09-16 04:27:03 -04:00
|
|
|
if info.FSType != "tmpfs" {
|
|
|
|
t.Fatalf("expected tmpfs mount, got %q", info.FSType)
|
2018-03-08 18:27:09 -05:00
|
|
|
}
|
|
|
|
if info.Source != "tmpfs" {
|
|
|
|
t.Fatalf("expected tmpfs mount, got %q", info.Source)
|
|
|
|
}
|
2020-09-16 04:27:03 -04:00
|
|
|
if !strings.Contains(info.VFSOptions, "uid=1000") {
|
|
|
|
t.Fatalf("expected mount info to have uid=1000: %q", info.VFSOptions)
|
2018-03-08 18:27:09 -05:00
|
|
|
}
|
2020-09-16 04:27:03 -04:00
|
|
|
if !strings.Contains(info.VFSOptions, "size=1024k") {
|
|
|
|
t.Fatalf("expected mount info to have size=1024k: %q", info.VFSOptions)
|
2016-02-11 21:48:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if v.active.count != 1 {
|
|
|
|
t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
|
|
|
|
}
|
|
|
|
|
|
|
|
// test double mount
|
2016-03-07 21:41:44 -05:00
|
|
|
if _, err := v.Mount("1234"); err != nil {
|
2016-02-11 21:48:16 -05:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if v.active.count != 2 {
|
|
|
|
t.Fatalf("Expected active mount count to be 2, got %d", v.active.count)
|
|
|
|
}
|
|
|
|
|
2016-03-07 21:41:44 -05:00
|
|
|
if err := v.Unmount("1234"); err != nil {
|
2016-02-11 21:48:16 -05:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if v.active.count != 1 {
|
|
|
|
t.Fatalf("Expected active mount count to be 1, got %d", v.active.count)
|
|
|
|
}
|
|
|
|
|
2020-03-13 19:38:24 -04:00
|
|
|
mounted, err := mountinfo.Mounted(v.path)
|
2016-02-11 21:48:16 -05:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if !mounted {
|
|
|
|
t.Fatal("expected mount to still be active")
|
|
|
|
}
|
2016-08-01 23:02:51 -04:00
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err = New(rootDir, idtools.Identity{UID: 0, GID: 0})
|
2016-08-01 23:02:51 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v2, exists := r.volumes["test"]
|
|
|
|
if !exists {
|
|
|
|
t.Fatal("missing volume on restart")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(v.opts, v2.opts) {
|
|
|
|
t.Fatal("missing volume options on restart")
|
|
|
|
}
|
2016-02-11 21:48:16 -05:00
|
|
|
}
|
2016-08-17 20:36:17 -04:00
|
|
|
|
2018-04-20 05:59:08 -04:00
|
|
|
func TestRelaodNoOpts(t *testing.T) {
|
2016-08-17 20:36:17 -04:00
|
|
|
rootDir, err := ioutil.TempDir("", "volume-test-reload-no-opts")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(rootDir)
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err := New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2016-08-17 20:36:17 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := r.Create("test1", nil); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if _, err := r.Create("test2", nil); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// make sure a file with `null` (.e.g. empty opts map from older daemon) is ok
|
2019-08-05 11:49:50 -04:00
|
|
|
if err := ioutil.WriteFile(filepath.Join(rootDir, "test2"), []byte("null"), 0600); err != nil {
|
2016-08-17 20:36:17 -04:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := r.Create("test3", nil); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// make sure an empty opts file doesn't break us too
|
2019-08-05 11:49:50 -04:00
|
|
|
if err := ioutil.WriteFile(filepath.Join(rootDir, "test3"), nil, 0600); err != nil {
|
2016-08-17 20:36:17 -04:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := r.Create("test4", map[string]string{}); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2017-11-16 01:20:33 -05:00
|
|
|
r, err = New(rootDir, idtools.Identity{UID: os.Geteuid(), GID: os.Getegid()})
|
2016-08-17 20:36:17 -04:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, name := range []string{"test1", "test2", "test3", "test4"} {
|
|
|
|
v, err := r.Get(name)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
lv, ok := v.(*localVolume)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("expected *localVolume got: %v", reflect.TypeOf(v))
|
|
|
|
}
|
|
|
|
if lv.opts != nil {
|
|
|
|
t.Fatalf("expected opts to be nil, got: %v", lv.opts)
|
|
|
|
}
|
|
|
|
if _, err := lv.Mount("1234"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|