ScaleIO Volume Plugin - volume attribute updates
This commit introduces the following updates and fixes: - Enable scaleIO volume multip-mapping based on accessMode - No longer uses "default" as default values for storagepool & protection domain - validates capacity when capacity is zero - Better naming for PV and volume - make mount ro when accessModes contains ROM
This commit is contained in:
		| @@ -22,6 +22,7 @@ go_test( | |||||||
|         "//pkg/volume:go_default_library", |         "//pkg/volume:go_default_library", | ||||||
|         "//pkg/volume/testing:go_default_library", |         "//pkg/volume/testing:go_default_library", | ||||||
|         "//vendor/github.com/codedellemc/goscaleio/types/v1:go_default_library", |         "//vendor/github.com/codedellemc/goscaleio/types/v1:go_default_library", | ||||||
|  |         "//vendor/github.com/golang/glog:go_default_library", | ||||||
|         "//vendor/k8s.io/api/core/v1:go_default_library", |         "//vendor/k8s.io/api/core/v1:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library", | ||||||
|         "//vendor/k8s.io/apimachinery/pkg/types:go_default_library", |         "//vendor/k8s.io/apimachinery/pkg/types:go_default_library", | ||||||
|   | |||||||
| @@ -45,7 +45,7 @@ type sioInterface interface { | |||||||
| 	FindVolume(name string) (*siotypes.Volume, error) | 	FindVolume(name string) (*siotypes.Volume, error) | ||||||
| 	Volume(sioVolumeID) (*siotypes.Volume, error) | 	Volume(sioVolumeID) (*siotypes.Volume, error) | ||||||
| 	CreateVolume(name string, sizeGB int64) (*siotypes.Volume, error) | 	CreateVolume(name string, sizeGB int64) (*siotypes.Volume, error) | ||||||
| 	AttachVolume(sioVolumeID) error | 	AttachVolume(sioVolumeID, bool) error | ||||||
| 	DetachVolume(sioVolumeID) error | 	DetachVolume(sioVolumeID) error | ||||||
| 	DeleteVolume(sioVolumeID) error | 	DeleteVolume(sioVolumeID) error | ||||||
| 	IID() (string, error) | 	IID() (string, error) | ||||||
| @@ -217,8 +217,9 @@ func (c *sioClient) CreateVolume(name string, sizeGB int64) (*siotypes.Volume, e | |||||||
| 	return c.Volume(sioVolumeID(createResponse.ID)) | 	return c.Volume(sioVolumeID(createResponse.ID)) | ||||||
| } | } | ||||||
|  |  | ||||||
| // AttachVolume maps the scaleio volume to an sdc node. | // AttachVolume maps the scaleio volume to an sdc node.  If the multipleMappings flag | ||||||
| func (c *sioClient) AttachVolume(id sioVolumeID) error { | // is true, ScaleIO will allow other SDC to map to that volume. | ||||||
|  | func (c *sioClient) AttachVolume(id sioVolumeID, multipleMappings bool) error { | ||||||
| 	if err := c.init(); err != nil { | 	if err := c.init(); err != nil { | ||||||
| 		glog.Error(log("failed to init'd client in attach volume: %v", err)) | 		glog.Error(log("failed to init'd client in attach volume: %v", err)) | ||||||
| 		return err | 		return err | ||||||
| @@ -232,7 +233,7 @@ func (c *sioClient) AttachVolume(id sioVolumeID) error { | |||||||
|  |  | ||||||
| 	params := &siotypes.MapVolumeSdcParam{ | 	params := &siotypes.MapVolumeSdcParam{ | ||||||
| 		SdcID: iid, | 		SdcID: iid, | ||||||
| 		AllowMultipleMappings: "false", | 		AllowMultipleMappings: strconv.FormatBool(multipleMappings), | ||||||
| 		AllSdcs:               "", | 		AllSdcs:               "", | ||||||
| 	} | 	} | ||||||
| 	volClient := sio.NewVolume(c.client) | 	volClient := sio.NewVolume(c.client) | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ import ( | |||||||
|  |  | ||||||
| type storageInterface interface { | type storageInterface interface { | ||||||
| 	CreateVolume(string, int64) (*siotypes.Volume, error) | 	CreateVolume(string, int64) (*siotypes.Volume, error) | ||||||
| 	AttachVolume(string) (string, error) | 	AttachVolume(string, bool) (string, error) | ||||||
| 	IsAttached(string) (bool, error) | 	IsAttached(string) (bool, error) | ||||||
| 	DetachVolume(string) error | 	DetachVolume(string) error | ||||||
| 	DeleteVolume(string) error | 	DeleteVolume(string) error | ||||||
| @@ -103,8 +103,9 @@ func (m *sioMgr) CreateVolume(volName string, sizeGB int64) (*siotypes.Volume, e | |||||||
| 	return vol, nil | 	return vol, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // AttachVolume maps a ScaleIO volume to the running node | // AttachVolume maps a ScaleIO volume to the running node.  If flag multiMaps, | ||||||
| func (m *sioMgr) AttachVolume(volName string) (string, error) { | // ScaleIO will allow other SDC to map to volume. | ||||||
|  | func (m *sioMgr) AttachVolume(volName string, multipleMappings bool) (string, error) { | ||||||
| 	client, err := m.getClient() | 	client, err := m.getClient() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		glog.Error(log("attach volume failed: %v", err)) | 		glog.Error(log("attach volume failed: %v", err)) | ||||||
| @@ -139,7 +140,7 @@ func (m *sioMgr) AttachVolume(volName string) (string, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// attach volume, get deviceName | 	// attach volume, get deviceName | ||||||
| 	if err := client.AttachVolume(sioVolumeID(vol.ID)); err != nil { | 	if err := client.AttachVolume(sioVolumeID(vol.ID), multipleMappings); err != nil { | ||||||
| 		glog.Error(log("attachment for volume %s failed :%v", volName, err)) | 		glog.Error(log("attachment for volume %s failed :%v", volName, err)) | ||||||
| 		return "", err | 		return "", err | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -99,7 +99,7 @@ func TestMgrCreateVolume(t *testing.T) { | |||||||
| func TestMgrAttachVolume(t *testing.T) { | func TestMgrAttachVolume(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	device, err := mgr.AttachVolume("test-vol-0001") | 	device, err := mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
| @@ -111,8 +111,8 @@ func TestMgrAttachVolume(t *testing.T) { | |||||||
| func TestMgrAttachVolume_AlreadyAttached(t *testing.T) { | func TestMgrAttachVolume_AlreadyAttached(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	mgr.AttachVolume("test-vol-0001") | 	mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	dev, err := mgr.AttachVolume("test-vol-0001") | 	dev, err := mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatalf("unexpected error: %v", err) | 		t.Fatalf("unexpected error: %v", err) | ||||||
| 	} | 	} | ||||||
| @@ -124,7 +124,8 @@ func TestMgrAttachVolume_AlreadyAttached(t *testing.T) { | |||||||
| func TestMgrAttachVolume_VolumeNotFoundError(t *testing.T) { | func TestMgrAttachVolume_VolumeNotFoundError(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	_, err := mgr.AttachVolume("test-vol-0002") | 	_, err := mgr.AttachVolume("test-vol-0002", false) | ||||||
|  |  | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		t.Error("attachVolume should fail with volume not found error") | 		t.Error("attachVolume should fail with volume not found error") | ||||||
| 	} | 	} | ||||||
| @@ -137,7 +138,7 @@ func TestMgrAttachVolume_WaitForAttachError(t *testing.T) { | |||||||
| 		c := mgr.client.(*fakeSio) | 		c := mgr.client.(*fakeSio) | ||||||
| 		close(c.waitAttachCtrl) | 		close(c.waitAttachCtrl) | ||||||
| 	}() | 	}() | ||||||
| 	_, err := mgr.AttachVolume("test-vol-0001") | 	_, err := mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		t.Error("attachVolume should fail with attach timeout error") | 		t.Error("attachVolume should fail with attach timeout error") | ||||||
| 	} | 	} | ||||||
| @@ -146,7 +147,7 @@ func TestMgrAttachVolume_WaitForAttachError(t *testing.T) { | |||||||
| func TestMgrDetachVolume(t *testing.T) { | func TestMgrDetachVolume(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	mgr.AttachVolume("test-vol-0001") | 	mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	if err := mgr.DetachVolume("test-vol-0001"); err != nil { | 	if err := mgr.DetachVolume("test-vol-0001"); err != nil { | ||||||
| 		t.Fatal(err) | 		t.Fatal(err) | ||||||
| 	} | 	} | ||||||
| @@ -162,7 +163,7 @@ func TestMgrDetachVolume(t *testing.T) { | |||||||
| func TestMgrDetachVolume_VolumeNotFound(t *testing.T) { | func TestMgrDetachVolume_VolumeNotFound(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	mgr.AttachVolume("test-vol-0001") | 	mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	err := mgr.DetachVolume("test-vol-0002") | 	err := mgr.DetachVolume("test-vol-0002") | ||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		t.Fatal("expected a volume not found failure") | 		t.Fatal("expected a volume not found failure") | ||||||
| @@ -181,7 +182,7 @@ func TestMgrDetachVolume_VolumeNotAttached(t *testing.T) { | |||||||
| func TestMgrDetachVolume_VolumeAlreadyDetached(t *testing.T) { | func TestMgrDetachVolume_VolumeAlreadyDetached(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	mgr.AttachVolume("test-vol-0001") | 	mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	mgr.DetachVolume("test-vol-0001") | 	mgr.DetachVolume("test-vol-0001") | ||||||
| 	err := mgr.DetachVolume("test-vol-0001") | 	err := mgr.DetachVolume("test-vol-0001") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -192,7 +193,7 @@ func TestMgrDetachVolume_VolumeAlreadyDetached(t *testing.T) { | |||||||
| func TestMgrDetachVolume_WaitForDetachError(t *testing.T) { | func TestMgrDetachVolume_WaitForDetachError(t *testing.T) { | ||||||
| 	mgr := newTestMgr(t) | 	mgr := newTestMgr(t) | ||||||
| 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | 	mgr.CreateVolume("test-vol-0001", 8*1024*1024) | ||||||
| 	mgr.AttachVolume("test-vol-0001") | 	mgr.AttachVolume("test-vol-0001", false) | ||||||
| 	err := mgr.DetachVolume("test-vol-0001") | 	err := mgr.DetachVolume("test-vol-0001") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Error("detachVolume failed") | 		t.Error("detachVolume failed") | ||||||
| @@ -227,6 +228,7 @@ type fakeSio struct { | |||||||
| 	waitAttachCtrl chan struct{} | 	waitAttachCtrl chan struct{} | ||||||
| 	waitDetachCtrl chan struct{} | 	waitDetachCtrl chan struct{} | ||||||
| 	devs           map[string]string | 	devs           map[string]string | ||||||
|  | 	isMultiMap     bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func newFakeSio() *fakeSio { | func newFakeSio() *fakeSio { | ||||||
| @@ -261,7 +263,8 @@ func (f *fakeSio) CreateVolume(volName string, sizeGB int64) (*siotypes.Volume, | |||||||
| 	return f.volume, nil | 	return f.volume, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (f *fakeSio) AttachVolume(id sioVolumeID) error { | func (f *fakeSio) AttachVolume(id sioVolumeID, multiMaps bool) error { | ||||||
|  | 	f.isMultiMap = multiMaps | ||||||
| 	_, err := f.Volume(id) | 	_, err := f.Volume(id) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
|   | |||||||
| @@ -149,6 +149,7 @@ var _ volume.PersistentVolumePlugin = &sioPlugin{} | |||||||
| func (p *sioPlugin) GetAccessModes() []api.PersistentVolumeAccessMode { | func (p *sioPlugin) GetAccessModes() []api.PersistentVolumeAccessMode { | ||||||
| 	return []api.PersistentVolumeAccessMode{ | 	return []api.PersistentVolumeAccessMode{ | ||||||
| 		api.ReadWriteOnce, | 		api.ReadWriteOnce, | ||||||
|  | 		api.ReadOnlyMany, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -68,11 +68,13 @@ var ( | |||||||
| 	nsSep       = "%" | 	nsSep       = "%" | ||||||
| 	sdcRootPath = "/opt/emc/scaleio/sdc/bin" | 	sdcRootPath = "/opt/emc/scaleio/sdc/bin" | ||||||
|  |  | ||||||
| 	secretNotFoundErr       = errors.New("secret not found") | 	secretNotFoundErr              = errors.New("secret not found") | ||||||
| 	configMapNotFoundErr    = errors.New("configMap not found") | 	configMapNotFoundErr           = errors.New("configMap not found") | ||||||
| 	gatewayNotProvidedErr   = errors.New("gateway not provided") | 	gatewayNotProvidedErr          = errors.New("ScaleIO gateway not provided") | ||||||
| 	secretRefNotProvidedErr = errors.New("secret ref not provided") | 	secretRefNotProvidedErr        = errors.New("secret ref not provided") | ||||||
| 	systemNotProvidedErr    = errors.New("secret not provided") | 	systemNotProvidedErr           = errors.New("ScaleIO system not provided") | ||||||
|  | 	storagePoolNotProvidedErr      = errors.New("ScaleIO storage pool not provided") | ||||||
|  | 	protectionDomainNotProvidedErr = errors.New("ScaleIO protection domain not provided") | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // mapScaleIOVolumeSource maps attributes from a ScaleIOVolumeSource to config | // mapScaleIOVolumeSource maps attributes from a ScaleIOVolumeSource to config | ||||||
| @@ -107,6 +109,12 @@ func validateConfigs(config map[string]string) error { | |||||||
| 	if config[confKey.system] == "" { | 	if config[confKey.system] == "" { | ||||||
| 		return systemNotProvidedErr | 		return systemNotProvidedErr | ||||||
| 	} | 	} | ||||||
|  | 	if config[confKey.storagePool] == "" { | ||||||
|  | 		return storagePoolNotProvidedErr | ||||||
|  | 	} | ||||||
|  | 	if config[confKey.protectionDomain] == "" { | ||||||
|  | 		return protectionDomainNotProvidedErr | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| @@ -119,8 +127,6 @@ func applyConfigDefaults(config map[string]string) { | |||||||
| 		b = false | 		b = false | ||||||
| 	} | 	} | ||||||
| 	config[confKey.sslEnabled] = strconv.FormatBool(b) | 	config[confKey.sslEnabled] = strconv.FormatBool(b) | ||||||
| 	config[confKey.protectionDomain] = defaultString(config[confKey.protectionDomain], "default") |  | ||||||
| 	config[confKey.storagePool] = defaultString(config[confKey.storagePool], "default") |  | ||||||
| 	config[confKey.storageMode] = defaultString(config[confKey.storageMode], "ThinProvisioned") | 	config[confKey.storageMode] = defaultString(config[confKey.storageMode], "ThinProvisioned") | ||||||
| 	config[confKey.fsType] = defaultString(config[confKey.fsType], "xfs") | 	config[confKey.fsType] = defaultString(config[confKey.fsType], "xfs") | ||||||
| 	b, err = strconv.ParseBool(config[confKey.readOnly]) | 	b, err = strconv.ParseBool(config[confKey.readOnly]) | ||||||
|   | |||||||
| @@ -115,10 +115,10 @@ func TestUtilApplyConfigDefaults(t *testing.T) { | |||||||
| 	if data[confKey.system] != "sio" { | 	if data[confKey.system] != "sio" { | ||||||
| 		t.Error("Unexpected system value") | 		t.Error("Unexpected system value") | ||||||
| 	} | 	} | ||||||
| 	if data[confKey.protectionDomain] != "default" { | 	if data[confKey.protectionDomain] != "" { | ||||||
| 		t.Error("Unexpected protection domain value") | 		t.Error("Unexpected protection domain value") | ||||||
| 	} | 	} | ||||||
| 	if data[confKey.storagePool] != "default" { | 	if data[confKey.storagePool] != "" { | ||||||
| 		t.Error("Unexpected storage pool value") | 		t.Error("Unexpected storage pool value") | ||||||
| 	} | 	} | ||||||
| 	if data[confKey.volumeName] != "sio-vol" { | 	if data[confKey.volumeName] != "sio-vol" { | ||||||
|   | |||||||
| @@ -88,7 +88,7 @@ func (v *sioVolume) SetUpAt(dir string, fsGroup *int64) error { | |||||||
| 	v.plugin.volumeMtx.LockKey(v.volSpecName) | 	v.plugin.volumeMtx.LockKey(v.volSpecName) | ||||||
| 	defer v.plugin.volumeMtx.UnlockKey(v.volSpecName) | 	defer v.plugin.volumeMtx.UnlockKey(v.volSpecName) | ||||||
|  |  | ||||||
| 	glog.V(4).Info(log("setting up volume %s", v.volSpecName)) | 	glog.V(4).Info(log("setting up volume for PV.spec %s", v.volSpecName)) | ||||||
| 	if err := v.setSioMgr(); err != nil { | 	if err := v.setSioMgr(); err != nil { | ||||||
| 		glog.Error(log("setup failed to create scalio manager: %v", err)) | 		glog.Error(log("setup failed to create scalio manager: %v", err)) | ||||||
| 		return err | 		return err | ||||||
| @@ -104,18 +104,36 @@ func (v *sioVolume) SetUpAt(dir string, fsGroup *int64) error { | |||||||
| 		return nil | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// attach the volume and mount | 	// should multiple-mapping be enabled | ||||||
|  | 	enableMultiMaps := false | ||||||
|  | 	isROM := false | ||||||
|  | 	if v.spec.PersistentVolume != nil { | ||||||
|  | 		ams := v.spec.PersistentVolume.Spec.AccessModes | ||||||
|  | 		for _, am := range ams { | ||||||
|  | 			if am == api.ReadOnlyMany { | ||||||
|  | 				enableMultiMaps = true | ||||||
|  | 				isROM = true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	glog.V(4).Info(log("multiple mapping enabled = %v", enableMultiMaps)) | ||||||
|  |  | ||||||
| 	volName := v.volName | 	volName := v.volName | ||||||
| 	devicePath, err := v.sioMgr.AttachVolume(volName) | 	devicePath, err := v.sioMgr.AttachVolume(volName, enableMultiMaps) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		glog.Error(log("setup of volume %v:  %v", v.volSpecName, err)) | 		glog.Error(log("setup of volume %v:  %v", v.volSpecName, err)) | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	options := []string{} | 	options := []string{} | ||||||
| 	if v.source.ReadOnly { | 	switch { | ||||||
| 		options = append(options, "ro") | 	default: | ||||||
| 	} else { |  | ||||||
| 		options = append(options, "rw") | 		options = append(options, "rw") | ||||||
|  | 	case isROM && !v.source.ReadOnly: | ||||||
|  | 		options = append(options, "rw") | ||||||
|  | 	case isROM: | ||||||
|  | 		options = append(options, "ro") | ||||||
|  | 	case v.source.ReadOnly: | ||||||
|  | 		options = append(options, "ro") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	glog.V(4).Info(log("mounting device  %s -> %s", devicePath, dir)) | 	glog.V(4).Info(log("mounting device  %s -> %s", devicePath, dir)) | ||||||
| @@ -140,7 +158,12 @@ func (v *sioVolume) SetUpAt(dir string, fsGroup *int64) error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	glog.V(4).Info(log("successfully setup volume %s attached %s:%s as %s", v.volSpecName, v.volName, devicePath, dir)) | 	if !v.readOnly && fsGroup != nil { | ||||||
|  | 		glog.V(4).Info(log("applying  value FSGroup ownership")) | ||||||
|  | 		volume.SetVolumeOwnership(v, fsGroup) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	glog.V(4).Info(log("successfully setup PV %s: volume %s mapped as %s mounted at %s", v.volSpecName, v.volName, devicePath, dir)) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -191,7 +214,7 @@ func (v *sioVolume) TearDownAt(dir string) error { | |||||||
| 	// use "last attempt wins" strategy to detach volume from node | 	// use "last attempt wins" strategy to detach volume from node | ||||||
| 	// only allow volume to detach when it is not busy (not being used by other pods) | 	// only allow volume to detach when it is not busy (not being used by other pods) | ||||||
| 	if !deviceBusy { | 	if !deviceBusy { | ||||||
| 		glog.V(4).Info(log("teardown is attempting to detach/unmap volume for %s", v.volSpecName)) | 		glog.V(4).Info(log("teardown is attempting to detach/unmap volume for PV %s", v.volSpecName)) | ||||||
| 		if err := v.resetSioMgr(); err != nil { | 		if err := v.resetSioMgr(); err != nil { | ||||||
| 			glog.Error(log("teardown failed, unable to reset scalio mgr: %v", err)) | 			glog.Error(log("teardown failed, unable to reset scalio mgr: %v", err)) | ||||||
| 		} | 		} | ||||||
| @@ -224,7 +247,7 @@ func (v *sioVolume) Delete() error { | |||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	glog.V(4).Info(log("successfully deleted pvc %s", v.volSpecName)) | 	glog.V(4).Info(log("successfully deleted PV %s with volume %s", v.volSpecName, v.volName)) | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -234,17 +257,30 @@ func (v *sioVolume) Delete() error { | |||||||
| var _ volume.Provisioner = &sioVolume{} | var _ volume.Provisioner = &sioVolume{} | ||||||
|  |  | ||||||
| func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | ||||||
| 	glog.V(4).Info(log("attempting to dynamically provision pvc %v", v.options.PVName)) | 	glog.V(4).Info(log("attempting to dynamically provision pvc %v", v.options.PVC.Name)) | ||||||
|  |  | ||||||
| 	if !volume.AccessModesContainedInAll(v.plugin.GetAccessModes(), v.options.PVC.Spec.AccessModes) { | 	if !volume.AccessModesContainedInAll(v.plugin.GetAccessModes(), v.options.PVC.Spec.AccessModes) { | ||||||
| 		return nil, fmt.Errorf("invalid AccessModes %v: only AccessModes %v are supported", v.options.PVC.Spec.AccessModes, v.plugin.GetAccessModes()) | 		return nil, fmt.Errorf("invalid AccessModes %v: only AccessModes %v are supported", v.options.PVC.Spec.AccessModes, v.plugin.GetAccessModes()) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// setup volume attrributes | 	// setup volume attrributes | ||||||
| 	name := v.generateVolName() | 	genName := v.generateName("k8svol", 11) | ||||||
|  | 	var oneGig int64 = 1024 * 1024 * 1024 | ||||||
|  | 	var eightGig int64 = 8 * oneGig | ||||||
|  |  | ||||||
| 	capacity := v.options.PVC.Spec.Resources.Requests[api.ResourceName(api.ResourceStorage)] | 	capacity := v.options.PVC.Spec.Resources.Requests[api.ResourceName(api.ResourceStorage)] | ||||||
| 	volSizeBytes := capacity.Value() | 	volSizeBytes := capacity.Value() | ||||||
| 	volSizeGB := int64(volume.RoundUpSize(volSizeBytes, 1024*1024*1024)) | 	volSizeGB := int64(volume.RoundUpSize(volSizeBytes, oneGig)) | ||||||
|  |  | ||||||
|  | 	if volSizeBytes == 0 { | ||||||
|  | 		return nil, fmt.Errorf("invalid volume size of 0 specified") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if volSizeBytes < eightGig { | ||||||
|  | 		volSizeGB = int64(volume.RoundUpSize(eightGig, oneGig)) | ||||||
|  | 		glog.V(4).Info(log("capacity less than 8Gi found, adjusted to %dGi", volSizeGB)) | ||||||
|  |  | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// create sio manager | 	// create sio manager | ||||||
| 	if err := v.setSioMgrFromConfig(); err != nil { | 	if err := v.setSioMgrFromConfig(); err != nil { | ||||||
| @@ -253,14 +289,15 @@ func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// create volume | 	// create volume | ||||||
| 	vol, err := v.sioMgr.CreateVolume(name, volSizeGB) | 	volName := genName | ||||||
|  | 	vol, err := v.sioMgr.CreateVolume(volName, volSizeGB) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		glog.Error(log("provision failed while creating volume: %v", err)) | 		glog.Error(log("provision failed while creating volume: %v", err)) | ||||||
| 		return nil, err | 		return nil, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// prepare data for pv | 	// prepare data for pv | ||||||
| 	v.configData[confKey.volumeName] = name | 	v.configData[confKey.volumeName] = volName | ||||||
| 	sslEnabled, err := strconv.ParseBool(v.configData[confKey.sslEnabled]) | 	sslEnabled, err := strconv.ParseBool(v.configData[confKey.sslEnabled]) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		glog.Warning(log("failed to parse parameter sslEnabled, setting to false")) | 		glog.Warning(log("failed to parse parameter sslEnabled, setting to false")) | ||||||
| @@ -273,9 +310,10 @@ func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// describe created pv | 	// describe created pv | ||||||
|  | 	pvName := genName | ||||||
| 	pv := &api.PersistentVolume{ | 	pv := &api.PersistentVolume{ | ||||||
| 		ObjectMeta: meta.ObjectMeta{ | 		ObjectMeta: meta.ObjectMeta{ | ||||||
| 			Name:      v.options.PVName, | 			Name:      pvName, | ||||||
| 			Namespace: v.options.PVC.Namespace, | 			Namespace: v.options.PVC.Namespace, | ||||||
| 			Labels:    map[string]string{}, | 			Labels:    map[string]string{}, | ||||||
| 			Annotations: map[string]string{ | 			Annotations: map[string]string{ | ||||||
| @@ -299,7 +337,7 @@ func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | |||||||
| 					ProtectionDomain: v.configData[confKey.protectionDomain], | 					ProtectionDomain: v.configData[confKey.protectionDomain], | ||||||
| 					StoragePool:      v.configData[confKey.storagePool], | 					StoragePool:      v.configData[confKey.storagePool], | ||||||
| 					StorageMode:      v.configData[confKey.storageMode], | 					StorageMode:      v.configData[confKey.storageMode], | ||||||
| 					VolumeName:       name, | 					VolumeName:       volName, | ||||||
| 					FSType:           v.configData[confKey.fsType], | 					FSType:           v.configData[confKey.fsType], | ||||||
| 					ReadOnly:         readOnly, | 					ReadOnly:         readOnly, | ||||||
| 				}, | 				}, | ||||||
| @@ -310,14 +348,14 @@ func (v *sioVolume) Provision() (*api.PersistentVolume, error) { | |||||||
| 		pv.Spec.AccessModes = v.plugin.GetAccessModes() | 		pv.Spec.AccessModes = v.plugin.GetAccessModes() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	glog.V(4).Info(log("provisioner dynamically created pvc %v with volume %s successfully", pv.Name, vol.Name)) | 	glog.V(4).Info(log("provisioner created pv %v and volume %s successfully", pvName, vol.Name)) | ||||||
| 	return pv, nil | 	return pv, nil | ||||||
| } | } | ||||||
|  |  | ||||||
| // setSioMgr creates scaleio mgr from cached config data if found | // setSioMgr creates scaleio mgr from cached config data if found | ||||||
| // otherwise, setups new config data and create mgr | // otherwise, setups new config data and create mgr | ||||||
| func (v *sioVolume) setSioMgr() error { | func (v *sioVolume) setSioMgr() error { | ||||||
| 	glog.V(4).Info(log("setting up sio mgr for vol  %s", v.volSpecName)) | 	glog.V(4).Info(log("setting up sio mgr for spec  %s", v.volSpecName)) | ||||||
| 	podDir := v.plugin.host.GetPodPluginDir(v.podUID, sioPluginName) | 	podDir := v.plugin.host.GetPodPluginDir(v.podUID, sioPluginName) | ||||||
| 	configName := path.Join(podDir, sioConfigFileName) | 	configName := path.Join(podDir, sioConfigFileName) | ||||||
| 	if v.sioMgr == nil { | 	if v.sioMgr == nil { | ||||||
| @@ -455,6 +493,6 @@ func (v *sioVolume) setSioMgrFromSpec() error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (v *sioVolume) generateVolName() string { | func (v *sioVolume) generateName(prefix string, size int) string { | ||||||
| 	return "sio-" + strings.Replace(string(uuid.NewUUID()), "-", "", -1)[0:25] | 	return fmt.Sprintf("%s-%s", prefix, strings.Replace(string(uuid.NewUUID()), "-", "", -1)[0:size]) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ import ( | |||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
|  |  | ||||||
|  | 	"github.com/golang/glog" | ||||||
|  |  | ||||||
| 	api "k8s.io/api/core/v1" | 	api "k8s.io/api/core/v1" | ||||||
| 	meta "k8s.io/apimachinery/pkg/apis/meta/v1" | 	meta "k8s.io/apimachinery/pkg/apis/meta/v1" | ||||||
| 	"k8s.io/apimachinery/pkg/types" | 	"k8s.io/apimachinery/pkg/types" | ||||||
| @@ -149,6 +151,7 @@ func TestVolumeMounterUnmounter(t *testing.T) { | |||||||
| 				VolumeName:       testSioVol, | 				VolumeName:       testSioVol, | ||||||
| 				FSType:           "ext4", | 				FSType:           "ext4", | ||||||
| 				SecretRef:        &api.LocalObjectReference{Name: "sio-secret"}, | 				SecretRef:        &api.LocalObjectReference{Name: "sio-secret"}, | ||||||
|  | 				ReadOnly:         false, | ||||||
| 			}, | 			}, | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| @@ -191,6 +194,10 @@ func TestVolumeMounterUnmounter(t *testing.T) { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	if sio.isMultiMap { | ||||||
|  | 		t.Errorf("SetUp() - expecting multiple volume disabled by default") | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// rebuild spec | 	// rebuild spec | ||||||
| 	builtSpec, err := sioPlug.ConstructVolumeSpec(volume.NewSpecFromVolume(vol).Name(), path) | 	builtSpec, err := sioPlug.ConstructVolumeSpec(volume.NewSpecFromVolume(vol).Name(), path) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -235,25 +242,23 @@ func TestVolumeProvisioner(t *testing.T) { | |||||||
|  |  | ||||||
| 	plug, err := plugMgr.FindPluginByName(sioPluginName) | 	plug, err := plugMgr.FindPluginByName(sioPluginName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Errorf("Can't find the plugin %v", sioPluginName) | 		t.Fatalf("Can't find the plugin %v", sioPluginName) | ||||||
| 	} | 	} | ||||||
| 	sioPlug, ok := plug.(*sioPlugin) | 	sioPlug, ok := plug.(*sioPlugin) | ||||||
| 	if !ok { | 	if !ok { | ||||||
| 		t.Errorf("Cannot assert plugin to be type sioPlugin") | 		t.Fatal("Cannot assert plugin to be type sioPlugin") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	options := volume.VolumeOptions{ | 	options := volume.VolumeOptions{ | ||||||
| 		ClusterName: "testcluster", | 		ClusterName: "testcluster", | ||||||
| 		PVName:      "pvc-sio-dynamic-vol", |  | ||||||
| 		PVC:         volumetest.CreateTestPVC("100Mi", []api.PersistentVolumeAccessMode{api.ReadWriteOnce}), | 		PVC:         volumetest.CreateTestPVC("100Mi", []api.PersistentVolumeAccessMode{api.ReadWriteOnce}), | ||||||
| 		PersistentVolumeReclaimPolicy: api.PersistentVolumeReclaimDelete, | 		PersistentVolumeReclaimPolicy: api.PersistentVolumeReclaimDelete, | ||||||
| 	} | 	} | ||||||
|  | 	options.PVC.Name = "testpvc" | ||||||
| 	options.PVC.Namespace = testns | 	options.PVC.Namespace = testns | ||||||
|  |  | ||||||
| 	// incomplete options, test should fail | 	options.PVC.Spec.AccessModes = []api.PersistentVolumeAccessMode{ | ||||||
| 	_, err = sioPlug.NewProvisioner(options) | 		api.ReadOnlyMany, | ||||||
| 	if err == nil { |  | ||||||
| 		t.Fatal("expected failure due to incomplete options") |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	options.Parameters = map[string]string{ | 	options.Parameters = map[string]string{ | ||||||
| @@ -288,10 +293,9 @@ func TestVolumeProvisioner(t *testing.T) { | |||||||
| 	// validate provision | 	// validate provision | ||||||
| 	actualSpecName := spec.Name | 	actualSpecName := spec.Name | ||||||
| 	actualVolName := spec.Spec.PersistentVolumeSource.ScaleIO.VolumeName | 	actualVolName := spec.Spec.PersistentVolumeSource.ScaleIO.VolumeName | ||||||
| 	if !strings.HasPrefix(actualSpecName, "pvc-") { | 	if !strings.HasPrefix(actualSpecName, "k8svol-") { | ||||||
| 		t.Errorf("expecting volume name to start with pov-, got %s", actualSpecName) | 		t.Errorf("expecting volume name to start with k8svol-, got %s", actualSpecName) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	vol, err := sio.FindVolume(actualVolName) | 	vol, err := sio.FindVolume(actualVolName) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		t.Fatalf("failed getting volume %v: %v", actualVolName, err) | 		t.Fatalf("failed getting volume %v: %v", actualVolName, err) | ||||||
| @@ -299,6 +303,9 @@ func TestVolumeProvisioner(t *testing.T) { | |||||||
| 	if vol.Name != actualVolName { | 	if vol.Name != actualVolName { | ||||||
| 		t.Errorf("expected volume name to be %s, got %s", actualVolName, vol.Name) | 		t.Errorf("expected volume name to be %s, got %s", actualVolName, vol.Name) | ||||||
| 	} | 	} | ||||||
|  | 	if vol.SizeInKb != 8*1024*1024 { | ||||||
|  | 		glog.V(4).Info(log("unexpected volume size")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// mount dynamic vol | 	// mount dynamic vol | ||||||
| 	sioMounter, err := sioPlug.NewMounter( | 	sioMounter, err := sioPlug.NewMounter( | ||||||
| @@ -315,8 +322,14 @@ func TestVolumeProvisioner(t *testing.T) { | |||||||
| 	} | 	} | ||||||
| 	sioVol.sioMgr.client = sio | 	sioVol.sioMgr.client = sio | ||||||
| 	if err := sioMounter.SetUp(nil); err != nil { | 	if err := sioMounter.SetUp(nil); err != nil { | ||||||
| 		t.Errorf("Expected success, got: %v", err) | 		t.Fatalf("Expected success, got: %v", err) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// isMultiMap applied | ||||||
|  | 	if !sio.isMultiMap { | ||||||
|  | 		t.Errorf("SetUp()  expecting attached volume with multi-mapping") | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	// teardown dynamic vol | 	// teardown dynamic vol | ||||||
| 	sioUnmounter, err := sioPlug.NewUnmounter(spec.Name, podUID) | 	sioUnmounter, err := sioPlug.NewUnmounter(spec.Name, podUID) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| @@ -351,3 +364,83 @@ func TestVolumeProvisioner(t *testing.T) { | |||||||
| 		t.Errorf("Deleter did not delete path %v: %v", path, err) | 		t.Errorf("Deleter did not delete path %v: %v", path, err) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func TestVolumeProvisionerWithIncompleteConfig(t *testing.T) { | ||||||
|  | 	plugMgr, tmpDir := newPluginMgr(t) | ||||||
|  | 	defer os.RemoveAll(tmpDir) | ||||||
|  |  | ||||||
|  | 	plug, err := plugMgr.FindPluginByName(sioPluginName) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Can't find the plugin %v", sioPluginName) | ||||||
|  | 	} | ||||||
|  | 	sioPlug, ok := plug.(*sioPlugin) | ||||||
|  | 	if !ok { | ||||||
|  | 		t.Fatal("Cannot assert plugin to be type sioPlugin") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	options := volume.VolumeOptions{ | ||||||
|  | 		ClusterName: "testcluster", | ||||||
|  | 		PVName:      "pvc-sio-dynamic-vol", | ||||||
|  | 		PVC:         volumetest.CreateTestPVC("100Mi", []api.PersistentVolumeAccessMode{api.ReadWriteOnce}), | ||||||
|  | 		PersistentVolumeReclaimPolicy: api.PersistentVolumeReclaimDelete, | ||||||
|  | 	} | ||||||
|  | 	options.PVC.Namespace = testns | ||||||
|  |  | ||||||
|  | 	options.PVC.Spec.AccessModes = []api.PersistentVolumeAccessMode{ | ||||||
|  | 		api.ReadWriteOnce, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// incomplete options, test should fail | ||||||
|  | 	_, err = sioPlug.NewProvisioner(options) | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatal("expected failure due to incomplete options") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func TestVolumeProvisionerWithZeroCapacity(t *testing.T) { | ||||||
|  | 	plugMgr, tmpDir := newPluginMgr(t) | ||||||
|  | 	defer os.RemoveAll(tmpDir) | ||||||
|  |  | ||||||
|  | 	plug, err := plugMgr.FindPluginByName(sioPluginName) | ||||||
|  | 	if err != nil { | ||||||
|  | 		t.Fatalf("Can't find the plugin %v", sioPluginName) | ||||||
|  | 	} | ||||||
|  | 	sioPlug, ok := plug.(*sioPlugin) | ||||||
|  | 	if !ok { | ||||||
|  | 		t.Fatal("Cannot assert plugin to be type sioPlugin") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	options := volume.VolumeOptions{ | ||||||
|  | 		ClusterName: "testcluster", | ||||||
|  | 		PVName:      "pvc-sio-dynamic-vol", | ||||||
|  | 		PVC:         volumetest.CreateTestPVC("0Mi", []api.PersistentVolumeAccessMode{api.ReadWriteOnce}), | ||||||
|  | 		PersistentVolumeReclaimPolicy: api.PersistentVolumeReclaimDelete, | ||||||
|  | 	} | ||||||
|  | 	options.PVC.Namespace = testns | ||||||
|  |  | ||||||
|  | 	options.PVC.Spec.AccessModes = []api.PersistentVolumeAccessMode{ | ||||||
|  | 		api.ReadWriteOnce, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	options.Parameters = map[string]string{ | ||||||
|  | 		confKey.gateway:          "http://test.scaleio:11111", | ||||||
|  | 		confKey.system:           "sio", | ||||||
|  | 		confKey.protectionDomain: testSioPD, | ||||||
|  | 		confKey.storagePool:      "default", | ||||||
|  | 		confKey.secretRef:        "sio-secret", | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	provisioner, _ := sioPlug.NewProvisioner(options) | ||||||
|  | 	sio := newFakeSio() | ||||||
|  | 	sioVol := provisioner.(*sioVolume) | ||||||
|  | 	if err := sioVol.setSioMgrFromConfig(); err != nil { | ||||||
|  | 		t.Fatalf("failed to create scaleio mgr from config: %v", err) | ||||||
|  | 	} | ||||||
|  | 	sioVol.sioMgr.client = sio | ||||||
|  |  | ||||||
|  | 	_, err = provisioner.Provision() | ||||||
|  | 	if err == nil { | ||||||
|  | 		t.Fatalf("call to Provision() should fail with invalid capacity") | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Vladimir Vivien
					Vladimir Vivien