PowerMax Driver - Deletion of group with volumes

1.  Add a clone check to make sure their is no snapVx session.
2.  Allow for bootable volumes to be added to groups when
    image_volume_cache_enabled=True.
3.  Cleanup, if a volume does not exist in an storage group, then
    don't try to remove it from the storage group.

Change-Id: Ic4fdd42ce44ffe50a423cc2a92641d5ce0ffdf28
Closes-Bug: #1863241
This commit is contained in:
Helen Walsh 2020-02-14 12:13:47 +00:00
parent 49b7e303c0
commit e5e6bc6866
2 changed files with 121 additions and 21 deletions

View File

@ -2105,6 +2105,25 @@ class PowerMaxCommonTest(test.TestCase):
self.common.update_group,
self.data.test_group_1, [], [])
@mock.patch.object(volume_utils, 'is_group_a_type',
return_value=False)
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
return_value=True)
def test_update_group_remove_volumes(self, mock_cg_type, mock_type_check):
group = self.data.test_group_1
add_vols = []
remove_vols = [self.data.test_volume_group_member]
ref_model_update = {'status': fields.GroupStatus.AVAILABLE}
with mock.patch.object(
rest.PowerMaxRest, 'is_volume_in_storagegroup',
return_value=False) as mock_exists:
model_update, __, __ = self.common.update_group(group,
add_vols,
remove_vols)
mock_exists.assert_called_once()
self.assertEqual(ref_model_update, model_update)
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
def test_delete_group(self, mock_check):
group = self.data.test_group_1
@ -2157,6 +2176,28 @@ class PowerMaxCommonTest(test.TestCase):
group, volumes)
self.assertEqual(ref_model_update, model_update)
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
return_value=True)
@mock.patch.object(rest.PowerMaxRest, 'get_volumes_in_storage_group',
return_value=[
tpd.PowerMaxData.test_volume_group_member])
@mock.patch.object(common.PowerMaxCommon, '_get_members_of_volume_group',
return_value=[tpd.PowerMaxData.device_id])
@mock.patch.object(common.PowerMaxCommon, '_find_device_on_array',
return_value= tpd.PowerMaxData.device_id)
@mock.patch.object(masking.PowerMaxMasking,
'remove_volumes_from_storage_group')
def test_delete_group_clone_check(
self, mock_rem, mock_find, mock_mems, mock_vols, mock_chk1,
mock_chk2):
group = self.data.test_group_1
volumes = [self.data.test_volume_group_member]
with mock.patch.object(
self.common, '_clone_check') as mock_clone_chk:
self.common._delete_group(group, volumes)
mock_clone_chk.assert_called_once()
@mock.patch.object(
common.PowerMaxCommon, '_remove_vol_and_cleanup_replication')
@mock.patch.object(
@ -2871,6 +2912,22 @@ class PowerMaxCommonTest(test.TestCase):
self.common._clone_check(array, device_id, extra_specs)
self.assertEqual(3, mck_del.call_count)
@mock.patch.object(
common.PowerMaxCommon, '_unlink_targets_and_delete_temp_snapvx')
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
return_value=(tpd.PowerMaxData.snap_src_sessions,
tpd.PowerMaxData.snap_tgt_session))
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
return_value=(True, True, False))
def test_clone_check_force_unlink(self, mck_rep, mck_find, mck_del):
array = self.data.array
device_id = self.data.device_id
extra_specs = self.data.extra_specs
self.common.snapvx_unlink_limit = 3
self.common._clone_check(
array, device_id, extra_specs, force_unlink=True)
self.assertEqual(3, mck_del.call_count)
@mock.patch.object(common.PowerMaxCommon,
'_unlink_targets_and_delete_temp_snapvx')
def test_delete_valid_snapshot(self, mck_unlink):

View File

@ -440,7 +440,6 @@ class PowerMaxCommon(object):
:returns: model_update - dict
"""
model_update, rep_driver_data = dict(), dict()
group_name, group_id = None, None
volume_id = volume.id
extra_specs = self._initial_setup(volume)
@ -459,14 +458,10 @@ class PowerMaxCommon(object):
rep_driver_data = rep_update['replication_driver_data']
model_update.update(rep_update)
# Add volume to group, if required
if volume.group_id is not None:
if (volume_utils.is_group_a_cg_snapshot_type(volume.group)
or volume.group.is_replicated):
group_id = volume.group_id
group_name = self._add_new_volume_to_volume_group(
volume, volume_dict['device_id'], volume_name,
extra_specs, rep_driver_data)
# Add volume to group
group_name = self._add_to_group(
volume, volume_dict['device_id'], volume_name, volume.group_id,
volume.group, extra_specs, rep_driver_data)
# Gather Metadata
model_update.update(
@ -482,7 +477,7 @@ class PowerMaxCommon(object):
extra_specs[utils.ARRAY])
self.volume_metadata.capture_create_volume(
volume_dict['device_id'], volume, group_name, group_id,
volume_dict['device_id'], volume, group_name, volume.group_id,
extra_specs, rep_info_dict, 'create',
array_tag_list=array_tag_list)
@ -491,6 +486,29 @@ class PowerMaxCommon(object):
return model_update
def _add_to_group(
self, volume, device_id, volume_name, group_id, group,
extra_specs, rep_driver_data=None):
"""Add a volume to a volume group
:param volume: volume object
:param device_id: the device id
:param volume_name: volume name
:param group_id: the group id
:param group: group object
:param extra_specs: extra specifications
:param rep_driver_data: replication data (optional)
:returns: group_id - string
"""
group_name = None
if group_id is not None:
if (volume_utils.is_group_a_cg_snapshot_type(group)
or group.is_replicated):
group_name = self._add_new_volume_to_volume_group(
volume, device_id, volume_name,
extra_specs, rep_driver_data)
return group_name
def _add_new_volume_to_volume_group(self, volume, device_id, volume_name,
extra_specs, rep_driver_data=None):
"""Add a new volume to a volume group.
@ -563,6 +581,7 @@ class PowerMaxCommon(object):
:returns: model_update, dict
"""
model_update, rep_info_dict = {}, {}
rep_driver_data = None
extra_specs = self._initial_setup(clone_volume)
array = extra_specs[utils.ARRAY]
source_device_id = self._find_device_on_array(
@ -578,8 +597,15 @@ class PowerMaxCommon(object):
clone_volume, source_volume, extra_specs)
# Update model with replication session info if applicable
if rep_update:
rep_driver_data = rep_update['replication_driver_data']
model_update.update(rep_update)
# Add volume to group
group_name = self._add_to_group(
clone_volume, clone_dict['device_id'], clone_volume.name,
source_volume.group_id, source_volume.group, extra_specs,
rep_driver_data)
model_update.update(
{'provider_location': six.text_type(clone_dict)})
model_update = self.update_metadata(
@ -590,9 +616,11 @@ class PowerMaxCommon(object):
utils.REP_CONFIG].get(utils.BACKEND_ID, 'None')
array_tag_list = self.get_tags_of_storage_array(
extra_specs[utils.ARRAY])
self.volume_metadata.capture_create_volume(
clone_dict['device_id'], clone_volume, None, None,
extra_specs, rep_info_dict, 'createFromVolume',
clone_dict['device_id'], clone_volume, group_name,
source_volume.group_id, extra_specs, rep_info_dict,
'createFromVolume',
temporary_snapvx=clone_dict.get('snap_name'),
source_device_id=clone_dict.get('source_device_id'),
array_tag_list=array_tag_list)
@ -2756,12 +2784,14 @@ class PowerMaxCommon(object):
return source_device_id
def _clone_check(self, array, device_id, extra_specs):
def _clone_check(
self, array, device_id, extra_specs, force_unlink=False):
"""Perform any snapvx cleanup before creating clones or snapshots
:param array: the array serial
:param device_id: the device ID of the volume
:param extra_specs: extra specifications
:param force_unlink: force unlink even if not expired
"""
snapvx_tgt, snapvx_src, __ = self.rest.is_vol_in_rep_session(
array, device_id)
@ -2773,32 +2803,36 @@ class PowerMaxCommon(object):
array, src_device_id)
count = 0
if tgt_session and count < self.snapvx_unlink_limit:
self._delete_valid_snapshot(array, tgt_session,
extra_specs)
self._delete_valid_snapshot(
array, tgt_session, extra_specs, force_unlink)
count += 1
if src_sessions:
src_sessions.sort(
key=lambda k: k['generation'], reverse=True)
for session in src_sessions:
if count < self.snapvx_unlink_limit:
self._delete_valid_snapshot(array, session,
extra_specs)
self._delete_valid_snapshot(
array, session, extra_specs, force_unlink)
count += 1
else:
break
do_unlink_and_delete_snap(device_id)
def _delete_valid_snapshot(self, array, session, extra_specs):
def _delete_valid_snapshot(
self, array, session, extra_specs, force_unlink=False):
"""Delete a snapshot if valid candidate for deletion.
:param array: the array serial
:param session: the snapvx session
:param extra_specs: extra specifications
:param force_unlink: force unlink even if not expired
"""
is_legacy = 'EMC_SMI' in session['snap_name']
is_temp = utils.CLONE_SNAPSHOT_NAME in session['snap_name']
is_expired = session['expired']
if force_unlink:
is_expired = True
is_valid = True if is_legacy or (is_temp and is_expired) else False
if is_valid:
try:
@ -4659,6 +4693,8 @@ class PowerMaxCommon(object):
device_id = self._find_device_on_array(
vol, extra_specs)
if device_id in volume_device_ids:
self._clone_check(
array, device_id, extra_specs, force_unlink=True)
self.masking.remove_and_reset_members(
array, vol, device_id, vol.name,
extra_specs, False)
@ -5014,6 +5050,13 @@ class PowerMaxCommon(object):
if group.is_replicated:
# Need force flag when manipulating RDF enabled SGs
interval_retries_dict[utils.FORCE_VOL_REMOVE] = True
# Check if the volumes exist in the storage group
temp_list = deepcopy(remove_device_ids)
for device_id in temp_list:
if not self.rest.is_volume_in_storagegroup(
array, device_id, vol_grp_name):
remove_device_ids.remove(device_id)
if remove_device_ids:
self.masking.remove_volumes_from_storage_group(
array, remove_device_ids,
vol_grp_name, interval_retries_dict)