From 9eef8e1df016d3467e2a0ccd8ec051ebada0dfe0 Mon Sep 17 00:00:00 2001 From: sinhaashish Date: Fri, 20 Dec 2024 08:50:18 +0000 Subject: [PATCH] fix(controller): prevent zfs volume cr deletion if snapshot exists Signed-off-by: sinhaashish --- pkg/driver/controller.go | 31 +++++++++++++++++++++++-------- pkg/zfs/volume.go | 9 +++++++++ 2 files changed, 32 insertions(+), 8 deletions(-) diff --git a/pkg/driver/controller.go b/pkg/driver/controller.go index e1ef8fdf..f48fb615 100644 --- a/pkg/driver/controller.go +++ b/pkg/driver/controller.go @@ -502,6 +502,18 @@ func (cs *controller) DeleteVolume( unlock := cs.volumeLock.LockVolume(volumeID) defer unlock() + // Fetch the list of snapshot for the given volume + snapList, err := zfs.GetSnapshotForVolume(volumeID) + if err != nil { + return nil, status.Errorf( + codes.NotFound, + "failed to handle delete volume request for {%s}, "+ + "validation failed checking for snapshots. Error: %s", + req.VolumeId, + err.Error(), + ) + } + // verify if the volume has already been deleted vol, err := zfs.GetVolume(volumeID) if vol != nil && vol.DeletionTimestamp != nil { @@ -524,14 +536,17 @@ func (cs *controller) DeleteVolume( return nil, status.Error(codes.Internal, "can not delete, volume creation is in progress") } - // Delete the corresponding ZV CR - err = zfs.DeleteVolume(volumeID) - if err != nil { - return nil, errors.Wrapf( - err, - "failed to handle delete volume request for {%s}", - volumeID, - ) + // Delete the corresponding ZV CR only if there are any snapshots present for the volume + + if len(snapList.Items) == 0 { + err = zfs.DeleteVolume(volumeID) + if err != nil { + return nil, errors.Wrapf( + err, + "failed to handle delete volume request for {%s}", + volumeID, + ) + } } sendEventOrIgnore("", volumeID, vol.Spec.Capacity, analytics.VolumeDeprovision) diff --git a/pkg/zfs/volume.go b/pkg/zfs/volume.go index 44a75427..7af796dc 100644 --- a/pkg/zfs/volume.go +++ b/pkg/zfs/volume.go @@ -437,3 +437,12 @@ func IsVolumeReady(vol *apis.ZFSVolume) bool { return false } + +// GetSnapshotForVolume fetches all the snapshots for the given volume +func GetSnapshotForVolume(volumeID string) (*apis.ZFSSnapshotList, error) { + listOptions := metav1.ListOptions{ + LabelSelector: ZFSVolKey + "=" + volumeID, + } + snapList, err := snapbuilder.NewKubeclient().WithNamespace(OpenEBSNamespace).List(listOptions) + return snapList, err +}