PowerMax Driver - Temporary snapshot enhancements
Improvements to unlinking of volume snapshots and deletion of temporary snapshots created during volume cloning operations. Including: 1. Merging of sync and clone check functionality. 2. Removal of snapvx unlink limit use during snapshot unlinking. 3. Adding early exit to volume delete, group delete and retype in cases where volume snapshots still exist after attempting unlink and deletion of snapvx sessions. Change-Id: I60f399eccb32daf5127c64108d5331cf798b2600
This commit is contained in:
parent
47cdd944b3
commit
1a267bf6ff
@ -224,10 +224,10 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
model_update = self.common.create_volume(self.data.test_volume)
|
model_update = self.common.create_volume(self.data.test_volume)
|
||||||
self.assertEqual(ref_model_update, model_update)
|
self.assertEqual(ref_model_update, model_update)
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
||||||
return_value='')
|
return_value='')
|
||||||
def test_create_volume_from_snapshot(self, mck_meta, mck_clone_chk):
|
def test_create_volume_from_snapshot(self, mck_meta, mck_cleanup_snaps):
|
||||||
ref_model_update = ({'provider_location': six.text_type(
|
ref_model_update = ({'provider_location': six.text_type(
|
||||||
deepcopy(self.data.provider_location_snapshot))})
|
deepcopy(self.data.provider_location_snapshot))})
|
||||||
model_update = self.common.create_volume_from_snapshot(
|
model_update = self.common.create_volume_from_snapshot(
|
||||||
@ -362,10 +362,14 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
mck_configure.assert_called_once()
|
mck_configure.assert_called_once()
|
||||||
mck_resume.assert_called_once()
|
mck_resume.assert_called_once()
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
||||||
return_value='')
|
return_value='')
|
||||||
def test_cloned_volume(self, mck_meta, mck_clone_chk):
|
def test_cloned_volume(self, mck_meta, mck_cleanup_snaps):
|
||||||
|
array = self.data.array
|
||||||
|
test_volume = self.data.test_clone_volume
|
||||||
|
source_device_id = self.data.device_id
|
||||||
|
extra_specs = self.common._initial_setup(test_volume)
|
||||||
ref_model_update = ({'provider_location': six.text_type(
|
ref_model_update = ({'provider_location': six.text_type(
|
||||||
self.data.provider_location_clone)})
|
self.data.provider_location_clone)})
|
||||||
model_update = self.common.create_cloned_volume(
|
model_update = self.common.create_cloned_volume(
|
||||||
@ -373,18 +377,55 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
ast.literal_eval(ref_model_update['provider_location']),
|
ast.literal_eval(ref_model_update['provider_location']),
|
||||||
ast.literal_eval(model_update['provider_location']))
|
ast.literal_eval(model_update['provider_location']))
|
||||||
|
mck_cleanup_snaps.assert_called_once_with(
|
||||||
|
array, source_device_id, extra_specs)
|
||||||
|
|
||||||
def test_delete_volume(self):
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
|
def test_delete_volume(self, mck_get_snaps):
|
||||||
with mock.patch.object(self.common, '_delete_volume') as mock_delete:
|
with mock.patch.object(self.common, '_delete_volume') as mock_delete:
|
||||||
self.common.delete_volume(self.data.test_volume)
|
self.common.delete_volume(self.data.test_volume)
|
||||||
mock_delete.assert_called_once_with(self.data.test_volume)
|
mock_delete.assert_called_once_with(self.data.test_volume)
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_delete_from_srp')
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap_info',
|
||||||
|
return_value=tpd.PowerMaxData.volume_snap_vx)
|
||||||
|
def test_delete_volume_fail_if_active_snapshots(
|
||||||
|
self, mck_get_snaps, mck_cleanup, mck_delete):
|
||||||
|
array = self.data.array
|
||||||
|
test_volume = self.data.test_volume
|
||||||
|
device_id = self.data.device_id
|
||||||
|
extra_specs = self.common._initial_setup(test_volume)
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.common._delete_volume, test_volume)
|
||||||
|
mck_cleanup.assert_called_once_with(array, device_id, extra_specs)
|
||||||
|
mck_delete.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_delete_from_srp')
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
|
@mock.patch.object(
|
||||||
|
rest.PowerMaxRest, 'find_snap_vx_sessions',
|
||||||
|
return_value=('', tpd.PowerMaxData.snap_tgt_session_cm_enabled))
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
|
def test_delete_volume_fail_if_snapvx_target(
|
||||||
|
self, mck_get_snaps, mck_tgt_snap, mck_cleanup, mck_delete):
|
||||||
|
array = self.data.array
|
||||||
|
test_volume = self.data.test_volume
|
||||||
|
device_id = self.data.device_id
|
||||||
|
extra_specs = self.common._initial_setup(test_volume)
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.common._delete_volume, test_volume)
|
||||||
|
mck_cleanup.assert_called_once_with(array, device_id, extra_specs)
|
||||||
|
mck_delete.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(
|
@mock.patch.object(
|
||||||
common.PowerMaxCommon, 'get_snapshot_metadata',
|
common.PowerMaxCommon, 'get_snapshot_metadata',
|
||||||
return_value={'snap-meta-key-1': 'snap-meta-value-1',
|
return_value={'snap-meta-key-1': 'snap-meta-value-1',
|
||||||
'snap-meta-key-2': 'snap-meta-value-2'})
|
'snap-meta-key-2': 'snap-meta-value-2'})
|
||||||
def test_create_snapshot(self, mck_meta, mck_clone_chk):
|
def test_create_snapshot(self, mck_meta, mck_cleanup_snaps):
|
||||||
ref_model_update = (
|
ref_model_update = (
|
||||||
{'provider_location': six.text_type(self.data.snap_location),
|
{'provider_location': six.text_type(self.data.snap_location),
|
||||||
'metadata': {'snap-meta-key-1': 'snap-meta-value-1',
|
'metadata': {'snap-meta-key-1': 'snap-meta-value-1',
|
||||||
@ -1171,8 +1212,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
volume, connector, extra_specs)
|
volume, connector, extra_specs)
|
||||||
self.assertEqual('NONE', masking_view_dict[utils.WORKLOAD])
|
self.assertEqual('NONE', masking_view_dict[utils.WORKLOAD])
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_create_cloned_volume(self, mck_clone_chk):
|
def test_create_cloned_volume(self, mck_cleanup_snaps):
|
||||||
volume = self.data.test_clone_volume
|
volume = self.data.test_clone_volume
|
||||||
source_volume = self.data.test_volume
|
source_volume = self.data.test_volume
|
||||||
extra_specs = self.data.extra_specs
|
extra_specs = self.data.extra_specs
|
||||||
@ -1182,8 +1223,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
volume, source_volume, extra_specs))
|
volume, source_volume, extra_specs))
|
||||||
self.assertEqual(ref_response, (clone_dict, rep_update, rep_info_dict))
|
self.assertEqual(ref_response, (clone_dict, rep_update, rep_info_dict))
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_create_cloned_volume_is_snapshot(self, mck_clone_chk):
|
def test_create_cloned_volume_is_snapshot(self, mck_cleanup_snaps):
|
||||||
volume = self.data.test_snapshot
|
volume = self.data.test_snapshot
|
||||||
source_volume = self.data.test_volume
|
source_volume = self.data.test_volume
|
||||||
extra_specs = self.data.extra_specs
|
extra_specs = self.data.extra_specs
|
||||||
@ -1193,8 +1234,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
volume, source_volume, extra_specs, True, False))
|
volume, source_volume, extra_specs, True, False))
|
||||||
self.assertEqual(ref_response, (clone_dict, rep_update, rep_info_dict))
|
self.assertEqual(ref_response, (clone_dict, rep_update, rep_info_dict))
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_create_cloned_volume_from_snapshot(self, mck_clone_chk):
|
def test_create_cloned_volume_from_snapshot(self, mck_cleanup_snaps):
|
||||||
volume = self.data.test_clone_volume
|
volume = self.data.test_clone_volume
|
||||||
source_volume = self.data.test_snapshot
|
source_volume = self.data.test_snapshot
|
||||||
extra_specs = self.data.extra_specs
|
extra_specs = self.data.extra_specs
|
||||||
@ -1232,7 +1273,7 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
'_find_device_on_array',
|
'_find_device_on_array',
|
||||||
return_value=None)
|
return_value=None)
|
||||||
@mock.patch.object(common.PowerMaxCommon,
|
@mock.patch.object(common.PowerMaxCommon,
|
||||||
'_clone_check')
|
'_cleanup_device_snapvx')
|
||||||
def test_create_cloned_volume_source_not_found(
|
def test_create_cloned_volume_source_not_found(
|
||||||
self, mock_check, mock_device):
|
self, mock_check, mock_device):
|
||||||
volume = self.data.test_clone_volume
|
volume = self.data.test_clone_volume
|
||||||
@ -1300,16 +1341,18 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
self.common._create_snapshot,
|
self.common._create_snapshot,
|
||||||
array, snapshot, source_device_id, extra_specs)
|
array, snapshot, source_device_id, extra_specs)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(masking.PowerMaxMasking,
|
@mock.patch.object(masking.PowerMaxMasking,
|
||||||
'remove_vol_from_storage_group')
|
'remove_vol_from_storage_group')
|
||||||
def test_delete_volume_from_srp(self, mock_rm):
|
def test_delete_volume_from_srp(self, mock_rm, mock_get_snaps):
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
device_id = self.data.device_id
|
device_id = self.data.device_id
|
||||||
volume_name = self.data.test_volume.name
|
volume_name = self.data.test_volume.name
|
||||||
ref_extra_specs = deepcopy(self.data.extra_specs_intervals_set)
|
ref_extra_specs = deepcopy(self.data.extra_specs_intervals_set)
|
||||||
ref_extra_specs[utils.PORTGROUPNAME] = self.data.port_group_name_f
|
ref_extra_specs[utils.PORTGROUPNAME] = self.data.port_group_name_f
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
with mock.patch.object(self.common, '_sync_check'):
|
with mock.patch.object(self.common, '_cleanup_device_snapvx'):
|
||||||
with mock.patch.object(
|
with mock.patch.object(
|
||||||
self.common, '_delete_from_srp') as mock_delete:
|
self.common, '_delete_from_srp') as mock_delete:
|
||||||
self.common._delete_volume(volume)
|
self.common._delete_volume(volume)
|
||||||
@ -2062,8 +2105,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
return_value=(False, False, False))
|
return_value=(False, False, False))
|
||||||
@mock.patch.object(common.PowerMaxCommon,
|
@mock.patch.object(common.PowerMaxCommon,
|
||||||
'_remove_vol_and_cleanup_replication')
|
'_remove_vol_and_cleanup_replication')
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_unmanage_success(self, mck_clone, mock_rm, mck_sess):
|
def test_unmanage_success(self, mck_cleanup_snaps, mock_rm, mck_sess):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
with mock.patch.object(self.rest, 'rename_volume') as mock_rename:
|
with mock.patch.object(self.rest, 'rename_volume') as mock_rename:
|
||||||
self.common.unmanage(volume)
|
self.common.unmanage(volume)
|
||||||
@ -2090,8 +2133,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
return_value=(True, True, False))
|
return_value=(True, True, False))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_unmanage_temp_snapshot_links(self, mck_clone, mck_sess):
|
def test_unmanage_temp_snapshot_links(self, mck_cleanup_snaps, mck_sess):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
self.assertRaises(exception.VolumeIsBusy, self.common.unmanage,
|
self.assertRaises(exception.VolumeIsBusy, self.common.unmanage,
|
||||||
volume)
|
volume)
|
||||||
@ -2251,6 +2294,62 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
target_slo, target_workload, target_extra_specs)
|
target_slo, target_workload, target_extra_specs)
|
||||||
self.assertTrue(success)
|
self.assertTrue(success)
|
||||||
|
|
||||||
|
@mock.patch.object(utils.PowerMaxUtils, 'get_rep_config',
|
||||||
|
return_value=tpd.PowerMaxData.rep_config_metro)
|
||||||
|
@mock.patch.object(utils.PowerMaxUtils, 'is_replication_enabled',
|
||||||
|
side_effect=[False, True])
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_slo_list', return_value=[])
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=[{'snapshotName': 'name',
|
||||||
|
'linkedDevices': 'details'}])
|
||||||
|
def test_migrate_to_metro_exception_on_linked_snapshot_source(
|
||||||
|
self, mck_get, mck_slo, mck_validate, mck_rep, mck_config):
|
||||||
|
array_id = self.data.array
|
||||||
|
volume = self.data.test_volume
|
||||||
|
device_id = self.data.device_id
|
||||||
|
srp = self.data.srp
|
||||||
|
target_slo = self.data.slo_silver
|
||||||
|
target_workload = self.data.workload
|
||||||
|
volume_name = volume.name
|
||||||
|
target_extra_specs = self.data.rep_extra_specs_rep_config_metro
|
||||||
|
new_type = {'extra_specs': target_extra_specs}
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.assertRaises(
|
||||||
|
exception.VolumeBackendAPIException, self.common._migrate_volume,
|
||||||
|
array_id, volume, device_id, srp, target_slo, target_workload,
|
||||||
|
volume_name, new_type, extra_specs)
|
||||||
|
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
|
@mock.patch.object(utils.PowerMaxUtils, 'get_rep_config',
|
||||||
|
return_value=tpd.PowerMaxData.rep_config_metro)
|
||||||
|
@mock.patch.object(utils.PowerMaxUtils, 'is_replication_enabled',
|
||||||
|
side_effect=[False, True])
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_slo_list', return_value=[])
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=[{'snapshotName': 'name'}])
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
|
||||||
|
return_value=('', {'source_vol_id': 'source_vol_id',
|
||||||
|
'snap_name': 'snap_name'}))
|
||||||
|
def test_migrate_to_metro_exception_on_snapshot_target(
|
||||||
|
self, mck_find, mck_snap, mck_slo, mck_validate, mck_rep,
|
||||||
|
mck_config, mck_cleanup):
|
||||||
|
array_id = self.data.array
|
||||||
|
volume = self.data.test_volume
|
||||||
|
device_id = self.data.device_id
|
||||||
|
srp = self.data.srp
|
||||||
|
target_slo = self.data.slo_silver
|
||||||
|
target_workload = self.data.workload
|
||||||
|
volume_name = volume.name
|
||||||
|
target_extra_specs = self.data.rep_extra_specs_rep_config_metro
|
||||||
|
new_type = {'extra_specs': target_extra_specs}
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.assertRaises(
|
||||||
|
exception.VolumeBackendAPIException, self.common._migrate_volume,
|
||||||
|
array_id, volume, device_id, srp, target_slo, target_workload,
|
||||||
|
volume_name, new_type, extra_specs)
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon,
|
@mock.patch.object(common.PowerMaxCommon,
|
||||||
'_post_retype_srdf_protect_storage_group',
|
'_post_retype_srdf_protect_storage_group',
|
||||||
return_value=(True, True, True))
|
return_value=(True, True, True))
|
||||||
@ -2821,8 +2920,10 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
self.common._update_group_promotion,
|
self.common._update_group_promotion,
|
||||||
group, add_vols, remove_vols)
|
group, add_vols, remove_vols)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
||||||
def test_delete_group(self, mock_check):
|
def test_delete_group(self, mock_check, mck_snaps):
|
||||||
group = self.data.test_group_1
|
group = self.data.test_group_1
|
||||||
volumes = [self.data.test_volume]
|
volumes = [self.data.test_volume]
|
||||||
context = None
|
context = None
|
||||||
@ -2837,8 +2938,10 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
context, group, volumes)
|
context, group, volumes)
|
||||||
self.assertEqual(ref_model_update, model_update)
|
self.assertEqual(ref_model_update, model_update)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
||||||
def test_delete_group_success(self, mock_check):
|
def test_delete_group_success(self, mock_check, mck_get_snaps):
|
||||||
group = self.data.test_group_1
|
group = self.data.test_group_1
|
||||||
volumes = []
|
volumes = []
|
||||||
ref_model_update = {'status': fields.GroupStatus.DELETED}
|
ref_model_update = {'status': fields.GroupStatus.DELETED}
|
||||||
@ -2850,7 +2953,9 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
model_update, __ = self.common._delete_group(group, volumes)
|
model_update, __ = self.common._delete_group(group, volumes)
|
||||||
self.assertEqual(ref_model_update, model_update)
|
self.assertEqual(ref_model_update, model_update)
|
||||||
|
|
||||||
def test_delete_group_already_deleted(self):
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
|
def test_delete_group_already_deleted(self, mck_get_snaps):
|
||||||
group = self.data.test_group_failed
|
group = self.data.test_group_failed
|
||||||
ref_model_update = {'status': fields.GroupStatus.DELETED}
|
ref_model_update = {'status': fields.GroupStatus.DELETED}
|
||||||
volumes = []
|
volumes = []
|
||||||
@ -2859,10 +2964,13 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
model_update, __ = self.common._delete_group(group, volumes)
|
model_update, __ = self.common._delete_group(group, volumes)
|
||||||
self.assertEqual(ref_model_update, model_update)
|
self.assertEqual(ref_model_update, model_update)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
|
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
|
||||||
return_value=True)
|
return_value=True)
|
||||||
def test_delete_group_failed(self, mock_check, mock_type_check):
|
def test_delete_group_failed(
|
||||||
|
self, mock_check, mock_type_check, mck_get_snaps):
|
||||||
group = self.data.test_group_1
|
group = self.data.test_group_1
|
||||||
volumes = []
|
volumes = []
|
||||||
ref_model_update = {'status': fields.GroupStatus.ERROR_DELETING}
|
ref_model_update = {'status': fields.GroupStatus.ERROR_DELETING}
|
||||||
@ -2873,6 +2981,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
group, volumes)
|
group, volumes)
|
||||||
self.assertEqual(ref_model_update, model_update)
|
self.assertEqual(ref_model_update, model_update)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
@mock.patch.object(volume_utils, 'is_group_a_type', return_value=False)
|
||||||
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
|
@mock.patch.object(volume_utils, 'is_group_a_cg_snapshot_type',
|
||||||
return_value=True)
|
return_value=True)
|
||||||
@ -2885,15 +2995,38 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
return_value= tpd.PowerMaxData.device_id)
|
return_value= tpd.PowerMaxData.device_id)
|
||||||
@mock.patch.object(masking.PowerMaxMasking,
|
@mock.patch.object(masking.PowerMaxMasking,
|
||||||
'remove_volumes_from_storage_group')
|
'remove_volumes_from_storage_group')
|
||||||
def test_delete_group_clone_check(
|
def test_delete_group_cleanup_snapvx(
|
||||||
self, mock_rem, mock_find, mock_mems, mock_vols, mock_chk1,
|
self, mock_rem, mock_find, mock_mems, mock_vols, mock_chk1,
|
||||||
mock_chk2):
|
mock_chk2, mck_get_snaps):
|
||||||
group = self.data.test_group_1
|
group = self.data.test_group_1
|
||||||
volumes = [self.data.test_volume_group_member]
|
volumes = [self.data.test_volume_group_member]
|
||||||
with mock.patch.object(
|
with mock.patch.object(
|
||||||
self.common, '_clone_check') as mock_clone_chk:
|
self.common, '_cleanup_device_snapvx') as mock_cleanup_snapvx:
|
||||||
self.common._delete_group(group, volumes)
|
self.common._delete_group(group, volumes)
|
||||||
mock_clone_chk.assert_called_once()
|
mock_cleanup_snapvx.assert_called_once()
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=[{'snapshotName': 'name'}])
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
|
def test_delete_group_with_volumes_exception_on_remaining_snapshots(
|
||||||
|
self, mck_cleanup, mck_get):
|
||||||
|
group = self.data.test_group_1
|
||||||
|
volumes = [self.data.test_volume_group_member]
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.common._delete_group, group, volumes)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
|
||||||
|
return_value=('', {'source_vol_id': 'id',
|
||||||
|
'snap_name': 'name'}))
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=None)
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
|
def test_delete_group_with_volumes_exception_on_target_links(
|
||||||
|
self, mck_cleanup, mck_get, mck_find):
|
||||||
|
group = self.data.test_group_1
|
||||||
|
volumes = [self.data.test_volume_group_member]
|
||||||
|
self.assertRaises(exception.VolumeBackendAPIException,
|
||||||
|
self.common._delete_group, group, volumes)
|
||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'delete_storage_group')
|
@mock.patch.object(rest.PowerMaxRest, 'delete_storage_group')
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_failover_replication',
|
@mock.patch.object(common.PowerMaxCommon, '_failover_replication',
|
||||||
@ -3104,12 +3237,12 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
self.common.unmanage_snapshot(self.data.test_snapshot_manage)
|
self.common.unmanage_snapshot(self.data.test_snapshot_manage)
|
||||||
mock_mod.assert_called_once()
|
mock_mod.assert_called_once()
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'modify_volume_snap')
|
@mock.patch.object(rest.PowerMaxRest, 'modify_volume_snap')
|
||||||
def test_unmanage_snapshot_no_sync_check(self, mock_mod, mock_sync):
|
def test_unmanage_snapshot_no_snapvx_cleanup(self, mock_mod, mock_cleanup):
|
||||||
self.common.unmanage_snapshot(self.data.test_snapshot_manage)
|
self.common.unmanage_snapshot(self.data.test_snapshot_manage)
|
||||||
mock_mod.assert_called_once()
|
mock_mod.assert_called_once()
|
||||||
mock_sync.assert_not_called()
|
mock_cleanup.assert_not_called()
|
||||||
|
|
||||||
@mock.patch.object(utils.PowerMaxUtils, 'is_volume_failed_over',
|
@mock.patch.object(utils.PowerMaxUtils, 'is_volume_failed_over',
|
||||||
return_value=True)
|
return_value=True)
|
||||||
@ -3133,7 +3266,7 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
@mock.patch.object(provision.PowerMaxProvision, 'delete_volume_snap')
|
@mock.patch.object(provision.PowerMaxProvision, 'delete_volume_snap')
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'is_restore_complete',
|
@mock.patch.object(provision.PowerMaxProvision, 'is_restore_complete',
|
||||||
return_value=True)
|
return_value=True)
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'revert_volume_snapshot')
|
@mock.patch.object(provision.PowerMaxProvision, 'revert_volume_snapshot')
|
||||||
def test_revert_to_snapshot(self, mock_revert, mock_clone,
|
def test_revert_to_snapshot(self, mock_revert, mock_clone,
|
||||||
mock_complete, mock_delete, mock_parse):
|
mock_complete, mock_delete, mock_parse):
|
||||||
@ -3388,8 +3521,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
return_value=(None, False, None))
|
return_value=(None, False, None))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_extend_vol_validation_checks_success(self, mck_sync, mck_rep):
|
def test_extend_vol_validation_checks_success(self, mck_cleanup, mck_rep):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
device_id = self.data.device_id
|
device_id = self.data.device_id
|
||||||
@ -3400,8 +3533,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
return_value=(None, False, None))
|
return_value=(None, False, None))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_extend_vol_val_check_no_device(self, mck_sync, mck_rep):
|
def test_extend_vol_val_check_no_device(self, mck_cleanup, mck_rep):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
device_id = None
|
device_id = None
|
||||||
@ -3414,8 +3547,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
return_value=(None, True, None))
|
return_value=(None, True, None))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_extend_vol_val_check_snap_src(self, mck_sync, mck_rep):
|
def test_extend_vol_val_check_snap_src(self, mck_cleanup, mck_rep):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
device_id = self.data.device_id
|
device_id = self.data.device_id
|
||||||
@ -3429,8 +3562,8 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
return_value=(None, False, None))
|
return_value=(None, False, None))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_extend_vol_val_check_wrong_size(self, mck_sync, mck_rep):
|
def test_extend_vol_val_check_wrong_size(self, mck_cleanup, mck_rep):
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
device_id = self.data.device_id
|
device_id = self.data.device_id
|
||||||
@ -3585,103 +3718,6 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
port = self.common._get_unisphere_port()
|
port = self.common._get_unisphere_port()
|
||||||
self.assertEqual(ref_port, port)
|
self.assertEqual(ref_port, port)
|
||||||
|
|
||||||
def test_sync_check(self):
|
|
||||||
array = self.data.array
|
|
||||||
device_id = self.data.device_id
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
|
|
||||||
with mock.patch.object(self.common, '_do_sync_check') as mck_sync:
|
|
||||||
self.common._sync_check(array, device_id, extra_specs, False,
|
|
||||||
self.data.device_id2)
|
|
||||||
mck_sync.assert_called_with(array, self.data.device_id,
|
|
||||||
extra_specs, False)
|
|
||||||
mck_sync.reset_mock()
|
|
||||||
with mock.patch.object(self.common, '_get_target_source_device',
|
|
||||||
return_value=self.data.device_id3):
|
|
||||||
self.common._sync_check(array, device_id, extra_specs, True)
|
|
||||||
mck_sync.assert_called_with(array, self.data.device_id,
|
|
||||||
extra_specs, True)
|
|
||||||
mck_sync.reset_mock()
|
|
||||||
self.common._sync_check(array, device_id, extra_specs)
|
|
||||||
mck_sync.assert_called_with(array, device_id, extra_specs, False)
|
|
||||||
|
|
||||||
@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_do_sync_check(self, mck_rep, mck_find, mck_unlink):
|
|
||||||
|
|
||||||
array = self.data.array
|
|
||||||
device_id = self.data.device_id
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
self.common._do_sync_check(array, device_id, extra_specs)
|
|
||||||
self.assertEqual(3, mck_unlink.call_count)
|
|
||||||
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'delete_temp_volume_snap')
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'unlink_snapvx_tgt_volume')
|
|
||||||
def test_unlink_targets_and_delete_temp_snapvx_legacy(
|
|
||||||
self, mck_break, mck_del):
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
session = deepcopy(self.data.snap_tgt_session_cm_enabled)
|
|
||||||
legacy_snap_name = 'EMC_SMI'
|
|
||||||
session['snap_name'] = legacy_snap_name
|
|
||||||
source = session['source_vol_id']
|
|
||||||
snap_id = session['snapid']
|
|
||||||
target = session['target_vol_id']
|
|
||||||
self.common._unlink_targets_and_delete_temp_snapvx(
|
|
||||||
session, array, extra_specs)
|
|
||||||
mck_break.assert_called_with(array, target, source, legacy_snap_name,
|
|
||||||
extra_specs, snap_id, True)
|
|
||||||
mck_del.assert_called_once_with(
|
|
||||||
array, legacy_snap_name, source, snap_id)
|
|
||||||
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'delete_temp_volume_snap')
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'unlink_snapvx_tgt_volume')
|
|
||||||
def test_unlink_targets_and_delete_temp_snapvx_is_temp(
|
|
||||||
self, mck_break, mck_del):
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
session = deepcopy(self.data.snap_tgt_session_cm_enabled)
|
|
||||||
snap_name = session['snap_name']
|
|
||||||
source = session['source_vol_id']
|
|
||||||
snap_id = session['snapid']
|
|
||||||
target = session['target_vol_id']
|
|
||||||
self.common._unlink_targets_and_delete_temp_snapvx(
|
|
||||||
session, array, extra_specs)
|
|
||||||
mck_break.assert_called_with(array, target, source, snap_name,
|
|
||||||
extra_specs, snap_id, True)
|
|
||||||
mck_del.assert_called_once_with(array, snap_name, source, snap_id)
|
|
||||||
|
|
||||||
def test_unlink_targets_and_delete_temp_snapvx_no_snap_name_found(self):
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
session = dict()
|
|
||||||
self.assertRaises(exception.VolumeBackendAPIException,
|
|
||||||
self.common._unlink_targets_and_delete_temp_snapvx,
|
|
||||||
session, array, extra_specs)
|
|
||||||
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'delete_temp_volume_snap')
|
|
||||||
@mock.patch.object(provision.PowerMaxProvision, 'unlink_snapvx_tgt_volume')
|
|
||||||
def test_unlink_targets_and_delete_temp_snapvx_not_legacy_not_temp(
|
|
||||||
self, mck_break, mck_del):
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
session = deepcopy(self.data.snap_tgt_session_cm_enabled)
|
|
||||||
session['snap_name'] = 'not_leg_or_tmp'
|
|
||||||
snap_name = session['snap_name']
|
|
||||||
source = session['source_vol_id']
|
|
||||||
snap_id = session['snapid']
|
|
||||||
target = session['target_vol_id']
|
|
||||||
self.common._unlink_targets_and_delete_temp_snapvx(
|
|
||||||
session, array, extra_specs)
|
|
||||||
mck_break.assert_called_with(array, target, source, snap_name,
|
|
||||||
extra_specs, snap_id, True)
|
|
||||||
mck_del.assert_not_called()
|
|
||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
|
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
|
||||||
return_value=(None, tpd.PowerMaxData.snap_tgt_session))
|
return_value=(None, tpd.PowerMaxData.snap_tgt_session))
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
@ -3692,83 +3728,6 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
src_device = self.common._get_target_source_device(array, tgt_device)
|
src_device = self.common._get_target_source_device(array, tgt_device)
|
||||||
self.assertEqual(src_device, self.data.device_id)
|
self.assertEqual(src_device, self.data.device_id)
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_delete_valid_snapshot')
|
|
||||||
@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(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)
|
|
||||||
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):
|
|
||||||
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
|
|
||||||
session = {'snap_name': 'EMC_SMI_TEST', 'expired': False}
|
|
||||||
self.common._delete_valid_snapshot(array, session, extra_specs)
|
|
||||||
mck_unlink.assert_called_with(session, array, extra_specs)
|
|
||||||
mck_unlink.reset_mock()
|
|
||||||
|
|
||||||
session = {'snap_name': 'temp-000AA-snapshot_for_clone',
|
|
||||||
'expired': True}
|
|
||||||
self.common._delete_valid_snapshot(array, session, extra_specs)
|
|
||||||
mck_unlink.assert_called_with(session, array, extra_specs)
|
|
||||||
mck_unlink.reset_mock()
|
|
||||||
|
|
||||||
session = {'snap_name': 'temp-000AA-snapshot_for_clone',
|
|
||||||
'expired': False}
|
|
||||||
self.common._delete_valid_snapshot(array, session, extra_specs)
|
|
||||||
mck_unlink.assert_not_called()
|
|
||||||
|
|
||||||
def test_delete_valid_snapshot_exception(self):
|
|
||||||
|
|
||||||
array = self.data.array
|
|
||||||
extra_specs = self.data.extra_specs
|
|
||||||
session = {'snap_name': 'temp-000AA-snapshot_for_clone',
|
|
||||||
'expired': True}
|
|
||||||
|
|
||||||
with mock.patch.object(
|
|
||||||
self.common, '_unlink_targets_and_delete_temp_snapvx',
|
|
||||||
side_effect=exception.VolumeBackendAPIException(
|
|
||||||
"404 temp-000AA-snapshot_for_clone does not exist")
|
|
||||||
) as mck_unlink:
|
|
||||||
self.common._delete_valid_snapshot(array, session, extra_specs)
|
|
||||||
mck_unlink.assert_called_with(session, array, extra_specs)
|
|
||||||
|
|
||||||
with mock.patch.object(
|
|
||||||
self.common, '_unlink_targets_and_delete_temp_snapvx',
|
|
||||||
side_effect=exception.VolumeBackendAPIException(
|
|
||||||
"500 internal server error")):
|
|
||||||
self.assertRaises(
|
|
||||||
exception.VolumeBackendAPIException,
|
|
||||||
self.common._unlink_targets_and_delete_temp_snapvx,
|
|
||||||
array, session, extra_specs)
|
|
||||||
|
|
||||||
@mock.patch.object(rest.PowerMaxRest, '_get_private_volume',
|
@mock.patch.object(rest.PowerMaxRest, '_get_private_volume',
|
||||||
return_value=tpd.PowerMaxData.priv_vol_response_rep)
|
return_value=tpd.PowerMaxData.priv_vol_response_rep)
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'get_array_model_info',
|
@mock.patch.object(rest.PowerMaxRest, 'get_array_model_info',
|
||||||
@ -4339,3 +4298,110 @@ class PowerMaxCommonTest(test.TestCase):
|
|||||||
rep_driver_data)
|
rep_driver_data)
|
||||||
self.assertIsNone(group_name)
|
self.assertIsNone(group_name)
|
||||||
mock_group.assert_not_called()
|
mock_group.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(
|
||||||
|
common.PowerMaxCommon, '_unlink_and_delete_temporary_snapshots')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
|
||||||
|
return_value=(None, 'tgt_session'))
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
|
return_value=(True, False, False))
|
||||||
|
def test_cleanup_device_snapvx(self, mck_is_rep, mck_find, mck_unlink):
|
||||||
|
array = self.data.array
|
||||||
|
device_id = self.data.device_id
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.common._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
|
mck_unlink.assert_called_once_with('tgt_session', array, extra_specs)
|
||||||
|
|
||||||
|
@mock.patch.object(
|
||||||
|
common.PowerMaxCommon, '_unlink_and_delete_temporary_snapshots')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'is_vol_in_rep_session',
|
||||||
|
return_value=(False, False, False))
|
||||||
|
def test_cleanup_device_snapvx_no_sessions(self, mck_is_rep, mck_unlink):
|
||||||
|
array = self.data.array
|
||||||
|
device_id = self.data.device_id
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.common._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
|
mck_unlink.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_delete_temp_snapshot')
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_unlink_snapshot',
|
||||||
|
return_value=True)
|
||||||
|
def test_unlink_and_delete_temporary_snapshots_session_unlinked(
|
||||||
|
self, mck_unlink, mck_delete):
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
array = self.data.array
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.common._unlink_and_delete_temporary_snapshots(
|
||||||
|
session, array, extra_specs)
|
||||||
|
mck_unlink.assert_called_once_with(session, array, extra_specs)
|
||||||
|
mck_delete.assert_called_once_with(session, array)
|
||||||
|
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_delete_temp_snapshot')
|
||||||
|
@mock.patch.object(common.PowerMaxCommon, '_unlink_snapshot',
|
||||||
|
return_value=False)
|
||||||
|
def test_unlink_and_delete_temporary_snapshots_session_not_unlinked(
|
||||||
|
self, mck_unlink, mck_delete):
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
array = self.data.array
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
self.common._unlink_and_delete_temporary_snapshots(
|
||||||
|
session, array, extra_specs)
|
||||||
|
mck_unlink.assert_called_once_with(session, array, extra_specs)
|
||||||
|
mck_delete.assert_not_called()
|
||||||
|
|
||||||
|
@mock.patch.object(provision.PowerMaxProvision, 'unlink_snapvx_tgt_volume')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap',
|
||||||
|
side_effect=[tpd.PowerMaxData.priv_snap_response.get(
|
||||||
|
'snapshotSrcs')[0], None])
|
||||||
|
def test_unlink_temp_snapshot(self, mck_get, mck_unlink):
|
||||||
|
array = self.data.array
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
source = session.get('source_vol_id')
|
||||||
|
target = session.get('target_vol_id')
|
||||||
|
snap_name = session.get('snap_name')
|
||||||
|
snap_id = session.get('snapid')
|
||||||
|
loop = False
|
||||||
|
is_unlinked = self.common._unlink_snapshot(session, array, extra_specs)
|
||||||
|
mck_unlink.assert_called_once_with(
|
||||||
|
array, target, source, snap_name, extra_specs, snap_id, loop)
|
||||||
|
self.assertTrue(is_unlinked)
|
||||||
|
|
||||||
|
@mock.patch.object(provision.PowerMaxProvision, 'unlink_snapvx_tgt_volume')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap',
|
||||||
|
return_value=tpd.PowerMaxData.priv_snap_response.get(
|
||||||
|
'snapshotSrcs')[0])
|
||||||
|
def test_unlink_temp_snapshot_not_unlinked(self, mck_get, mck_unlink):
|
||||||
|
array = self.data.array
|
||||||
|
extra_specs = self.data.extra_specs
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
source = session.get('source_vol_id')
|
||||||
|
target = session.get('target_vol_id')
|
||||||
|
snap_name = session.get('snap_name')
|
||||||
|
snap_id = session.get('snapid')
|
||||||
|
loop = False
|
||||||
|
is_unlinked = self.common._unlink_snapshot(session, array, extra_specs)
|
||||||
|
mck_unlink.assert_called_once_with(
|
||||||
|
array, target, source, snap_name, extra_specs, snap_id, loop)
|
||||||
|
self.assertFalse(is_unlinked)
|
||||||
|
|
||||||
|
@mock.patch.object(provision.PowerMaxProvision, 'delete_temp_volume_snap')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap',
|
||||||
|
return_value=dict())
|
||||||
|
def test_delete_temp_snapshot(self, mck_get, mck_delete):
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
array = self.data.array
|
||||||
|
snap_name = session.get('snap_name')
|
||||||
|
source = session.get('source_vol_id')
|
||||||
|
snap_id = session.get('snapid')
|
||||||
|
self.common._delete_temp_snapshot(session, array)
|
||||||
|
mck_delete.assert_called_once_with(array, snap_name, source, snap_id)
|
||||||
|
|
||||||
|
@mock.patch.object(provision.PowerMaxProvision, 'delete_temp_volume_snap')
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap',
|
||||||
|
return_value={'linkedDevices': 'details'})
|
||||||
|
def test_delete_temp_snapshot_is_linked(self, mck_get, mck_delete):
|
||||||
|
session = self.data.snap_tgt_session
|
||||||
|
array = self.data.array
|
||||||
|
self.common._delete_temp_snapshot(session, array)
|
||||||
|
mck_delete.assert_not_called()
|
||||||
|
@ -235,9 +235,9 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
@mock.patch.object(masking.PowerMaxMasking,
|
@mock.patch.object(masking.PowerMaxMasking,
|
||||||
'remove_vol_from_storage_group')
|
'remove_vol_from_storage_group')
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_delete_from_srp')
|
@mock.patch.object(common.PowerMaxCommon, '_delete_from_srp')
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_cleanup_replication_source(
|
def test_cleanup_replication_source(
|
||||||
self, mck_sync, mock_del, mock_rm, mock_clean):
|
self, mck_cleanup, mock_del, mock_rm, mock_clean):
|
||||||
self.common._cleanup_replication_source(
|
self.common._cleanup_replication_source(
|
||||||
self.data.array, self.data.test_volume, 'vol1',
|
self.data.array, self.data.test_volume, 'vol1',
|
||||||
{'device_id': self.data.device_id}, self.extra_specs)
|
{'device_id': self.data.device_id}, self.extra_specs)
|
||||||
@ -460,10 +460,10 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_failover_replication',
|
@mock.patch.object(common.PowerMaxCommon, '_failover_replication',
|
||||||
return_value=({}, {}))
|
return_value=({}, {}))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(rest.PowerMaxRest, 'get_arrays_list',
|
@mock.patch.object(rest.PowerMaxRest, 'get_arrays_list',
|
||||||
return_value=['123'])
|
return_value=['123'])
|
||||||
def test_failover_host_async(self, mck_arrays, mck_sync, mock_fg):
|
def test_failover_host_async(self, mck_arrays, mck_cleanup, mock_fg):
|
||||||
volumes = [self.data.test_volume]
|
volumes = [self.data.test_volume]
|
||||||
extra_specs = deepcopy(self.extra_specs)
|
extra_specs = deepcopy(self.extra_specs)
|
||||||
extra_specs['rep_mode'] = utils.REP_ASYNC
|
extra_specs['rep_mode'] = utils.REP_ASYNC
|
||||||
@ -928,8 +928,8 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
return_value=(
|
return_value=(
|
||||||
tpd.PowerMaxData.provider_location_clone,
|
tpd.PowerMaxData.provider_location_clone,
|
||||||
tpd.PowerMaxData.replication_update, {}))
|
tpd.PowerMaxData.replication_update, {}))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_clone_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
def test_cloned_rep_volume(self, mck_clone, mck_meta, mck_clone_chk):
|
def test_cloned_rep_volume(self, mck_cleanup, mck_meta, mck_clone_chk):
|
||||||
metadata = deepcopy(self.data.volume_metadata)
|
metadata = deepcopy(self.data.volume_metadata)
|
||||||
metadata['BackendID'] = self.data.rep_backend_id_sync
|
metadata['BackendID'] = self.data.rep_backend_id_sync
|
||||||
ref_model_update = {
|
ref_model_update = {
|
||||||
@ -1069,7 +1069,7 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
self.assertTrue(success)
|
self.assertTrue(success)
|
||||||
|
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(
|
@mock.patch.object(
|
||||||
common.PowerMaxCommon, 'get_volume_metadata', return_value='')
|
common.PowerMaxCommon, 'get_volume_metadata', return_value='')
|
||||||
@mock.patch.object(
|
@mock.patch.object(
|
||||||
@ -1086,7 +1086,7 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
'remote_array': tpd.PowerMaxData.remote_array},
|
'remote_array': tpd.PowerMaxData.remote_array},
|
||||||
tpd.PowerMaxData.rep_extra_specs, False))
|
tpd.PowerMaxData.rep_extra_specs, False))
|
||||||
def test_migrate_volume_success_no_rep_to_rep(
|
def test_migrate_volume_success_no_rep_to_rep(
|
||||||
self, mck_configure, mck_retype, mck_protect, mck_get, mck_check,
|
self, mck_configure, mck_retype, mck_protect, mck_get, mck_cleanup,
|
||||||
mck_valid):
|
mck_valid):
|
||||||
self.common.rep_config = {'mode': utils.REP_SYNC,
|
self.common.rep_config = {'mode': utils.REP_SYNC,
|
||||||
'array': self.data.array}
|
'array': self.data.array}
|
||||||
@ -1127,6 +1127,8 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
self.data.rep_extra_specs, volume)
|
self.data.rep_extra_specs, volume)
|
||||||
self.assertTrue(success)
|
self.assertTrue(success)
|
||||||
|
|
||||||
|
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
|
||||||
|
return_value=list())
|
||||||
@mock.patch.object(utils.PowerMaxUtils, 'get_rep_config',
|
@mock.patch.object(utils.PowerMaxUtils, 'get_rep_config',
|
||||||
return_value=tpd.PowerMaxData.rep_config_async)
|
return_value=tpd.PowerMaxData.rep_config_async)
|
||||||
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
@mock.patch.object(common.PowerMaxCommon, 'get_volume_metadata',
|
||||||
@ -1143,14 +1145,15 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
tpd.PowerMaxData.rep_info_dict,
|
tpd.PowerMaxData.rep_info_dict,
|
||||||
tpd.PowerMaxData.rep_extra_specs_mgmt,
|
tpd.PowerMaxData.rep_extra_specs_mgmt,
|
||||||
True))
|
True))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_sync_check')
|
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
|
||||||
@mock.patch.object(common.PowerMaxCommon, 'break_rdf_device_pair_session',
|
@mock.patch.object(common.PowerMaxCommon, 'break_rdf_device_pair_session',
|
||||||
return_value=(tpd.PowerMaxData.rep_extra_specs_mgmt,
|
return_value=(tpd.PowerMaxData.rep_extra_specs_mgmt,
|
||||||
True))
|
True))
|
||||||
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
@mock.patch.object(common.PowerMaxCommon, '_validate_rdfg_status')
|
||||||
def test_migrate_volume_success_rep_to_rep(
|
def test_migrate_volume_success_rep_to_rep(
|
||||||
self, mck_valid, mck_break, mck_sync, mck_rep, mck_retype,
|
self, mck_valid, mck_break, mck_cleanup, mck_rep, mck_retype,
|
||||||
mck_resume, mck_slo, mck_upd_meta, mck_get_meta, mck_rep_conf):
|
mck_resume, mck_slo, mck_upd_meta, mck_get_meta, mck_rep_conf,
|
||||||
|
mck_get_snaps):
|
||||||
array = self.data.array
|
array = self.data.array
|
||||||
volume = self.data.test_volume
|
volume = self.data.test_volume
|
||||||
device_id = self.data.device_id
|
device_id = self.data.device_id
|
||||||
@ -1179,7 +1182,7 @@ class PowerMaxReplicationTest(test.TestCase):
|
|||||||
mck_valid.assert_any_call(array, extra_specs)
|
mck_valid.assert_any_call(array, extra_specs)
|
||||||
mck_break.assert_called_once_with(
|
mck_break.assert_called_once_with(
|
||||||
array, device_id, volume_name, extra_specs, volume)
|
array, device_id, volume_name, extra_specs, volume)
|
||||||
mck_sync.assert_called_once_with(array, device_id, extra_specs)
|
mck_cleanup.assert_called_once_with(array, device_id, extra_specs)
|
||||||
mck_rep.assert_called_once_with(
|
mck_rep.assert_called_once_with(
|
||||||
array, volume, device_id, target_extra_specs)
|
array, volume, device_id, target_extra_specs)
|
||||||
mck_retype.assert_called_once()
|
mck_retype.assert_called_once()
|
||||||
|
@ -98,12 +98,6 @@ powermax_opts = [
|
|||||||
cfg.MultiOpt(utils.U4P_FAILOVER_TARGETS,
|
cfg.MultiOpt(utils.U4P_FAILOVER_TARGETS,
|
||||||
item_type=types.Dict(),
|
item_type=types.Dict(),
|
||||||
help='Dictionary of Unisphere failover target info.'),
|
help='Dictionary of Unisphere failover target info.'),
|
||||||
cfg.IntOpt(utils.POWERMAX_SNAPVX_UNLINK_LIMIT,
|
|
||||||
default=3,
|
|
||||||
help='Use this value to specify '
|
|
||||||
'the maximum number of unlinks '
|
|
||||||
'for the temporary snapshots '
|
|
||||||
'before a clone operation.'),
|
|
||||||
cfg.StrOpt(utils.POWERMAX_ARRAY,
|
cfg.StrOpt(utils.POWERMAX_ARRAY,
|
||||||
help='Serial number of the array to connect to.'),
|
help='Serial number of the array to connect to.'),
|
||||||
cfg.StrOpt(utils.POWERMAX_SRP,
|
cfg.StrOpt(utils.POWERMAX_SRP,
|
||||||
@ -242,8 +236,6 @@ class PowerMaxCommon(object):
|
|||||||
"""Get relevent details from configuration file."""
|
"""Get relevent details from configuration file."""
|
||||||
self.interval = self.configuration.safe_get('interval')
|
self.interval = self.configuration.safe_get('interval')
|
||||||
self.retries = self.configuration.safe_get('retries')
|
self.retries = self.configuration.safe_get('retries')
|
||||||
self.snapvx_unlink_limit = self.configuration.safe_get(
|
|
||||||
utils.POWERMAX_SNAPVX_UNLINK_LIMIT)
|
|
||||||
self.powermax_array_tag_list = self.configuration.safe_get(
|
self.powermax_array_tag_list = self.configuration.safe_get(
|
||||||
utils.POWERMAX_ARRAY_TAG_LIST)
|
utils.POWERMAX_ARRAY_TAG_LIST)
|
||||||
self.powermax_short_host_name_template = self.configuration.safe_get(
|
self.powermax_short_host_name_template = self.configuration.safe_get(
|
||||||
@ -626,11 +618,7 @@ class PowerMaxCommon(object):
|
|||||||
source_device_id = self._find_device_on_array(
|
source_device_id = self._find_device_on_array(
|
||||||
source_volume, extra_specs)
|
source_volume, extra_specs)
|
||||||
|
|
||||||
if not self.next_gen and (
|
self._cleanup_device_snapvx(array, source_device_id, extra_specs)
|
||||||
extra_specs.get('rep_mode', None) == utils.REP_METRO):
|
|
||||||
self._sync_check(array, source_device_id, extra_specs)
|
|
||||||
else:
|
|
||||||
self._clone_check(array, source_device_id, extra_specs)
|
|
||||||
|
|
||||||
clone_dict, rep_update, rep_info_dict = self._create_cloned_volume(
|
clone_dict, rep_update, rep_info_dict = self._create_cloned_volume(
|
||||||
clone_volume, source_volume, extra_specs)
|
clone_volume, source_volume, extra_specs)
|
||||||
@ -1219,7 +1207,7 @@ class PowerMaxCommon(object):
|
|||||||
|
|
||||||
# 2 - Check if volume is part of an on-going clone operation or if vol
|
# 2 - Check if volume is part of an on-going clone operation or if vol
|
||||||
# has source snapshots but not next-gen array
|
# has source snapshots but not next-gen array
|
||||||
self._sync_check(array, device_id, ex_specs)
|
self._cleanup_device_snapvx(array, device_id, ex_specs)
|
||||||
__, snapvx_src, __ = self.rest.is_vol_in_rep_session(array, device_id)
|
__, snapvx_src, __ = self.rest.is_vol_in_rep_session(array, device_id)
|
||||||
if snapvx_src:
|
if snapvx_src:
|
||||||
if not self.next_gen:
|
if not self.next_gen:
|
||||||
@ -1946,7 +1934,7 @@ class PowerMaxCommon(object):
|
|||||||
|
|
||||||
# Perform any snapvx cleanup if required before creating the clone
|
# Perform any snapvx cleanup if required before creating the clone
|
||||||
if is_snapshot or from_snapvx:
|
if is_snapshot or from_snapvx:
|
||||||
self._clone_check(array, source_device_id, extra_specs)
|
self._cleanup_device_snapvx(array, source_device_id, extra_specs)
|
||||||
|
|
||||||
if not is_snapshot:
|
if not is_snapshot:
|
||||||
clone_dict, rep_update, rep_info_dict = self._create_replica(
|
clone_dict, rep_update, rep_info_dict = self._create_replica(
|
||||||
@ -2047,14 +2035,8 @@ class PowerMaxCommon(object):
|
|||||||
:param volume: volume object to be deleted
|
:param volume: volume object to be deleted
|
||||||
:returns: volume_name (string vol name)
|
:returns: volume_name (string vol name)
|
||||||
"""
|
"""
|
||||||
source_device_id = None
|
|
||||||
volume_name = volume.name
|
volume_name = volume.name
|
||||||
extra_specs = self._initial_setup(volume)
|
extra_specs = self._initial_setup(volume)
|
||||||
prov_loc = volume.provider_location
|
|
||||||
|
|
||||||
if isinstance(prov_loc, six.string_types):
|
|
||||||
name = ast.literal_eval(prov_loc)
|
|
||||||
source_device_id = name.get('source_device_id')
|
|
||||||
|
|
||||||
device_id = self._find_device_on_array(volume, extra_specs)
|
device_id = self._find_device_on_array(volume, extra_specs)
|
||||||
if device_id is None:
|
if device_id is None:
|
||||||
@ -2069,8 +2051,27 @@ class PowerMaxCommon(object):
|
|||||||
|
|
||||||
# Check if the volume being deleted is a
|
# Check if the volume being deleted is a
|
||||||
# source or target for copy session
|
# source or target for copy session
|
||||||
self._sync_check(array, device_id, extra_specs,
|
self._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
source_device_id=source_device_id)
|
# Confirm volume has no more snapshots associated and is not a target
|
||||||
|
snapshots = self.rest.get_volume_snapshot_list(array, device_id)
|
||||||
|
__, snapvx_target_details = self.rest.find_snap_vx_sessions(
|
||||||
|
array, device_id, tgt_only=True)
|
||||||
|
if snapshots:
|
||||||
|
snapshot_names = ', '.join(
|
||||||
|
snap.get('snapshotName') for snap in snapshots)
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Cannot delete device %s as it currently has the following '
|
||||||
|
'active snapshots: %s. Please try again once these snapshots '
|
||||||
|
'are no longer active.') % (device_id, snapshot_names))
|
||||||
|
if snapvx_target_details:
|
||||||
|
source_device = snapvx_target_details.get('source_vol_id')
|
||||||
|
snapshot_name = snapvx_target_details.get('snap_name')
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Cannot delete device %s as it is currently a linked target '
|
||||||
|
'of snapshot %s. The source device of this link is %s. '
|
||||||
|
'Please try again once this snapshots is no longer '
|
||||||
|
'active.') % (device_id, snapshot_name, source_device))
|
||||||
|
|
||||||
# Remove from any storage groups and cleanup replication
|
# Remove from any storage groups and cleanup replication
|
||||||
self._remove_vol_and_cleanup_replication(
|
self._remove_vol_and_cleanup_replication(
|
||||||
array, device_id, volume_name, extra_specs, volume)
|
array, device_id, volume_name, extra_specs, volume)
|
||||||
@ -2910,111 +2911,6 @@ class PowerMaxCommon(object):
|
|||||||
self._delete_from_srp(
|
self._delete_from_srp(
|
||||||
array, target_device_id, clone_name, extra_specs)
|
array, target_device_id, clone_name, extra_specs)
|
||||||
|
|
||||||
@retry(retry_exc_tuple, interval=1, retries=3)
|
|
||||||
def _sync_check(self, array, device_id, extra_specs,
|
|
||||||
tgt_only=False, source_device_id=None):
|
|
||||||
"""Check if volume is part of a SnapVx sync process.
|
|
||||||
|
|
||||||
:param array: the array serial number
|
|
||||||
:param device_id: volume instance
|
|
||||||
:param tgt_only: Flag - return only sessions where device is target
|
|
||||||
:param extra_specs: extra specifications
|
|
||||||
:param tgt_only: Flag to specify if it is a target
|
|
||||||
:param source_device_id: source_device_id if it has one
|
|
||||||
"""
|
|
||||||
if not source_device_id and tgt_only:
|
|
||||||
source_device_id = self._get_target_source_device(
|
|
||||||
array, device_id)
|
|
||||||
if source_device_id:
|
|
||||||
@coordination.synchronized("emc-source-{src_device_id}")
|
|
||||||
def do_unlink_and_delete_snap(src_device_id):
|
|
||||||
LOG.debug("Locking on source device %(device_id)s.",
|
|
||||||
{'device_id': src_device_id})
|
|
||||||
self._do_sync_check(
|
|
||||||
array, device_id, extra_specs, tgt_only)
|
|
||||||
|
|
||||||
do_unlink_and_delete_snap(source_device_id)
|
|
||||||
else:
|
|
||||||
self._do_sync_check(
|
|
||||||
array, device_id, extra_specs, tgt_only)
|
|
||||||
|
|
||||||
def _do_sync_check(
|
|
||||||
self, array, device_id, extra_specs, tgt_only=False):
|
|
||||||
"""Check if volume is part of a SnapVx sync process.
|
|
||||||
|
|
||||||
:param array: the array serial number
|
|
||||||
:param device_id: volume instance
|
|
||||||
:param tgt_only: Flag - return only sessions where device is target
|
|
||||||
:param extra_specs: extra specifications
|
|
||||||
:param tgt_only: Flag to specify if it is a target
|
|
||||||
"""
|
|
||||||
snapvx_tgt, snapvx_src, __ = self.rest.is_vol_in_rep_session(
|
|
||||||
array, device_id)
|
|
||||||
get_sessions = True if snapvx_tgt or snapvx_src else False
|
|
||||||
|
|
||||||
if get_sessions:
|
|
||||||
src_sessions, tgt_session = self.rest.find_snap_vx_sessions(
|
|
||||||
array, device_id, tgt_only)
|
|
||||||
if tgt_session:
|
|
||||||
self._unlink_targets_and_delete_temp_snapvx(
|
|
||||||
tgt_session, array, extra_specs)
|
|
||||||
if src_sessions and not tgt_only:
|
|
||||||
if not self.rest.is_snap_id:
|
|
||||||
src_sessions.sort(key=lambda k: k['snapid'], reverse=True)
|
|
||||||
for session in src_sessions:
|
|
||||||
self._unlink_targets_and_delete_temp_snapvx(
|
|
||||||
session, array, extra_specs)
|
|
||||||
|
|
||||||
def _unlink_targets_and_delete_temp_snapvx(
|
|
||||||
self, session, array, extra_specs):
|
|
||||||
"""Unlink target and delete the temporary snapvx if valid candidate.
|
|
||||||
|
|
||||||
:param session: the snapvx session
|
|
||||||
:param array: the array serial number
|
|
||||||
:param extra_specs: extra specifications
|
|
||||||
"""
|
|
||||||
snap_name = session.get('snap_name')
|
|
||||||
if not snap_name:
|
|
||||||
msg = _("Snap_name not present in volume's snapvx session. "
|
|
||||||
"Unable to perform unlink and cleanup.")
|
|
||||||
LOG.error(msg)
|
|
||||||
raise exception.VolumeBackendAPIException(msg)
|
|
||||||
source = session.get('source_vol_id')
|
|
||||||
snap_id = session.get('snapid')
|
|
||||||
is_legacy = 'EMC_SMI' in snap_name
|
|
||||||
is_temp = utils.CLONE_SNAPSHOT_NAME in snap_name
|
|
||||||
|
|
||||||
target, cm_enabled = None, False
|
|
||||||
if session.get('target_vol_id'):
|
|
||||||
target = session.get('target_vol_id')
|
|
||||||
cm_enabled = session.get('copy_mode')
|
|
||||||
|
|
||||||
if target and snap_name:
|
|
||||||
loop = True if cm_enabled else False
|
|
||||||
LOG.debug(
|
|
||||||
"Unlinking source from target. Source: %(vol)s, Target: "
|
|
||||||
"%(tgt)s, Snap id: %(snapid)s.", {'vol': source, 'tgt': target,
|
|
||||||
'snapid': snap_id})
|
|
||||||
self.provision.unlink_snapvx_tgt_volume(
|
|
||||||
array, target, source, snap_name, extra_specs, snap_id,
|
|
||||||
loop)
|
|
||||||
|
|
||||||
# Candidates for deletion:
|
|
||||||
# 1. If legacy snapshot with 'EMC_SMI' in snapshot name
|
|
||||||
# 2. If snapVX snapshot is temporary
|
|
||||||
# If a target is present in the snap session it will be unlinked by
|
|
||||||
# the provision.unlink_snapvx_tgt_volume call above. This accounts
|
|
||||||
# for both CM enabled and disabled, as such by this point there will
|
|
||||||
# either be no target or it would have been unlinked, therefore
|
|
||||||
# delete if it is legacy or temp.
|
|
||||||
if is_legacy or is_temp:
|
|
||||||
LOG.debug(
|
|
||||||
"Deleting temporary snapshot. Source: %(vol)s, snap name: "
|
|
||||||
"%(name)s, snap id: %(snapid)s.", {
|
|
||||||
'vol': source, 'name': snap_name, 'snapid': snap_id})
|
|
||||||
self.provision.delete_temp_volume_snap(
|
|
||||||
array, snap_name, source, snap_id)
|
|
||||||
|
|
||||||
def _get_target_source_device(self, array, device_id):
|
def _get_target_source_device(self, array, device_id):
|
||||||
"""Get the source device id of the target.
|
"""Get the source device id of the target.
|
||||||
|
|
||||||
@ -3036,14 +2932,14 @@ class PowerMaxCommon(object):
|
|||||||
|
|
||||||
return source_device_id
|
return source_device_id
|
||||||
|
|
||||||
def _clone_check(
|
@retry(retry_exc_tuple, interval=1, retries=3)
|
||||||
self, array, device_id, extra_specs, force_unlink=False):
|
def _cleanup_device_snapvx(
|
||||||
|
self, array, device_id, extra_specs):
|
||||||
"""Perform any snapvx cleanup before creating clones or snapshots
|
"""Perform any snapvx cleanup before creating clones or snapshots
|
||||||
|
|
||||||
:param array: the array serial
|
:param array: the array serial
|
||||||
:param device_id: the device ID of the volume
|
:param device_id: the device ID of the volume
|
||||||
:param extra_specs: extra specifications
|
: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(
|
snapvx_tgt, snapvx_src, __ = self.rest.is_vol_in_rep_session(
|
||||||
array, device_id)
|
array, device_id)
|
||||||
@ -3053,47 +2949,109 @@ class PowerMaxCommon(object):
|
|||||||
def do_unlink_and_delete_snap(src_device_id):
|
def do_unlink_and_delete_snap(src_device_id):
|
||||||
src_sessions, tgt_session = self.rest.find_snap_vx_sessions(
|
src_sessions, tgt_session = self.rest.find_snap_vx_sessions(
|
||||||
array, src_device_id)
|
array, src_device_id)
|
||||||
count = 0
|
if tgt_session:
|
||||||
if tgt_session and count < self.snapvx_unlink_limit:
|
self._unlink_and_delete_temporary_snapshots(
|
||||||
self._delete_valid_snapshot(
|
tgt_session, array, extra_specs)
|
||||||
array, tgt_session, extra_specs, force_unlink)
|
|
||||||
count += 1
|
|
||||||
if src_sessions:
|
if src_sessions:
|
||||||
for session in src_sessions:
|
if not self.rest.is_snap_id:
|
||||||
if count < self.snapvx_unlink_limit:
|
src_sessions.sort(
|
||||||
self._delete_valid_snapshot(
|
key=lambda k: k['snapid'], reverse=True)
|
||||||
array, session, extra_specs, force_unlink)
|
for src_session in src_sessions:
|
||||||
count += 1
|
self._unlink_and_delete_temporary_snapshots(
|
||||||
else:
|
src_session, array, extra_specs)
|
||||||
break
|
|
||||||
|
|
||||||
do_unlink_and_delete_snap(device_id)
|
do_unlink_and_delete_snap(device_id)
|
||||||
|
|
||||||
def _delete_valid_snapshot(
|
def _unlink_and_delete_temporary_snapshots(
|
||||||
self, array, session, extra_specs, force_unlink=False):
|
self, session, array, extra_specs):
|
||||||
"""Delete a snapshot if valid candidate for deletion.
|
"""Helper for unlinking and deleting temporary snapshot sessions
|
||||||
|
|
||||||
:param array: the array serial
|
:param session: snapvx session
|
||||||
:param session: the snapvx session
|
:param array: the array serial number
|
||||||
:param extra_specs: extra specifications
|
:param extra_specs: extra specifications
|
||||||
:param force_unlink: force unlink even if not expired
|
|
||||||
"""
|
"""
|
||||||
is_legacy = 'EMC_SMI' in session['snap_name']
|
try:
|
||||||
is_temp = utils.CLONE_SNAPSHOT_NAME in session['snap_name']
|
session_unlinked = self._unlink_snapshot(
|
||||||
is_expired = session['expired']
|
session, array, extra_specs)
|
||||||
if force_unlink:
|
if session_unlinked:
|
||||||
is_expired = True
|
self._delete_temp_snapshot(session, array)
|
||||||
is_valid = True if is_legacy or (is_temp and is_expired) else False
|
except exception.VolumeBackendAPIException as e:
|
||||||
if is_valid:
|
# Ignore and continue as snapshot has been unlinked
|
||||||
try:
|
# successfully with incorrect status code returned
|
||||||
self._unlink_targets_and_delete_temp_snapvx(
|
if ('404' and session['snap_name'] and
|
||||||
session, array, extra_specs)
|
'does not exist' in six.text_type(e)):
|
||||||
except exception.VolumeBackendAPIException as e:
|
pass
|
||||||
# Ignore and continue as snapshot has been unlinked
|
|
||||||
# successfully with incorrect status code returned
|
def _unlink_snapshot(self, session, array, extra_specs):
|
||||||
if ('404' and session['snap_name'] and
|
"""Helper for unlinking temporary snapshot during cleanup.
|
||||||
'does not exist' in six.text_type(e)):
|
|
||||||
pass
|
:param session: session that contains snapshot
|
||||||
|
:param array: the array serial number
|
||||||
|
:param extra_specs: extra specifications
|
||||||
|
:return:
|
||||||
|
"""
|
||||||
|
snap_name = session.get('snap_name')
|
||||||
|
source = session.get('source_vol_id')
|
||||||
|
snap_id = session.get('snapid')
|
||||||
|
|
||||||
|
snap_info = self.rest.get_volume_snap(
|
||||||
|
array, source, snap_name, snap_id)
|
||||||
|
is_linked = snap_info.get('linkedDevices')
|
||||||
|
|
||||||
|
target, cm_enabled = None, False
|
||||||
|
if session.get('target_vol_id'):
|
||||||
|
target = session.get('target_vol_id')
|
||||||
|
cm_enabled = session.get('copy_mode')
|
||||||
|
|
||||||
|
if target and snap_name and is_linked:
|
||||||
|
loop = True if cm_enabled else False
|
||||||
|
LOG.debug(
|
||||||
|
"Unlinking source from target. Source: %(vol)s, Target: "
|
||||||
|
"%(tgt)s, Snap id: %(snapid)s.",
|
||||||
|
{'vol': source, 'tgt': target, 'snapid': snap_id})
|
||||||
|
self.provision.unlink_snapvx_tgt_volume(
|
||||||
|
array, target, source, snap_name, extra_specs, snap_id,
|
||||||
|
loop)
|
||||||
|
|
||||||
|
is_unlinked = True
|
||||||
|
snap_info = self.rest.get_volume_snap(
|
||||||
|
array, source, snap_name, snap_id)
|
||||||
|
if snap_info and snap_info.get('linkedDevices'):
|
||||||
|
is_unlinked = False
|
||||||
|
return is_unlinked
|
||||||
|
|
||||||
|
def _delete_temp_snapshot(self, session, array):
|
||||||
|
"""Helper for deleting temporary snapshot during cleanup.
|
||||||
|
|
||||||
|
:param session: Session that contains snapshot
|
||||||
|
:param array: the array serial number
|
||||||
|
"""
|
||||||
|
snap_name = session.get('snap_name')
|
||||||
|
source = session.get('source_vol_id')
|
||||||
|
snap_id = session.get('snapid')
|
||||||
|
LOG.debug(
|
||||||
|
"Deleting temp snapshot if it exists. Snap name is: "
|
||||||
|
"%(snap_name)s, Source is: %(source)s, "
|
||||||
|
"Snap id: %(snap_id)s.",
|
||||||
|
{'snap_name': snap_name, 'source': source,
|
||||||
|
'snap_id': snap_id})
|
||||||
|
is_legacy = 'EMC_SMI' in snap_name if snap_name else False
|
||||||
|
is_temp = (
|
||||||
|
utils.CLONE_SNAPSHOT_NAME in snap_name if snap_name else False)
|
||||||
|
snap_info = self.rest.get_volume_snap(
|
||||||
|
array, source, snap_name, snap_id)
|
||||||
|
is_linked = snap_info.get('linkedDevices') if snap_info else False
|
||||||
|
|
||||||
|
# Candidates for deletion:
|
||||||
|
# 1. If legacy snapshot with 'EMC_SMI' in snapshot name
|
||||||
|
# 2. If snapVX snapshot is temporary
|
||||||
|
# 3. Snapshot is unlinked. Call _unlink_snapshot before delete.
|
||||||
|
if (is_legacy or is_temp) and not is_linked:
|
||||||
|
LOG.debug(
|
||||||
|
"Deleting temporary snapshot. Source: %(vol)s, snap name: "
|
||||||
|
"%(name)s, snap id: %(snapid)s.", {
|
||||||
|
'vol': source, 'name': snap_name, 'snapid': snap_id})
|
||||||
|
self.provision.delete_temp_volume_snap(
|
||||||
|
array, snap_name, source, snap_id)
|
||||||
|
|
||||||
def manage_existing(self, volume, external_ref):
|
def manage_existing(self, volume, external_ref):
|
||||||
"""Manages an existing PowerMax/VMAX Volume (import to Cinder).
|
"""Manages an existing PowerMax/VMAX Volume (import to Cinder).
|
||||||
@ -3367,7 +3325,7 @@ class PowerMaxCommon(object):
|
|||||||
{'id': volume_id})
|
{'id': volume_id})
|
||||||
else:
|
else:
|
||||||
# Check if volume is snap source
|
# Check if volume is snap source
|
||||||
self._clone_check(array, device_id, extra_specs)
|
self._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
snapvx_tgt, snapvx_src, __ = self.rest.is_vol_in_rep_session(
|
snapvx_tgt, snapvx_src, __ = self.rest.is_vol_in_rep_session(
|
||||||
array, device_id)
|
array, device_id)
|
||||||
if snapvx_src or snapvx_tgt:
|
if snapvx_src or snapvx_tgt:
|
||||||
@ -3933,13 +3891,7 @@ class PowerMaxCommon(object):
|
|||||||
:param extra_specs: the extra specifications
|
:param extra_specs: the extra specifications
|
||||||
:returns: bool
|
:returns: bool
|
||||||
"""
|
"""
|
||||||
model_update, success = None, False
|
orig_mgmt_sg_name = None
|
||||||
rep_info_dict, rep_extra_specs, rep_mode, rep_status = (
|
|
||||||
None, None, None, False)
|
|
||||||
resume_target_sg, resume_original_sg = False, False
|
|
||||||
resume_original_sg_dict = dict()
|
|
||||||
orig_mgmt_sg_name = ''
|
|
||||||
is_partitioned = False
|
|
||||||
|
|
||||||
target_extra_specs = dict(new_type['extra_specs'])
|
target_extra_specs = dict(new_type['extra_specs'])
|
||||||
target_extra_specs.update({
|
target_extra_specs.update({
|
||||||
@ -3952,30 +3904,10 @@ class PowerMaxCommon(object):
|
|||||||
target_extra_specs)
|
target_extra_specs)
|
||||||
target_extra_specs.update(
|
target_extra_specs.update(
|
||||||
{utils.DISABLECOMPRESSION: compression_disabled})
|
{utils.DISABLECOMPRESSION: compression_disabled})
|
||||||
|
(was_rep_enabled, is_rep_enabled, backend_ids_differ, rep_mode,
|
||||||
was_rep_enabled = self.utils.is_replication_enabled(extra_specs)
|
target_extra_specs) = (
|
||||||
if self.utils.is_replication_enabled(target_extra_specs):
|
self._get_replication_flags(
|
||||||
target_backend_id = target_extra_specs.get(
|
extra_specs, target_extra_specs))
|
||||||
utils.REPLICATION_DEVICE_BACKEND_ID,
|
|
||||||
utils.BACKEND_ID_LEGACY_REP)
|
|
||||||
target_rep_config = self.utils.get_rep_config(
|
|
||||||
target_backend_id, self.rep_configs)
|
|
||||||
rep_mode = target_rep_config['mode']
|
|
||||||
target_extra_specs[utils.REP_MODE] = rep_mode
|
|
||||||
target_extra_specs[utils.REP_CONFIG] = target_rep_config
|
|
||||||
is_rep_enabled = True
|
|
||||||
else:
|
|
||||||
is_rep_enabled = False
|
|
||||||
|
|
||||||
backend_ids_differ = False
|
|
||||||
if was_rep_enabled and is_rep_enabled:
|
|
||||||
curr_backend_id = extra_specs.get(
|
|
||||||
utils.REPLICATION_DEVICE_BACKEND_ID,
|
|
||||||
utils.BACKEND_ID_LEGACY_REP)
|
|
||||||
tgt_backend_id = target_extra_specs.get(
|
|
||||||
utils.REPLICATION_DEVICE_BACKEND_ID,
|
|
||||||
utils.BACKEND_ID_LEGACY_REP)
|
|
||||||
backend_ids_differ = curr_backend_id != tgt_backend_id
|
|
||||||
|
|
||||||
if was_rep_enabled and not self.promotion:
|
if was_rep_enabled and not self.promotion:
|
||||||
self._validate_rdfg_status(array, extra_specs)
|
self._validate_rdfg_status(array, extra_specs)
|
||||||
@ -3992,51 +3924,26 @@ class PowerMaxCommon(object):
|
|||||||
rdf_pair_broken, rdf_pair_created, vol_retyped, remote_retyped = (
|
rdf_pair_broken, rdf_pair_created, vol_retyped, remote_retyped = (
|
||||||
False, False, False, False)
|
False, False, False, False)
|
||||||
|
|
||||||
|
self._perform_snapshot_cleanup(
|
||||||
|
array, device_id, was_rep_enabled, is_rep_enabled,
|
||||||
|
backend_ids_differ, extra_specs, target_extra_specs)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
# Scenario 1: Rep -> Non-Rep
|
# Scenario 1: Rep -> Non-Rep
|
||||||
# Scenario 2: Cleanup for Rep -> Diff Rep type
|
# Scenario 2: Cleanup for Rep -> Diff Rep type
|
||||||
if (was_rep_enabled and not is_rep_enabled) or backend_ids_differ:
|
(model_update, resume_original_sg_dict, rdf_pair_broken,
|
||||||
if self.promotion:
|
resume_original_sg, is_partitioned) = (
|
||||||
resume_original_sg = False
|
self._prep_rep_to_non_rep(
|
||||||
rdf_group = extra_specs['rdf_group_no']
|
array, device_id, volume_name, volume, was_rep_enabled,
|
||||||
is_partitioned = self._rdf_vols_partitioned(
|
is_rep_enabled, backend_ids_differ, extra_specs))
|
||||||
array, [volume], rdf_group)
|
|
||||||
if not is_partitioned:
|
|
||||||
self.break_rdf_device_pair_session_promotion(
|
|
||||||
array, device_id, volume_name, extra_specs)
|
|
||||||
else:
|
|
||||||
rep_extra_specs, resume_original_sg = (
|
|
||||||
self.break_rdf_device_pair_session(
|
|
||||||
array, device_id, volume_name, extra_specs,
|
|
||||||
volume))
|
|
||||||
status = (REPLICATION_ERROR if self.promotion else
|
|
||||||
REPLICATION_DISABLED)
|
|
||||||
model_update = {
|
|
||||||
'replication_status': status,
|
|
||||||
'replication_driver_data': None}
|
|
||||||
rdf_pair_broken = True
|
|
||||||
if resume_original_sg:
|
|
||||||
resume_original_sg_dict = {
|
|
||||||
utils.ARRAY: array,
|
|
||||||
utils.SG_NAME: rep_extra_specs['mgmt_sg_name'],
|
|
||||||
utils.RDF_GROUP_NO: rep_extra_specs['rdf_group_no'],
|
|
||||||
utils.EXTRA_SPECS: rep_extra_specs}
|
|
||||||
|
|
||||||
# Scenario 1: Non-Rep -> Rep
|
# Scenario 1: Non-Rep -> Rep
|
||||||
# Scenario 2: Rep -> Diff Rep type
|
# Scenario 2: Rep -> Diff Rep type
|
||||||
if (not was_rep_enabled and is_rep_enabled) or backend_ids_differ:
|
(model_update, rdf_pair_created, rep_status, rep_driver_data,
|
||||||
self._sync_check(array, device_id, extra_specs)
|
rep_info_dict, rep_extra_specs, resume_target_sg) = (
|
||||||
(rep_status, rep_driver_data, rep_info_dict,
|
self._prep_non_rep_to_rep(
|
||||||
rep_extra_specs, resume_target_sg) = (
|
array, device_id, volume, was_rep_enabled,
|
||||||
self.configure_volume_replication(
|
is_rep_enabled, backend_ids_differ, target_extra_specs))
|
||||||
array, volume, device_id, target_extra_specs))
|
|
||||||
if rep_status != 'first_vol_in_rdf_group':
|
|
||||||
rdf_pair_created = True
|
|
||||||
model_update = {
|
|
||||||
'replication_status': rep_status,
|
|
||||||
'replication_driver_data': six.text_type(
|
|
||||||
{'device_id': rep_info_dict['target_device_id'],
|
|
||||||
'array': rep_info_dict['remote_array']})}
|
|
||||||
|
|
||||||
success, target_sg_name = self._retype_volume(
|
success, target_sg_name = self._retype_volume(
|
||||||
array, srp, device_id, volume, volume_name, extra_specs,
|
array, srp, device_id, volume, volume_name, extra_specs,
|
||||||
@ -4128,6 +4035,185 @@ class PowerMaxCommon(object):
|
|||||||
finally:
|
finally:
|
||||||
raise e
|
raise e
|
||||||
|
|
||||||
|
def _get_replication_flags(self, extra_specs, target_extra_specs):
|
||||||
|
"""Get replication flags from extra specifications.
|
||||||
|
|
||||||
|
:param extra_specs: extra specification -- dict
|
||||||
|
:param target_extra_specs: target extra specification -- dict
|
||||||
|
:returns: was_rep_enabled -- bool, is_rep_enabled -- bool,
|
||||||
|
backend_ids_differ -- bool, rep_mode -- str,
|
||||||
|
target_extra_specs -- dict
|
||||||
|
"""
|
||||||
|
rep_mode = None
|
||||||
|
was_rep_enabled = self.utils.is_replication_enabled(extra_specs)
|
||||||
|
if self.utils.is_replication_enabled(target_extra_specs):
|
||||||
|
target_backend_id = target_extra_specs.get(
|
||||||
|
utils.REPLICATION_DEVICE_BACKEND_ID,
|
||||||
|
utils.BACKEND_ID_LEGACY_REP)
|
||||||
|
target_rep_config = self.utils.get_rep_config(
|
||||||
|
target_backend_id, self.rep_configs)
|
||||||
|
rep_mode = target_rep_config['mode']
|
||||||
|
target_extra_specs[utils.REP_MODE] = rep_mode
|
||||||
|
target_extra_specs[utils.REP_CONFIG] = target_rep_config
|
||||||
|
is_rep_enabled = True
|
||||||
|
else:
|
||||||
|
is_rep_enabled = False
|
||||||
|
|
||||||
|
backend_ids_differ = False
|
||||||
|
if was_rep_enabled and is_rep_enabled:
|
||||||
|
curr_backend_id = extra_specs.get(
|
||||||
|
utils.REPLICATION_DEVICE_BACKEND_ID,
|
||||||
|
utils.BACKEND_ID_LEGACY_REP)
|
||||||
|
tgt_backend_id = target_extra_specs.get(
|
||||||
|
utils.REPLICATION_DEVICE_BACKEND_ID,
|
||||||
|
utils.BACKEND_ID_LEGACY_REP)
|
||||||
|
backend_ids_differ = curr_backend_id != tgt_backend_id
|
||||||
|
|
||||||
|
return (was_rep_enabled, is_rep_enabled, backend_ids_differ, rep_mode,
|
||||||
|
target_extra_specs)
|
||||||
|
|
||||||
|
def _prep_non_rep_to_rep(
|
||||||
|
self, array, device_id, volume, was_rep_enabled,
|
||||||
|
is_rep_enabled, backend_ids_differ, target_extra_specs):
|
||||||
|
"""Prepare for non rep to rep retype.
|
||||||
|
|
||||||
|
:param array: the array serial number -- str
|
||||||
|
:param device_id: the device id -- str
|
||||||
|
:param volume: the volume object -- objects.Volume
|
||||||
|
:param was_rep_enabled: flag -- bool
|
||||||
|
:param is_rep_enabled: flag -- bool
|
||||||
|
:param backend_ids_differ: flag -- bool
|
||||||
|
:param target_extra_specs: target extra specs -- dict
|
||||||
|
:returns: model_update -- dict, rdf_pair_created -- bool,
|
||||||
|
rep_status -- str, rep_driver_data -- dict,
|
||||||
|
rep_info_dict -- dict, rep_extra_specs -- dict,
|
||||||
|
resume_target_sg -- bool
|
||||||
|
"""
|
||||||
|
model_update, rep_status = None, None
|
||||||
|
resume_target_sg = False
|
||||||
|
rdf_pair_created = False
|
||||||
|
rep_driver_data, rep_info_dict = dict(), dict()
|
||||||
|
rep_extra_specs = dict()
|
||||||
|
if (not was_rep_enabled and is_rep_enabled) or backend_ids_differ:
|
||||||
|
(rep_status, rep_driver_data, rep_info_dict,
|
||||||
|
rep_extra_specs, resume_target_sg) = (
|
||||||
|
self.configure_volume_replication(
|
||||||
|
array, volume, device_id, target_extra_specs))
|
||||||
|
if rep_status != 'first_vol_in_rdf_group':
|
||||||
|
rdf_pair_created = True
|
||||||
|
model_update = {
|
||||||
|
'replication_status': rep_status,
|
||||||
|
'replication_driver_data': six.text_type(
|
||||||
|
{'device_id': rep_info_dict['target_device_id'],
|
||||||
|
'array': rep_info_dict['remote_array']})}
|
||||||
|
|
||||||
|
return (model_update, rdf_pair_created, rep_status, rep_driver_data,
|
||||||
|
rep_info_dict, rep_extra_specs, resume_target_sg)
|
||||||
|
|
||||||
|
def _prep_rep_to_non_rep(
|
||||||
|
self, array, device_id, volume_name, volume, was_rep_enabled,
|
||||||
|
is_rep_enabled, backend_ids_differ, extra_specs):
|
||||||
|
"""Preparation for replication to non-replicated.
|
||||||
|
|
||||||
|
:param array: the array serial number -- str
|
||||||
|
:param device_id: device_id: the device id -- str
|
||||||
|
:param volume_name: the volume name -- str
|
||||||
|
:param volume: the volume object -- objects.Volume
|
||||||
|
:param was_rep_enabled: flag -- bool
|
||||||
|
:param is_rep_enabled: flag -- bool
|
||||||
|
:param backend_ids_differ: flag -- bool
|
||||||
|
:param extra_specs: extra specs -- dict
|
||||||
|
:returns: model_update --dict , resume_original_sg_dict -- dict,
|
||||||
|
rdf_pair_broken -- bool, resume_original_sg -- bool,
|
||||||
|
is_partitioned -- bool
|
||||||
|
"""
|
||||||
|
model_update = dict()
|
||||||
|
resume_original_sg_dict = dict()
|
||||||
|
rdf_pair_broken = False
|
||||||
|
resume_original_sg = False
|
||||||
|
is_partitioned = False
|
||||||
|
if (was_rep_enabled and not is_rep_enabled) or backend_ids_differ:
|
||||||
|
if self.promotion:
|
||||||
|
resume_original_sg = False
|
||||||
|
rdf_group = extra_specs['rdf_group_no']
|
||||||
|
is_partitioned = self._rdf_vols_partitioned(
|
||||||
|
array, [volume], rdf_group)
|
||||||
|
if not is_partitioned:
|
||||||
|
self.break_rdf_device_pair_session_promotion(
|
||||||
|
array, device_id, volume_name, extra_specs)
|
||||||
|
else:
|
||||||
|
rep_extra_specs, resume_original_sg = (
|
||||||
|
self.break_rdf_device_pair_session(
|
||||||
|
array, device_id, volume_name, extra_specs,
|
||||||
|
volume))
|
||||||
|
status = (REPLICATION_ERROR if self.promotion else
|
||||||
|
REPLICATION_DISABLED)
|
||||||
|
model_update = {
|
||||||
|
'replication_status': status,
|
||||||
|
'replication_driver_data': None}
|
||||||
|
rdf_pair_broken = True
|
||||||
|
if resume_original_sg:
|
||||||
|
resume_original_sg_dict = {
|
||||||
|
utils.ARRAY: array,
|
||||||
|
utils.SG_NAME: rep_extra_specs['mgmt_sg_name'],
|
||||||
|
utils.RDF_GROUP_NO: rep_extra_specs['rdf_group_no'],
|
||||||
|
utils.EXTRA_SPECS: rep_extra_specs}
|
||||||
|
return (model_update, resume_original_sg_dict, rdf_pair_broken,
|
||||||
|
resume_original_sg, is_partitioned)
|
||||||
|
|
||||||
|
def _perform_snapshot_cleanup(
|
||||||
|
self, array, device_id, was_rep_enabled, is_rep_enabled,
|
||||||
|
backend_ids_differ, extra_specs, target_extra_specs):
|
||||||
|
"""Perform snapshot cleanup.
|
||||||
|
|
||||||
|
Perform snapshot cleanup before any other changes. If retyping
|
||||||
|
to either async or metro then there should be no linked snapshots
|
||||||
|
on the volume.
|
||||||
|
:param array: the array serial number -- str
|
||||||
|
:param device_id: device_id: the device id -- str
|
||||||
|
:param was_rep_enabled: flag -- bool
|
||||||
|
:param is_rep_enabled: flag -- bool
|
||||||
|
:param backend_ids_differ: flag -- bool
|
||||||
|
:param extra_specs: extra specs -- dict
|
||||||
|
:param target_extra_specs: target extra specs -- dict
|
||||||
|
"""
|
||||||
|
if (not was_rep_enabled and is_rep_enabled) or backend_ids_differ:
|
||||||
|
target_rep_mode = target_extra_specs.get(utils.REP_MODE)
|
||||||
|
target_is_async = target_rep_mode == utils.REP_ASYNC
|
||||||
|
target_is_metro = target_rep_mode == utils.REP_METRO
|
||||||
|
if target_is_async or target_is_metro:
|
||||||
|
self._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
|
snapshots = self.rest.get_volume_snapshot_list(
|
||||||
|
array, device_id)
|
||||||
|
__, snapvx_target_details = self.rest.find_snap_vx_sessions(
|
||||||
|
array, device_id, tgt_only=True)
|
||||||
|
|
||||||
|
linked_snapshots = list()
|
||||||
|
for snapshot in snapshots:
|
||||||
|
linked_devices = snapshot.get('linkedDevices')
|
||||||
|
if linked_devices:
|
||||||
|
snapshot_name = snapshot.get('snapshotName')
|
||||||
|
linked_snapshots.append(snapshot_name)
|
||||||
|
if linked_snapshots:
|
||||||
|
snapshot_names = ', '.join(linked_snapshots)
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Unable to complete retype as volume has active'
|
||||||
|
'snapvx links. Cannot retype to Asynchronous or '
|
||||||
|
'Metro modes while the volume has active links. '
|
||||||
|
'Please wait until these snapvx operations have '
|
||||||
|
'completed and try again. Snapshots: '
|
||||||
|
'%s') % snapshot_names)
|
||||||
|
if snapvx_target_details:
|
||||||
|
source_vol_id = snapvx_target_details.get('source_vol_id')
|
||||||
|
snap_name = snapvx_target_details.get('snap_name')
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Unable to complete retype as volume is a snapvx '
|
||||||
|
'target. Cannot retype to Asynchronous or Metro '
|
||||||
|
'modes in this state. Please wait until these snapvx '
|
||||||
|
'operations complete and try again. Volume %s is '
|
||||||
|
'currently a target of snapshot %s with source device '
|
||||||
|
'%s') % (device_id, snap_name, source_vol_id))
|
||||||
|
|
||||||
def _cleanup_on_migrate_failure(
|
def _cleanup_on_migrate_failure(
|
||||||
self, rdf_pair_broken, rdf_pair_created, vol_retyped,
|
self, rdf_pair_broken, rdf_pair_created, vol_retyped,
|
||||||
remote_retyped, extra_specs, target_extra_specs, volume,
|
remote_retyped, extra_specs, target_extra_specs, volume,
|
||||||
@ -5168,7 +5254,7 @@ class PowerMaxCommon(object):
|
|||||||
{'sourceName': volume_name})
|
{'sourceName': volume_name})
|
||||||
device_id = volume_dict['device_id']
|
device_id = volume_dict['device_id']
|
||||||
# Check if volume is snap target (e.g. if clone volume)
|
# Check if volume is snap target (e.g. if clone volume)
|
||||||
self._sync_check(array, device_id, extra_specs)
|
self._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
# Remove from any storage groups and cleanup replication
|
# Remove from any storage groups and cleanup replication
|
||||||
self._remove_vol_and_cleanup_replication(
|
self._remove_vol_and_cleanup_replication(
|
||||||
array, device_id, volume_name, extra_specs, volume)
|
array, device_id, volume_name, extra_specs, volume)
|
||||||
@ -5571,6 +5657,61 @@ class PowerMaxCommon(object):
|
|||||||
array, vol_grp_name)
|
array, vol_grp_name)
|
||||||
deleted_volume_device_ids = []
|
deleted_volume_device_ids = []
|
||||||
|
|
||||||
|
# If volumes are being deleted along with the group, ensure snapshot
|
||||||
|
# cleanup completes before doing any replication/storage group cleanup.
|
||||||
|
remaining_device_snapshots = list()
|
||||||
|
remaining_snapvx_targets = list()
|
||||||
|
for vol in volumes:
|
||||||
|
extra_specs = self._initial_setup(vol)
|
||||||
|
device_id = self._find_device_on_array(vol, extra_specs)
|
||||||
|
self._cleanup_device_snapvx(array, device_id, extra_specs)
|
||||||
|
snapshots = self.rest.get_volume_snapshot_list(array, device_id)
|
||||||
|
__, snapvx_target_details = self.rest.find_snap_vx_sessions(
|
||||||
|
array, device_id, tgt_only=True)
|
||||||
|
if snapshots:
|
||||||
|
snapshot_names = ', '.join(
|
||||||
|
snap.get('snapshotName') for snap in snapshots)
|
||||||
|
snap_details = {
|
||||||
|
'device_id': device_id, 'snapshot_names': snapshot_names}
|
||||||
|
remaining_device_snapshots.append(snap_details)
|
||||||
|
if snapvx_target_details:
|
||||||
|
source_vol_id = snapvx_target_details.get('source_vol_id')
|
||||||
|
snap_name = snapvx_target_details.get('snap_name')
|
||||||
|
target_details = {
|
||||||
|
'device_id': device_id, 'source_vol_id': source_vol_id,
|
||||||
|
'snapshot_name': snap_name}
|
||||||
|
remaining_snapvx_targets.append(target_details)
|
||||||
|
|
||||||
|
# Fail out if volumes to be deleted still have snapshots.
|
||||||
|
if remaining_device_snapshots:
|
||||||
|
for details in remaining_device_snapshots:
|
||||||
|
device_id = details.get('device_id')
|
||||||
|
snapshot_names = details.get('snapshot_names')
|
||||||
|
LOG.error('Cannot delete device %s, it has the '
|
||||||
|
'following active snapshots, %s.',
|
||||||
|
device_id, snapshot_names)
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Group volumes have active snapshots. Cannot perform group '
|
||||||
|
'delete. Wait for snapvx sessions to complete their '
|
||||||
|
'processes or remove volumes from group before attempting '
|
||||||
|
'to delete again. Please see previously logged error '
|
||||||
|
'message for device and snapshot details.'))
|
||||||
|
|
||||||
|
if remaining_snapvx_targets:
|
||||||
|
for details in remaining_snapvx_targets:
|
||||||
|
device_id = details.get('device_id')
|
||||||
|
snap_name = details.get('snapshot_name')
|
||||||
|
source_vol_id = details.get('source_vol_id')
|
||||||
|
LOG.error('Cannot delete device %s, it is current a target '
|
||||||
|
'of snapshot %s with source device id %s',
|
||||||
|
device_id, snap_name, source_vol_id)
|
||||||
|
raise exception.VolumeBackendAPIException(_(
|
||||||
|
'Some group volumes are targets of a snapvx session. Cannot '
|
||||||
|
'perform group delete. Wait for snapvx sessions to complete '
|
||||||
|
'their processes or remove volumes from group before '
|
||||||
|
'attempting to delete again. Please see previously logged '
|
||||||
|
'error message for device and snapshot details.'))
|
||||||
|
|
||||||
# Remove replication for group, if applicable
|
# Remove replication for group, if applicable
|
||||||
if group.is_replicated:
|
if group.is_replicated:
|
||||||
vt_extra_specs = self.utils.get_volumetype_extra_specs(
|
vt_extra_specs = self.utils.get_volumetype_extra_specs(
|
||||||
@ -5591,11 +5732,8 @@ class PowerMaxCommon(object):
|
|||||||
interval_retries_dict)
|
interval_retries_dict)
|
||||||
for vol in volumes:
|
for vol in volumes:
|
||||||
extra_specs = self._initial_setup(vol)
|
extra_specs = self._initial_setup(vol)
|
||||||
device_id = self._find_device_on_array(
|
device_id = self._find_device_on_array(vol, extra_specs)
|
||||||
vol, extra_specs)
|
|
||||||
if device_id in volume_device_ids:
|
if device_id in volume_device_ids:
|
||||||
self._clone_check(
|
|
||||||
array, device_id, extra_specs, force_unlink=True)
|
|
||||||
self.masking.remove_and_reset_members(
|
self.masking.remove_and_reset_members(
|
||||||
array, vol, device_id, vol.name,
|
array, vol, device_id, vol.name,
|
||||||
extra_specs, False)
|
extra_specs, False)
|
||||||
@ -6706,7 +6844,7 @@ class PowerMaxCommon(object):
|
|||||||
message=exception_message)
|
message=exception_message)
|
||||||
else:
|
else:
|
||||||
snap_id = snap_id_list[0]
|
snap_id = snap_id_list[0]
|
||||||
self._clone_check(array, sourcedevice_id, extra_specs)
|
self._cleanup_device_snapvx(array, sourcedevice_id, extra_specs)
|
||||||
try:
|
try:
|
||||||
LOG.info("Reverting device: %(deviceid)s "
|
LOG.info("Reverting device: %(deviceid)s "
|
||||||
"to snapshot: %(snapname)s.",
|
"to snapshot: %(snapname)s.",
|
||||||
|
@ -132,7 +132,6 @@ POWERMAX_ARRAY = 'powermax_array'
|
|||||||
POWERMAX_SRP = 'powermax_srp'
|
POWERMAX_SRP = 'powermax_srp'
|
||||||
POWERMAX_SERVICE_LEVEL = 'powermax_service_level'
|
POWERMAX_SERVICE_LEVEL = 'powermax_service_level'
|
||||||
POWERMAX_PORT_GROUPS = 'powermax_port_groups'
|
POWERMAX_PORT_GROUPS = 'powermax_port_groups'
|
||||||
POWERMAX_SNAPVX_UNLINK_LIMIT = 'powermax_snapvx_unlink_limit'
|
|
||||||
POWERMAX_ARRAY_TAG_LIST = 'powermax_array_tag_list'
|
POWERMAX_ARRAY_TAG_LIST = 'powermax_array_tag_list'
|
||||||
POWERMAX_SHORT_HOST_NAME_TEMPLATE = 'powermax_short_host_name_template'
|
POWERMAX_SHORT_HOST_NAME_TEMPLATE = 'powermax_short_host_name_template'
|
||||||
POWERMAX_PORT_GROUP_NAME_TEMPLATE = 'powermax_port_group_name_template'
|
POWERMAX_PORT_GROUP_NAME_TEMPLATE = 'powermax_port_group_name_template'
|
||||||
|
Loading…
x
Reference in New Issue
Block a user