cephfs: DeleteVolume() calls are allowed only for volumes with provisionVolume=true parameter

This commit is contained in:
gman 2018-06-13 16:23:13 +02:00
parent f5af7d0a94
commit 675ee93e46
3 changed files with 43 additions and 90 deletions

View File

@ -18,7 +18,6 @@ package cephfs
import ( import (
"fmt" "fmt"
"os"
"github.com/golang/glog" "github.com/golang/glog"
"golang.org/x/net/context" "golang.org/x/net/context"
@ -39,7 +38,7 @@ const (
func (cs *controllerServer) validateCreateVolumeRequest(req *csi.CreateVolumeRequest) error { func (cs *controllerServer) validateCreateVolumeRequest(req *csi.CreateVolumeRequest) error {
if err := cs.Driver.ValidateControllerServiceRequest(csi.ControllerServiceCapability_RPC_CREATE_DELETE_VOLUME); err != nil { if err := cs.Driver.ValidateControllerServiceRequest(csi.ControllerServiceCapability_RPC_CREATE_DELETE_VOLUME); err != nil {
return fmt.Errorf("Invalid CreateVolumeRequest: %v", err) return fmt.Errorf("invalid CreateVolumeRequest: %v", err)
} }
if req.GetName() == "" { if req.GetName() == "" {
@ -55,7 +54,7 @@ func (cs *controllerServer) validateCreateVolumeRequest(req *csi.CreateVolumeReq
func (cs *controllerServer) validateDeleteVolumeRequest(req *csi.DeleteVolumeRequest) error { func (cs *controllerServer) validateDeleteVolumeRequest(req *csi.DeleteVolumeRequest) error {
if err := cs.Driver.ValidateControllerServiceRequest(csi.ControllerServiceCapability_RPC_CREATE_DELETE_VOLUME); err != nil { if err := cs.Driver.ValidateControllerServiceRequest(csi.ControllerServiceCapability_RPC_CREATE_DELETE_VOLUME); err != nil {
return fmt.Errorf("Invalid DeleteVolumeRequest: %v", err) return fmt.Errorf("invalid DeleteVolumeRequest: %v", err)
} }
return nil return nil
@ -112,10 +111,15 @@ func (cs *controllerServer) CreateVolume(ctx context.Context, req *csi.CreateVol
glog.Warningf("failed to store a volume cache entry: %v", err) glog.Warningf("failed to store a volume cache entry: %v", err)
} }
sz := req.GetCapacityRange().GetRequiredBytes()
if sz == 0 {
sz = oneGB
}
return &csi.CreateVolumeResponse{ return &csi.CreateVolumeResponse{
Volume: &csi.Volume{ Volume: &csi.Volume{
Id: volId.id, Id: volId.id,
CapacityBytes: req.GetCapacityRange().GetRequiredBytes(), CapacityBytes: sz,
Attributes: req.GetParameters(), Attributes: req.GetParameters(),
}, },
}, nil }, nil
@ -128,8 +132,6 @@ func (cs *controllerServer) DeleteVolume(ctx context.Context, req *csi.DeleteVol
} }
var ( var (
cr *credentials
err error
volId = req.GetVolumeId() volId = req.GetVolumeId()
volUuid = uuidFromVolumeId(volId) volUuid = uuidFromVolumeId(volId)
) )
@ -143,46 +145,35 @@ func (cs *controllerServer) DeleteVolume(ctx context.Context, req *csi.DeleteVol
return nil, status.Error(codes.Internal, msg) return nil, status.Error(codes.Internal, msg)
} }
// Set the correct user for mounting if !ent.VolOptions.ProvisionVolume {
// DeleteVolume() is forbidden for statically provisioned volumes!
if ent.VolOptions.ProvisionVolume { msg := fmt.Sprintf("volume %s is provisioned statically, aborting delete", volId)
// Admin access is required glog.Warningf(msg)
return &csi.DeleteVolumeResponse{}, nil
cr, err = getAdminCredentials(req.GetControllerDeleteSecrets())
if err != nil {
return nil, status.Error(codes.InvalidArgument, err.Error())
}
} else {
cr, err = getUserCredentials(req.GetControllerDeleteSecrets())
if err != nil {
return nil, status.Error(codes.InvalidArgument, err.Error())
}
} }
// Delete the volume contents // Requires admin credentials
if err := purgeVolume(volId, cr, &ent.VolOptions); err != nil { cr, err := getAdminCredentials(req.GetControllerDeleteSecrets())
glog.Error(err) if err != nil {
glog.Errorf("failed to retrieve admin credentials: %v", err)
return nil, status.Error(codes.InvalidArgument, err.Error())
}
// Remove the volume, the user and the volume cache entry
if err = purgeVolume(volId, cr, &ent.VolOptions); err != nil {
glog.Errorf("failed to delete volume %s: %v", volId, err)
return nil, status.Error(codes.Internal, err.Error()) return nil, status.Error(codes.Internal, err.Error())
} }
// Clean up remaining files if err = deleteCephUser(cr, volUuid); err != nil {
glog.Errorf("failed to delete ceph user %s: %v", getCephUserName(volUuid), err)
if ent.VolOptions.ProvisionVolume { return nil, status.Error(codes.Internal, err.Error())
// The user is no longer needed
if err := deleteCephUser(cr, volUuid); err != nil {
glog.Warningf("failed to delete ceph user '%s': %v", cr.id, err)
}
userId := getCephUserName(volUuid)
os.Remove(getCephKeyringPath(volUuid, userId))
os.Remove(getCephSecretPath(volUuid, userId))
} else {
os.Remove(getCephKeyringPath(volUuid, cr.id))
os.Remove(getCephSecretPath(volUuid, cr.id))
} }
if err := volCache.erase(volUuid); err != nil { if err = volCache.erase(volUuid); err != nil {
glog.Errorf("failed to delete cache entry for volume %s: %v", volId, err)
return nil, status.Error(codes.Internal, err.Error()) return nil, status.Error(codes.Internal, err.Error())
} }

View File

@ -31,9 +31,6 @@ const (
cephVolumePrefix = PluginFolder + "/controller/volumes/vol-" cephVolumePrefix = PluginFolder + "/controller/volumes/vol-"
cephVolumesRoot = "csi-volumes" cephVolumesRoot = "csi-volumes"
volumeDataSuffix = "volume-data"
volumeDeletingSuffix = "volume-deleting"
namespacePrefix = "csi-ns-" namespacePrefix = "csi-ns-"
) )
@ -45,14 +42,6 @@ func getCephRootVolumePath_local(volUuid string) string {
return path.Join(getCephRootPath_local(volUuid), cephVolumesRoot, volUuid) return path.Join(getCephRootPath_local(volUuid), cephVolumesRoot, volUuid)
} }
func getCephRootVolumeDataPath_local(volUuid string) string {
return path.Join(getCephRootVolumePath_local(volUuid), volumeDataSuffix)
}
func getCephRootVolumeDeletingPath_local(volUuid string) string {
return path.Join(getCephRootVolumePath_local(volUuid), volumeDeletingSuffix)
}
func getVolumeRootPath_local(volUuid string) string { func getVolumeRootPath_local(volUuid string) string {
return cephVolumePrefix + volUuid return cephVolumePrefix + volUuid
} }
@ -61,14 +50,6 @@ func getVolumeRootPath_ceph(volUuid string) string {
return path.Join("/", cephVolumesRoot, volUuid) return path.Join("/", cephVolumesRoot, volUuid)
} }
func getVolumeDataPath_local(volUuid string) string {
return path.Join(getVolumeRootPath_local(volUuid), volumeDataSuffix)
}
func getVolumeDeletingPath_local(volUuid string) string {
return path.Join(getVolumeRootPath_local(volUuid), volumeDeletingSuffix)
}
func getVolumeNamespace(volUuid string) string { func getVolumeNamespace(volUuid string) string {
return namespacePrefix + volUuid return namespacePrefix + volUuid
} }
@ -120,54 +101,35 @@ func createVolume(volOptions *volumeOptions, adminCr *credentials, volUuid strin
} }
func purgeVolume(volId string, cr *credentials, volOptions *volumeOptions) error { func purgeVolume(volId string, cr *credentials, volOptions *volumeOptions) error {
// Root path is not set for dynamically provisioned volumes
volOptions.RootPath = "/"
var ( var (
volUuid = uuidFromVolumeId(volId) volUuid = uuidFromVolumeId(volId)
volRoot string root = getCephRootPath_local(volUuid)
dataPath string volRoot = getCephRootVolumePath_local(volUuid)
delPath string volRootDeleting = volRoot + "-deleting"
) )
if volOptions.ProvisionVolume { if err := createMountPoint(root); err != nil {
// RootPath is not set for a dynamically provisioned volume
volOptions.RootPath = "/"
volRoot = getCephRootPath_local(volUuid)
dataPath = getCephRootVolumeDataPath_local(volUuid)
delPath = getCephRootVolumeDeletingPath_local(volUuid)
} else {
volRoot = getVolumeRootPath_local(volUuid)
dataPath = getVolumeDataPath_local(volUuid)
delPath = getVolumeDeletingPath_local(volUuid)
}
if err := createMountPoint(volRoot); err != nil {
return err return err
} }
if err := mountKernel(volRoot, cr, volOptions, volUuid); err != nil { if err := mountKernel(root, cr, volOptions, volUuid); err != nil {
return err return err
} }
defer func() { defer func() {
if volOptions.ProvisionVolume {
os.Remove(getCephRootVolumePath_local(volUuid))
}
unmountVolume(volRoot) unmountVolume(volRoot)
os.Remove(volRoot) os.Remove(volRoot)
}() }()
if err := os.Rename(dataPath, delPath); err != nil { if err := os.Rename(volRoot, volRootDeleting); err != nil {
if os.IsNotExist(err) { return fmt.Errorf("coudln't mark volume %s for deletion: %v", volId, err)
// dataPath doesn't exist if NodePublishVolume wasn't called
return nil
} else {
return fmt.Errorf("couldn't mark volume %s for deletion: %v", volId, err)
}
} }
if err := os.RemoveAll(delPath); err != nil { if err := os.RemoveAll(volRootDeleting); err != nil {
return fmt.Errorf("couldn't delete volume %s: %v", volId, err) return fmt.Errorf("failed to delete volume %s: %v", volId, err)
} }
return nil return nil

View File

@ -121,7 +121,7 @@ func bindMount(from, to string, readOnly bool) error {
} }
func bindVolume(volUuid, target string, readOnly bool) error { func bindVolume(volUuid, target string, readOnly bool) error {
volDataRoot := getVolumeDataPath_local(volUuid) volDataRoot := getVolumeRootPath_local(volUuid)
if err := createMountPoint(volDataRoot); err != nil { if err := createMountPoint(volDataRoot); err != nil {
return err return err