PowerMax Driver - Improve error handling around deletes

1. Bring storage group check within delete volume so that
it can be used in every place it is called and not just the
delete method.
2. Remove the reset identifier_name name as this makes the
device available for use straight away even if is not deleted.
3. Allow for None value in unlink_snapshot.
4. Add extra debugging information for device id.
5. Retry on getting connection info for volume.
6. Validate a masking view on an attach operation.
7. Reset identifier name on a rollback.
8. Retry a cleanup if snap id not available at the time.
9. Pylint fixes.
10. Retry on a failed delete with a cleanup snapvx in exception.

Change-Id: I0c0f3d2246b840326579748027b9cd15bf174ff8
This commit is contained in:
Helen Walsh 2021-06-14 15:32:18 +01:00
parent 5917307a10
commit e1f6de6f83
10 changed files with 496 additions and 93 deletions

View File

@ -665,6 +665,13 @@ class PowerMaxData(object):
{'host_lun_address': '0003'}]}, {'host_lun_address': '0003'}]},
{}] {}]
maskingview_no_lun = {
'maskingViewId': masking_view_name_f,
'portGroupId': port_group_name_f,
'storageGroupId': storagegroup_name_f,
'hostId': initiatorgroup_name_f,
'maskingViewConnection': []}
portgroup = [{'portGroupId': port_group_name_f, portgroup = [{'portGroupId': port_group_name_f,
'symmetrixPortKey': [ 'symmetrixPortKey': [
{'directorId': 'FA-1D', {'directorId': 'FA-1D',

View File

@ -146,7 +146,7 @@ class FakeRequestsSession(object):
if '/private' in url: if '/private' in url:
return_object = self.data.private_vol_details return_object = self.data.private_vol_details
elif params: elif params:
if '1' in params.values(): if '1' in params.values() or 'volume_identifier' in params:
return_object = self.data.volume_list[0] return_object = self.data.volume_list[0]
elif '2' in params.values(): elif '2' in params.values():
return_object = self.data.volume_list[1] return_object = self.data.volume_list[1]

View File

@ -399,7 +399,7 @@ class PowerMaxCommonTest(test.TestCase):
extra_specs = self.common._initial_setup(test_volume) extra_specs = self.common._initial_setup(test_volume)
self.assertRaises(exception.VolumeBackendAPIException, self.assertRaises(exception.VolumeBackendAPIException,
self.common._delete_volume, test_volume) self.common._delete_volume, test_volume)
mck_cleanup.assert_called_once_with(array, device_id, extra_specs) mck_cleanup.assert_called_with(array, device_id, extra_specs)
mck_delete.assert_not_called() mck_delete.assert_not_called()
@mock.patch.object(common.PowerMaxCommon, '_delete_from_srp') @mock.patch.object(common.PowerMaxCommon, '_delete_from_srp')
@ -417,7 +417,7 @@ class PowerMaxCommonTest(test.TestCase):
extra_specs = self.common._initial_setup(test_volume) extra_specs = self.common._initial_setup(test_volume)
self.assertRaises(exception.VolumeBackendAPIException, self.assertRaises(exception.VolumeBackendAPIException,
self.common._delete_volume, test_volume) self.common._delete_volume, test_volume)
mck_cleanup.assert_called_once_with(array, device_id, extra_specs) mck_cleanup.assert_called_with(array, device_id, extra_specs)
mck_delete.assert_not_called() mck_delete.assert_not_called()
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx') @mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
@ -1941,12 +1941,15 @@ class PowerMaxCommonTest(test.TestCase):
clone_name, snap_name, self.data.extra_specs, clone_name, snap_name, self.data.extra_specs,
target_volume=clone_volume) target_volume=clone_volume)
@mock.patch.object(rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=[])
@mock.patch.object(rest.PowerMaxRest, 'get_snap_id', @mock.patch.object(rest.PowerMaxRest, 'get_snap_id',
return_value=tpd.PowerMaxData.snap_id) return_value=tpd.PowerMaxData.snap_id)
@mock.patch.object( @mock.patch.object(
masking.PowerMaxMasking, masking.PowerMaxMasking,
'remove_and_reset_members') 'remove_and_reset_members')
def test_cleanup_target_sync_present(self, mock_remove, mock_snaps): def test_cleanup_target_sync_present(
self, mock_remove, mock_snaps, mock_sgs):
array = self.data.array array = self.data.array
clone_volume = self.data.test_clone_volume clone_volume = self.data.test_clone_volume
source_device_id = self.data.device_id source_device_id = self.data.device_id
@ -2974,6 +2977,7 @@ 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(common.PowerMaxCommon, '_delete_from_srp')
@mock.patch.object(masking.PowerMaxMasking, 'remove_and_reset_members') @mock.patch.object(masking.PowerMaxMasking, 'remove_and_reset_members')
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx') @mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
@mock.patch.object(common.PowerMaxCommon, '_get_members_of_volume_group', @mock.patch.object(common.PowerMaxCommon, '_get_members_of_volume_group',
@ -2983,7 +2987,7 @@ class PowerMaxCommonTest(test.TestCase):
@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_snapshot_and_volume_cleanup( def test_delete_group_snapshot_and_volume_cleanup(
self, mock_check, mck_get_snaps, mock_members, mock_cleanup, self, mock_check, mck_get_snaps, mock_members, mock_cleanup,
mock_remove): mock_remove, mock_del):
group = self.data.test_group_1 group = self.data.test_group_1
volumes = [fake_volume.fake_volume_obj( volumes = [fake_volume.fake_volume_obj(
context='cxt', provider_location=None)] context='cxt', provider_location=None)]
@ -3837,6 +3841,13 @@ class PowerMaxCommonTest(test.TestCase):
act_metadata = self.common.get_volume_metadata(array, device_id) act_metadata = self.common.get_volume_metadata(array, device_id)
self.assertEqual(ref_metadata, act_metadata) self.assertEqual(ref_metadata, act_metadata)
def test_get_volume_metadata_device_none(self):
ref_metadata = {}
array = self.data.array
device_id = None
act_metadata = self.common.get_volume_metadata(array, device_id)
self.assertEqual(ref_metadata, act_metadata)
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snap_info', @mock.patch.object(rest.PowerMaxRest, 'get_volume_snap_info',
return_value=tpd.PowerMaxData.priv_snap_response) return_value=tpd.PowerMaxData.priv_snap_response)
def test_get_snapshot_metadata(self, mck_snap): def test_get_snapshot_metadata(self, mck_snap):
@ -3925,6 +3936,23 @@ class PowerMaxCommonTest(test.TestCase):
model_update, existing_metadata, object_metadata) model_update, existing_metadata, object_metadata)
self.assertEqual(ref_model_update, model_update) self.assertEqual(ref_model_update, model_update)
def test_update_metadata_no_object_metadata(self):
model_update = {'provider_location': six.text_type(
self.data.provider_location)}
ref_model_update = (
{'provider_location': six.text_type(self.data.provider_location),
'metadata': {'user-meta-key-1': 'user-meta-value-1',
'user-meta-key-2': 'user-meta-value-2'}})
existing_metadata = {'user-meta-key-1': 'user-meta-value-1',
'user-meta-key-2': 'user-meta-value-2'}
object_metadata = {}
model_update = self.common.update_metadata(
model_update, existing_metadata, object_metadata)
self.assertEqual(ref_model_update, model_update)
def test_update_metadata_model_list_exception(self): def test_update_metadata_model_list_exception(self):
model_update = [{'provider_location': six.text_type( model_update = [{'provider_location': six.text_type(
self.data.provider_location)}] self.data.provider_location)}]
@ -4562,3 +4590,66 @@ class PowerMaxCommonTest(test.TestCase):
self.assertTrue(rnr.rdf_pair_broken) self.assertTrue(rnr.rdf_pair_broken)
self.assertTrue(rnr.resume_original_sg) self.assertTrue(rnr.resume_original_sg)
self.assertFalse(rnr.is_partitioned) self.assertFalse(rnr.is_partitioned)
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
@mock.patch.object(rest.PowerMaxRest, 'get_volume_snapshot_list',
return_value=[])
@mock.patch.object(rest.PowerMaxRest, 'find_snap_vx_sessions',
side_effect=[(None, tpd.PowerMaxData.snap_tgt_session),
(None, None)])
def test_cleanup_device_retry(self, mock_snapvx, mock_ss_list, mock_clean):
self.common._cleanup_device_retry(
self.data.array, self.data.device_id, self.data.extra_specs)
self.assertEqual(2, mock_clean.call_count)
@mock.patch.object(rest.PowerMaxRest, 'find_volume_device_id',
return_value=[tpd.PowerMaxData.device_id,
tpd.PowerMaxData.device_id2])
def test_get_device_id_from_identifier_list(self, mock_dev_id):
ret_dev = self.common._get_device_id_from_identifier(
self.data.array, 'vol', self.data.device_id)
self.assertEqual(self.data.device_id, ret_dev)
@mock.patch.object(rest.PowerMaxRest, 'find_volume_device_id',
return_value=tpd.PowerMaxData.device_id2)
def test_get_device_id_from_identifier_wrong(self, mock_dev_id):
ret_dev = self.common._get_device_id_from_identifier(
self.data.array, 'vol', self.data.device_id)
self.assertEqual(self.data.device_id2, ret_dev)
@mock.patch.object(rest.PowerMaxRest, 'find_volume_device_id',
return_value=tpd.PowerMaxData.device_id)
def test_get_device_id_from_identifier_same(self, mock_dev_id):
ret_dev = self.common._get_device_id_from_identifier(
self.data.array, 'vol', self.data.device_id)
self.assertIsNone(ret_dev)
@mock.patch.object(rest.PowerMaxRest, 'rename_volume')
@mock.patch.object(rest.PowerMaxRest, 'find_volume_identifier',
return_value='vol')
@mock.patch.object(rest.PowerMaxRest, 'find_volume_device_id',
return_value=tpd.PowerMaxData.device_id)
def test_reset_identifier_on_rollback_rename(
self, mock_dev, mock_ident, mock_rename):
self.common._reset_identifier_on_rollback(self.data.array, 'vol')
mock_rename.assert_called_once()
@mock.patch.object(rest.PowerMaxRest, 'rename_volume')
@mock.patch.object(rest.PowerMaxRest, 'find_volume_identifier',
return_value='diff_vol_name')
@mock.patch.object(rest.PowerMaxRest, 'find_volume_device_id',
return_value=tpd.PowerMaxData.device_id)
def test_reset_identifier_on_rollback_no_rename(
self, mock_dev, mock_ident, mock_rename):
self.common._reset_identifier_on_rollback(self.data.array, 'vol')
mock_rename.assert_not_called()
@mock.patch.object(common.PowerMaxCommon, '_cleanup_device_snapvx')
@mock.patch.object(
provision.PowerMaxProvision, 'delete_volume_from_srp',
side_effect=[exception.VolumeBackendAPIException, None])
def test_test_delete_from_srp(self, mock_del, mock_clean):
self.common._delete_from_srp(
self.data.array, 'vol_name', self.data.device_id,
self.data.extra_specs)
mock_clean.assert_called_once()

View File

@ -96,6 +96,8 @@ class PowerMaxMaskingTest(test.TestCase):
self.maskingviewdict, self.extra_specs) self.maskingviewdict, self.extra_specs)
mock_get_or_create_mv.assert_called_once() mock_get_or_create_mv.assert_called_once()
@mock.patch.object(masking.PowerMaxMasking, '_validate_attach',
return_value=True)
@mock.patch.object(masking.PowerMaxMasking, @mock.patch.object(masking.PowerMaxMasking,
'_check_adding_volume_to_storage_group') '_check_adding_volume_to_storage_group')
@mock.patch.object(masking.PowerMaxMasking, '_move_vol_from_default_sg', @mock.patch.object(masking.PowerMaxMasking, '_move_vol_from_default_sg',
@ -108,7 +110,7 @@ class PowerMaxMaskingTest(test.TestCase):
Exception('Exception')]) Exception('Exception')])
def test_get_or_create_masking_view_and_map_lun( def test_get_or_create_masking_view_and_map_lun(
self, mock_masking_view_element, mock_masking, mock_move, self, mock_masking_view_element, mock_masking, mock_move,
mock_add_volume): mock_add_volume, mock_validate):
rollback_dict = ( rollback_dict = (
self.driver.masking.get_or_create_masking_view_and_map_lun( self.driver.masking.get_or_create_masking_view_and_map_lun(
self.data.array, self.data.test_volume, self.data.array, self.data.test_volume,
@ -864,6 +866,9 @@ class PowerMaxMaskingTest(test.TestCase):
self.data.array, self.data.masking_view_name_i) self.data.array, self.data.masking_view_name_i)
mock_delete_mv.assert_called_once() mock_delete_mv.assert_called_once()
@mock.patch.object(
rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=list())
@mock.patch.object(masking.PowerMaxMasking, @mock.patch.object(masking.PowerMaxMasking,
'return_volume_to_volume_group') 'return_volume_to_volume_group')
@mock.patch.object(rest.PowerMaxRest, 'move_volume_between_storage_groups') @mock.patch.object(rest.PowerMaxRest, 'move_volume_between_storage_groups')
@ -871,7 +876,7 @@ class PowerMaxMaskingTest(test.TestCase):
'get_or_create_default_storage_group') 'get_or_create_default_storage_group')
@mock.patch.object(masking.PowerMaxMasking, 'add_volume_to_storage_group') @mock.patch.object(masking.PowerMaxMasking, 'add_volume_to_storage_group')
def test_add_volume_to_default_storage_group( def test_add_volume_to_default_storage_group(
self, mock_add_sg, mock_get_sg, mock_move, mock_return): self, mock_add_sg, mock_get_sg, mock_move, mock_return, mock_sgs):
self.mask.add_volume_to_default_storage_group( self.mask.add_volume_to_default_storage_group(
self.data.array, self.device_id, self.volume_name, self.data.array, self.device_id, self.volume_name,
self.extra_specs) self.extra_specs)
@ -1372,3 +1377,36 @@ class PowerMaxMaskingTest(test.TestCase):
exception_message): exception_message):
self.mask._check_director_and_port_status( self.mask._check_director_and_port_status(
self.data.array, self.data.port_group_name_f) self.data.array, self.data.port_group_name_f)
@mock.patch.object(
rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=([tpd.PowerMaxData.storagegroup_name_f]))
@mock.patch.object(
rest.PowerMaxRest, 'get_masking_views_from_storage_group',
return_value=[tpd.PowerMaxData.masking_view_name_f])
def test_validate_attach(self, mock_sgs, mock_mvs):
self.assertTrue(self.mask._validate_attach(
self.data.array, self.data.device_id,
self.data.storagegroup_name_f, self.data.masking_view_name_f))
@mock.patch.object(
rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=([tpd.PowerMaxData.storagegroup_name_f]))
@mock.patch.object(
rest.PowerMaxRest, 'get_masking_views_from_storage_group',
return_value=[])
def test_validate_attach_no_mvs(self, mock_sgs, mock_mvs):
self.assertFalse(self.mask._validate_attach(
self.data.array, self.data.device_id,
self.data.storagegroup_name_f, self.data.masking_view_name_f))
@mock.patch.object(
rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=([tpd.PowerMaxData.defaultstoragegroup_name]))
@mock.patch.object(
rest.PowerMaxRest, 'get_masking_views_from_storage_group',
return_value=[])
def test_validate_attach_incorrect_sg(self, mock_sgs, mock_mvs):
self.assertFalse(self.mask._validate_attach(
self.data.array, self.data.device_id,
self.data.storagegroup_name_f, self.data.masking_view_name_f))

View File

@ -216,12 +216,21 @@ class PowerMaxReplicationTest(test.TestCase):
self.iscsi_common.initialize_connection, self.iscsi_common.initialize_connection,
self.data.test_volume, self.data.connector) self.data.test_volume, self.data.connector)
@mock.patch.object(masking.PowerMaxMasking, '_validate_attach',
return_value=True)
@mock.patch.object(
rest.PowerMaxRest, 'get_storage_groups_from_volume',
side_effect=[
[], [], [], [], [tpd.PowerMaxData.storagegroup_name_f], [],
[tpd.PowerMaxData.storagegroup_name_f],
[tpd.PowerMaxData.storagegroup_name_f]], )
@mock.patch.object( @mock.patch.object(
masking.PowerMaxMasking, '_check_director_and_port_status') masking.PowerMaxMasking, '_check_director_and_port_status')
@mock.patch.object( @mock.patch.object(
masking.PowerMaxMasking, 'pre_multiattach', masking.PowerMaxMasking, 'pre_multiattach',
return_value=tpd.PowerMaxData.masking_view_dict_multiattach) return_value=tpd.PowerMaxData.masking_view_dict_multiattach)
def test_attach_metro_volume(self, mock_pre, mock_check): def test_attach_metro_volume(
self, mock_pre, mock_check, mock_sgs, mock_validate):
rep_extra_specs = deepcopy(tpd.PowerMaxData.rep_extra_specs) rep_extra_specs = deepcopy(tpd.PowerMaxData.rep_extra_specs)
rep_extra_specs[utils.PORTGROUPNAME] = self.data.port_group_name_f rep_extra_specs[utils.PORTGROUPNAME] = self.data.port_group_name_f
hostlunid, remote_port_group = self.common._attach_metro_volume( hostlunid, remote_port_group = self.common._attach_metro_volume(
@ -802,10 +811,12 @@ class PowerMaxReplicationTest(test.TestCase):
self.data.test_rep_group, self.data.rep_extra_specs) self.data.test_rep_group, self.data.rep_extra_specs)
mock_rm.assert_called_once() mock_rm.assert_called_once()
@mock.patch.object(rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=[])
@mock.patch.object(masking.PowerMaxMasking, 'remove_and_reset_members') @mock.patch.object(masking.PowerMaxMasking, 'remove_and_reset_members')
@mock.patch.object(masking.PowerMaxMasking, @mock.patch.object(masking.PowerMaxMasking,
'remove_volumes_from_storage_group') 'remove_volumes_from_storage_group')
def test_cleanup_group_replication(self, mock_rm, mock_rm_reset): def test_cleanup_group_replication(self, mock_rm, mock_rm_reset, mock_sgs):
self.common._cleanup_group_replication( self.common._cleanup_group_replication(
self.data.array, self.data.test_vol_grp_name, self.data.array, self.data.test_vol_grp_name,
[self.data.device_id], self.extra_specs, self.data.rep_config_sync) [self.data.device_id], self.extra_specs, self.data.rep_config_sync)

View File

@ -741,7 +741,9 @@ class PowerMaxRestTest(test.TestCase):
mck_modify.assert_called_once_with(array, device_id, mck_modify.assert_called_once_with(array, device_id,
extend_vol_payload) extend_vol_payload)
def test_legacy_delete_volume(self): @mock.patch.object(rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=[])
def test_legacy_delete_volume(self, mock_sgs):
device_id = self.data.device_id device_id = self.data.device_id
vb_except = exception.VolumeBackendAPIException vb_except = exception.VolumeBackendAPIException
with mock.patch.object(self.rest, 'delete_resource') as mock_delete, ( with mock.patch.object(self.rest, 'delete_resource') as mock_delete, (
@ -755,21 +757,28 @@ class PowerMaxRestTest(test.TestCase):
mock_delete.assert_called_once_with( mock_delete.assert_called_once_with(
self.data.array, 'sloprovisioning', 'volume', device_id) self.data.array, 'sloprovisioning', 'volume', device_id)
def test_delete_volume(self): @mock.patch.object(rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=[])
def test_delete_volume(self, mock_sgs):
device_id = self.data.device_id device_id = self.data.device_id
self.rest.ucode_major_level = utils.UCODE_5978 self.rest.ucode_major_level = utils.UCODE_5978
self.rest.ucode_minor_level = utils.UCODE_5978_HICKORY self.rest.ucode_minor_level = utils.UCODE_5978_HICKORY
with mock.patch.object( with mock.patch.object(
self.rest, 'delete_resource') as mock_delete, ( self.rest, 'delete_resource') as mock_delete:
mock.patch.object(
self.rest, '_modify_volume')) as mock_modify:
self.rest.delete_volume(self.data.array, device_id) self.rest.delete_volume(self.data.array, device_id)
mod_call_count = mock_modify.call_count
self.assertEqual(1, mod_call_count)
mock_delete.assert_called_once_with( mock_delete.assert_called_once_with(
self.data.array, 'sloprovisioning', 'volume', device_id) self.data.array, 'sloprovisioning', 'volume', device_id)
@mock.patch.object(rest.PowerMaxRest, 'get_storage_groups_from_volume',
return_value=['OS-SG'])
def test_delete_volume_in_sg(self, mock_sgs):
device_id = self.data.device_id
self.rest.ucode_major_level = utils.UCODE_5978
self.rest.ucode_minor_level = utils.UCODE_5978_HICKORY
self.assertRaises(
exception.VolumeBackendAPIException,
self.rest.delete_volume, self.data.array, device_id)
def test_rename_volume(self): def test_rename_volume(self):
device_id = self.data.device_id device_id = self.data.device_id
payload = {'editVolumeActionParam': { payload = {'editVolumeActionParam': {
@ -847,6 +856,14 @@ class PowerMaxRestTest(test.TestCase):
self.data.array, self.data.masking_view_name_f, device_id) self.data.array, self.data.masking_view_name_f, device_id)
self.assertEqual(ref_lun_id, host_lun_id) self.assertEqual(ref_lun_id, host_lun_id)
def test_find_mv_connections_for_vol_missing_host_lun_address(self):
with mock.patch.object(self.rest, 'get_resource',
return_value=self.data.maskingview_no_lun):
host_lun_id = self.rest.find_mv_connections_for_vol(
self.data.array, self.data.masking_view_name_f,
self.data.device_id)
self.assertIsNone(host_lun_id)
def test_find_mv_connections_for_vol_failed(self): def test_find_mv_connections_for_vol_failed(self):
# no masking view info retrieved # no masking view info retrieved
device_id = self.data.volume_details[0]['volumeId'] device_id = self.data.volume_details[0]['volumeId']

View File

@ -2057,7 +2057,7 @@ class PowerMaxCommon(object):
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:
LOG.error("Volume %(name)s not found on the array. " LOG.warning("Volume %(name)s not found on the array. "
"No volume to delete.", "No volume to delete.",
{'name': volume_name}) {'name': volume_name})
return volume_name return volume_name
@ -2066,6 +2066,23 @@ class PowerMaxCommon(object):
if self.utils.is_replication_enabled(extra_specs): if self.utils.is_replication_enabled(extra_specs):
self._validate_rdfg_status(array, extra_specs) self._validate_rdfg_status(array, extra_specs)
self._cleanup_device_retry(array, device_id, extra_specs)
# Remove from any storage groups and cleanup replication
self._remove_vol_and_cleanup_replication(
array, device_id, volume_name, extra_specs, volume)
self._delete_from_srp(
array, device_id, volume_name, extra_specs)
return volume_name
@retry(retry_exc_tuple, interval=1, retries=3)
def _cleanup_device_retry(self, array, device_id, extra_specs):
"""Cleanup snapvx on the device
:param array: the serial number of the array -- str
:param device_id: the device id -- str
:param extra_specs: extra specs -- dict
"""
# 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._cleanup_device_snapvx(array, device_id, extra_specs) self._cleanup_device_snapvx(array, device_id, extra_specs)
@ -2083,25 +2100,13 @@ class PowerMaxCommon(object):
if snapvx_target_details: if snapvx_target_details:
source_device = snapvx_target_details.get('source_vol_id') source_device = snapvx_target_details.get('source_vol_id')
snapshot_name = snapvx_target_details.get('snap_name') snapshot_name = snapvx_target_details.get('snap_name')
if snapshot_name:
raise exception.VolumeBackendAPIException(_( raise exception.VolumeBackendAPIException(_(
'Cannot delete device %s as it is currently a linked target ' 'Cannot delete device %s as it is currently a linked '
'of snapshot %s. The source device of this link is %s. ' 'target of snapshot %s. The source device of this link '
'Please try again once this snapshots is no longer ' 'is %s. Please try again once this snapshot is no longer '
'active.') % (device_id, snapshot_name, source_device)) 'active.') % (device_id, snapshot_name, source_device))
# Remove from any storage groups and cleanup replication
self._remove_vol_and_cleanup_replication(
array, device_id, volume_name, extra_specs, volume)
# Check if volume is in any storage group
sg_list = self.rest.get_storage_groups_from_volume(array, device_id)
if sg_list:
LOG.error("Device %(device_id)s is in storage group(s) "
"%(sg_list)s prior to delete. Delete will fail.",
{'device_id': device_id, 'sg_list': sg_list})
self._delete_from_srp(
array, device_id, volume_name, extra_specs)
return volume_name
def _create_volume(self, volume, volume_name, volume_size, extra_specs): def _create_volume(self, volume, volume_name, volume_size, extra_specs):
"""Create a volume. """Create a volume.
@ -2162,15 +2167,40 @@ class PowerMaxCommon(object):
array, volume, volume_name, volume_size, extra_specs, array, volume, volume_name, volume_size, extra_specs,
storagegroup_name, rep_mode)) storagegroup_name, rep_mode))
# Compare volume ID against identifier on array. Update if needed. device_id = self._get_device_id_from_identifier(
# This can occur in cases where multiple edits are occurring at once. array, volume_name, volume_dict['device_id'])
found_device_id = self.rest.find_volume_device_id(array, volume_name) if device_id:
returning_device_id = volume_dict['device_id'] volume_dict['device_id'] = device_id
if found_device_id != returning_device_id:
volume_dict['device_id'] = found_device_id
return volume_dict, rep_update, rep_info_dict return volume_dict, rep_update, rep_info_dict
def _get_device_id_from_identifier(
self, array, volume_name, orig_device_id):
"""Get the device(s) using the identifier name
:param array: the serial number of the array -- str
:param volume_name: the user supplied volume name -- str
:param orig_device_id: the original device id -- str
:returns: device id -- str
"""
# Compare volume ID against identifier on array. Update if needed.
# This can occur in cases where multiple edits are occurring at once.
dev_id_from_identifier = self.rest.find_volume_device_id(
array, volume_name)
if isinstance(dev_id_from_identifier, list):
if orig_device_id in dev_id_from_identifier:
return orig_device_id
else:
if dev_id_from_identifier != orig_device_id:
LOG.warning(
"The device id %(dev_ident)s associated with %(vol_name)s "
"is not the same as device %(dev_orig)s.",
{'dev_ident': dev_id_from_identifier,
'vol_name': volume_name,
'dev_orig': orig_device_id})
return dev_id_from_identifier
return None
@coordination.synchronized("emc-nonrdf-vol-{storagegroup_name}-{array}") @coordination.synchronized("emc-nonrdf-vol-{storagegroup_name}-{array}")
def _create_non_replicated_volume( def _create_non_replicated_volume(
self, array, volume, volume_name, storagegroup_name, volume_size, self, array, volume, volume_name, storagegroup_name, volume_size,
@ -2195,6 +2225,7 @@ class PowerMaxCommon(object):
return volume_dict return volume_dict
except Exception as e: except Exception as e:
try: try:
self._reset_identifier_on_rollback(array, volume_name)
# Attempt cleanup of storage group post exception. # Attempt cleanup of storage group post exception.
updated_devices = set(self.rest.get_volumes_in_storage_group( updated_devices = set(self.rest.get_volumes_in_storage_group(
array, storagegroup_name)) array, storagegroup_name))
@ -2518,6 +2549,7 @@ class PowerMaxCommon(object):
LOG.info("The tag list %(tag_list)s has been verified.", LOG.info("The tag list %(tag_list)s has been verified.",
{'tag_list': array_tag_list}) {'tag_list': array_tag_list})
@retry(retry_exc_tuple, interval=3, retries=3)
def _delete_from_srp(self, array, device_id, volume_name, def _delete_from_srp(self, array, device_id, volume_name,
extra_specs): extra_specs):
"""Delete from srp. """Delete from srp.
@ -2534,11 +2566,17 @@ class PowerMaxCommon(object):
self.provision.delete_volume_from_srp( self.provision.delete_volume_from_srp(
array, device_id, volume_name) array, device_id, volume_name)
except Exception as e: except Exception as e:
error_message = (_("Failed to delete volume %(volume_name)s. " error_message = (_(
"Exception received: %(e)s") % "Failed to delete volume %(volume_name)s with device id "
"%(dev)s. Exception received: %(e)s.") %
{'volume_name': volume_name, {'volume_name': volume_name,
'dev': device_id,
'e': six.text_type(e)}) 'e': six.text_type(e)})
LOG.error(error_message) LOG.error(error_message)
LOG.warning("Attempting device cleanup after a failed delete of: "
"%(name)s. device_id: %(device_id)s.",
{'name': volume_name, 'device_id': device_id})
self._cleanup_device_snapvx(array, device_id, extra_specs)
raise exception.VolumeBackendAPIException(message=error_message) raise exception.VolumeBackendAPIException(message=error_message)
def _remove_vol_and_cleanup_replication( def _remove_vol_and_cleanup_replication(
@ -2953,7 +2991,7 @@ class PowerMaxCommon(object):
return source_device_id return source_device_id
@retry(retry_exc_tuple, interval=1, retries=3) @retry(retry_exc_tuple, interval=10, retries=3)
def _cleanup_device_snapvx( def _cleanup_device_snapvx(
self, array, device_id, extra_specs): self, array, device_id, extra_specs):
"""Perform any snapvx cleanup before creating clones or snapshots """Perform any snapvx cleanup before creating clones or snapshots
@ -2990,17 +3028,15 @@ class PowerMaxCommon(object):
:param array: the array serial number :param array: the array serial number
:param extra_specs: extra specifications :param extra_specs: extra specifications
""" """
try:
session_unlinked = self._unlink_snapshot( session_unlinked = self._unlink_snapshot(
session, array, extra_specs) session, array, extra_specs)
if session_unlinked: if session_unlinked:
self._delete_temp_snapshot(session, array) self._delete_temp_snapshot(session, array)
except exception.VolumeBackendAPIException as e: else:
# Ignore and continue as snapshot has been unlinked LOG.warning(
# successfully with incorrect status code returned "Snap name %(snap)s is still linked. The delete of "
if ('404' and session['snap_name'] and "the temporary snapshot has not occurred.",
'does not exist' in six.text_type(e)): {'snap': session.get('snap_name')})
pass
def _unlink_snapshot(self, session, array, extra_specs): def _unlink_snapshot(self, session, array, extra_specs):
"""Helper for unlinking temporary snapshot during cleanup. """Helper for unlinking temporary snapshot during cleanup.
@ -3013,10 +3049,19 @@ class PowerMaxCommon(object):
snap_name = session.get('snap_name') snap_name = session.get('snap_name')
source = session.get('source_vol_id') source = session.get('source_vol_id')
snap_id = session.get('snapid') snap_id = session.get('snapid')
if snap_id is None:
exception_message = (
_("Unable to get snapid from session %(session)s for source "
"device %(dev)s. Retrying...")
% {'dev': source, 'session': session})
# Warning only as there will be a retry
LOG.warning(exception_message)
raise exception.VolumeBackendAPIException(
message=exception_message)
snap_info = self.rest.get_volume_snap( snap_info = self.rest.get_volume_snap(
array, source, snap_name, snap_id) array, source, snap_name, snap_id)
is_linked = snap_info.get('linkedDevices') is_linked = snap_info.get('linkedDevices') if snap_info else None
target, cm_enabled = None, False target, cm_enabled = None, False
if session.get('target_vol_id'): if session.get('target_vol_id'):
@ -3252,7 +3297,7 @@ class PowerMaxCommon(object):
volume_identifier = None volume_identifier = None
# Check if volume is already cinder managed # Check if volume is already cinder managed
if volume_details.get('volume_identifier'): if volume_details.get('volume_identifier'):
volume_identifier = volume_details['volume_identifier'] volume_identifier = volume_details.get('volume_identifier')
if volume_identifier.startswith(utils.VOLUME_ELEMENT_NAME_PREFIX): if volume_identifier.startswith(utils.VOLUME_ELEMENT_NAME_PREFIX):
raise exception.ManageExistingAlreadyManaged( raise exception.ManageExistingAlreadyManaged(
volume_ref=volume_id) volume_ref=volume_id)
@ -3569,6 +3614,7 @@ class PowerMaxCommon(object):
"they can be managed into Cinder.") "they can be managed into Cinder.")
return manageable_vols return manageable_vols
volumes = volumes or list()
for device in volumes: for device in volumes:
# Determine if volume is valid for management # Determine if volume is valid for management
if self.utils.is_volume_manageable(device): if self.utils.is_volume_manageable(device):
@ -3677,6 +3723,7 @@ class PowerMaxCommon(object):
"Cinder.") "Cinder.")
return manageable_snaps return manageable_snaps
volumes = volumes or list()
for device in volumes: for device in volumes:
# Determine if volume is valid for management # Determine if volume is valid for management
manageable_snaps = self._is_snapshot_valid_for_management( manageable_snaps = self._is_snapshot_valid_for_management(
@ -4760,7 +4807,7 @@ class PowerMaxCommon(object):
if self.rest.is_next_gen_array(target_array_serial): if self.rest.is_next_gen_array(target_array_serial):
target_workload = 'NONE' target_workload = 'NONE'
except IndexError: except IndexError:
LOG.error("Error parsing array, pool, SLO and workload.") LOG.debug("Error parsing array, pool, SLO and workload.")
return false_ret return false_ret
if self.promotion: if self.promotion:
@ -6092,7 +6139,7 @@ class PowerMaxCommon(object):
# Get the volume group dict for getting the group name # Get the volume group dict for getting the group name
volume_group = (self._find_volume_group(array, source_group)) volume_group = (self._find_volume_group(array, source_group))
if volume_group and volume_group.get('name'): if volume_group and volume_group.get('name'):
vol_grp_name = volume_group['name'] vol_grp_name = volume_group.get('name')
if vol_grp_name is None: if vol_grp_name is None:
LOG.warning("Cannot find generic volume group %(grp_ss_id)s. " LOG.warning("Cannot find generic volume group %(grp_ss_id)s. "
"on array %(array)s", "on array %(array)s",
@ -7107,6 +7154,8 @@ class PowerMaxCommon(object):
:param device_id: the device ID :param device_id: the device ID
:returns: dict -- volume metadata :returns: dict -- volume metadata
""" """
if device_id is None:
return dict()
vol_info = self.rest._get_private_volume(array, device_id) vol_info = self.rest._get_private_volume(array, device_id)
vol_header = vol_info['volumeHeader'] vol_header = vol_info['volumeHeader']
array_model, __ = self.rest.get_array_model_info(array) array_model, __ = self.rest.get_array_model_info(array)
@ -7576,3 +7625,26 @@ class PowerMaxCommon(object):
management_sg_name, rdf_group_number, missing_volumes_str) management_sg_name, rdf_group_number, missing_volumes_str)
is_valid = False is_valid = False
return is_valid return is_valid
def _reset_identifier_on_rollback(self, array, volume_name):
"""Reset the user supplied name on a rollback
:param array: the serial number -- str
:param volume_name: the volume name assigned -- str
"""
# Find volume based on identifier name
dev_id_from_identifier = self.rest.find_volume_device_id(
array, volume_name)
if dev_id_from_identifier and isinstance(
dev_id_from_identifier, str):
vol_identifier_name = self.rest.find_volume_identifier(
array, dev_id_from_identifier)
if vol_identifier_name and (
vol_identifier_name == volume_name):
LOG.warning(
"Attempting to reset name of %(vol_name)s on device "
"%(dev_ident)s on a create volume rollback operation.",
{'vol_name': volume_name,
'dev_ident': dev_id_from_identifier})
self.rest.rename_volume(
array, dev_id_from_identifier, None)

View File

@ -90,12 +90,14 @@ class PowerMaxMasking(object):
error_message = self._get_or_create_masking_view( error_message = self._get_or_create_masking_view(
serial_number, masking_view_dict, serial_number, masking_view_dict,
default_sg_name, extra_specs) default_sg_name, extra_specs)
LOG.debug( # Check that the device is in the correct storage group
"The masking view in the attach operation is " if not self._validate_attach(
"%(masking_name)s. The storage group " serial_number, device_id, storagegroup_name,
"in the masking view is %(storage_name)s.", maskingview_name):
{'masking_name': maskingview_name, error_message = ("The attach validation for device "
'storage_name': storagegroup_name}) "%(dev)s was unsuccessful.")
raise exception.VolumeBackendAPIException(
message=error_message)
rollback_dict['portgroup_name'] = ( rollback_dict['portgroup_name'] = (
self.rest.get_element_from_masking_view( self.rest.get_element_from_masking_view(
serial_number, maskingview_name, portgroup=True)) serial_number, maskingview_name, portgroup=True))
@ -131,6 +133,71 @@ class PowerMaxMasking(object):
return rollback_dict return rollback_dict
def _validate_attach(
self, serial_number, device_id, dest_sg_name, dest_mv_name):
"""Validate that the attach was successful.
:param serial_number: the array serial number
:param device_id: the device id
:param dest_sg_name: the correct storage group
:param dest_mv_name: the correct masking view
:returns: bool
"""
sg_list = self.rest.get_storage_groups_from_volume(
serial_number, device_id)
def _validate_masking_view():
mv_list = self.rest.get_masking_views_from_storage_group(
serial_number, dest_sg_name)
if not mv_list:
LOG.error(
"The masking view list of %(sg_name)s where device "
"%(dev)s exists is empty.",
{'sg_name': dest_sg_name,
'dev': device_id})
return False
if dest_mv_name.lower() in (
mv_name.lower() for mv_name in mv_list):
LOG.debug(
"The masking view in the attach operation is "
"%(masking_name)s. The storage group "
"in the masking view is %(storage_name)s. "
"The device id is %(dev)s.",
{'masking_name': dest_mv_name,
'storage_name': dest_sg_name,
'dev': device_id})
return True
else:
LOG.error(
"The masking view %(masking_name)s is not in "
"the masking view list %(mv_list)s. "
"%(sg_name)s is the storage group where device "
"%(dev)s exists.",
{'masking_name': dest_mv_name,
'mv_list': mv_list,
'sg_name': dest_sg_name,
'dev': device_id})
return False
if not sg_list:
LOG.error(
"Device %(dev)s does not exist in any storage group.",
{'dev': device_id})
return False
if dest_sg_name.lower() in (
sg_name.lower() for sg_name in sg_list):
return _validate_masking_view()
else:
LOG.error(
"The storage group %(sg_name)s is not in "
"the storage group list %(sg_list)s "
"where device %(dev)s exists.",
{'sg_name': dest_sg_name,
'sg_list': sg_list,
'dev': device_id})
return False
def _move_vol_from_default_sg( def _move_vol_from_default_sg(
self, serial_number, device_id, volume_name, self, serial_number, device_id, volume_name,
default_sg_name, dest_storagegroup, extra_specs, default_sg_name, dest_storagegroup, extra_specs,
@ -725,6 +792,13 @@ class PowerMaxMasking(object):
{'volume_name': volume_name, {'volume_name': volume_name,
'sg_name': storagegroup_name}) 'sg_name': storagegroup_name})
else: else:
storage_group_list = self.rest.get_storage_groups_from_volume(
serial_number, device_id)
if storage_group_list:
msg = self._check_add_volume_suitability(
serial_number, device_id, volume_name, storagegroup_name)
if msg:
return msg
try: try:
force = True if extra_specs.get(utils.IS_RE) else False force = True if extra_specs.get(utils.IS_RE) else False
self.add_volume_to_storage_group( self.add_volume_to_storage_group(
@ -738,6 +812,42 @@ class PowerMaxMasking(object):
LOG.error(msg) LOG.error(msg)
return msg return msg
def _check_add_volume_suitability(
self, serial_number, device_id, volume_name, add_sg_name):
"""Check if possible to add a volume to a storage group
If a volume already belongs to a storage group that is associated
with FAST it is not possible to add that same volume to another
storage group which is also associated with FAST
:param serial_number: the array serial number
:param device_id: the device id
:param volume_name: the client supplied volume name
:param add_sg_name: storage group that the volume is to be added to
:returns: msg -- str or None
"""
storage_group = self.rest.get_storage_group(
serial_number, add_sg_name)
if storage_group and storage_group.get('slo') and (
storage_group.get('slo').lower() == 'none'):
return None
storage_group_list = self.rest.get_storage_groups_from_volume(
serial_number, device_id)
if storage_group_list:
msg = ("Volume %(vol)s with device id %(dev)s belongs "
"to storage group(s) %(sgs)s. Cannot add volume "
"to another storage group associated with FAST."
% {'vol': volume_name, 'dev': device_id,
'sgs': storage_group_list})
for storage_group_name in storage_group_list:
storage_group = self.rest.get_storage_group(
serial_number, storage_group_name)
if storage_group and storage_group.get('slo') and (
storage_group.get('slo').lower() != 'none'):
LOG.error(msg)
return msg
return None
def add_volume_to_storage_group( def add_volume_to_storage_group(
self, serial_number, device_id, storagegroup_name, self, serial_number, device_id, storagegroup_name,
volume_name, extra_specs, force=False): volume_name, extra_specs, force=False):

View File

@ -119,8 +119,9 @@ class PowerMaxProvision(object):
:param volume_name: the volume name :param volume_name: the volume name
""" """
start_time = time.time() start_time = time.time()
LOG.debug("Delete volume %(volume_name)s from srp.", LOG.debug("Delete volume %(volume_name)s with device id %(dev)s "
{'volume_name': volume_name}) "from srp.", {'volume_name': volume_name,
'dev': device_id})
self.rest.delete_volume(array, device_id) self.rest.delete_volume(array, device_id)
LOG.debug("Delete volume took: %(delta)s H:MM:SS.", LOG.debug("Delete volume took: %(delta)s H:MM:SS.",
{'delta': self.utils.get_time_delta( {'delta': self.utils.get_time_delta(

View File

@ -1224,6 +1224,11 @@ class PowerMaxRest(object):
element_name = self.utils.get_volume_element_name(name_id) element_name = self.utils.get_volume_element_name(name_id)
if vol_identifier == element_name: if vol_identifier == element_name:
found_device_id = device_id found_device_id = device_id
else:
LOG.error("We cannot verify that device %(dev)s was "
"created/managed by openstack by its "
"identifier name.", {'dev': device_id})
return found_device_id return found_device_id
def add_vol_to_sg(self, array, storagegroup_name, device_id, extra_specs, def add_vol_to_sg(self, array, storagegroup_name, device_id, extra_specs,
@ -1412,6 +1417,9 @@ class PowerMaxRest(object):
:returns: volume dict :returns: volume dict
:raises: VolumeBackendAPIException :raises: VolumeBackendAPIException
""" """
if not device_id:
LOG.warning('No device id supplied to get_volume.')
return dict()
version = self.get_uni_version()[1] version = self.get_uni_version()[1]
volume_dict = self.get_resource( volume_dict = self.get_resource(
array, SLOPROVISIONING, 'volume', resource_name=device_id, array, SLOPROVISIONING, 'volume', resource_name=device_id,
@ -1530,6 +1538,9 @@ class PowerMaxRest(object):
:param device_id: the volume device id :param device_id: the volume device id
:param new_name: the new name for the volume, can be None :param new_name: the new name for the volume, can be None
""" """
if not device_id:
LOG.warning('No device id supplied to rename operation.')
return
if new_name is not None: if new_name is not None:
vol_identifier_dict = { vol_identifier_dict = {
"identifier_name": new_name, "identifier_name": new_name,
@ -1547,17 +1558,26 @@ class PowerMaxRest(object):
:param array: the array serial number :param array: the array serial number
:param device_id: volume device id :param device_id: volume device id
""" """
# Check if volume is in any storage group
sg_list = self.get_storage_groups_from_volume(
array, device_id)
if sg_list:
exception_message = (_(
"Device %(device_id)s is in storage group(s) "
"%(sg_list)s prior to delete.")
% {'device_id': device_id, 'sg_list': sg_list})
LOG.error(exception_message)
raise exception.VolumeBackendAPIException(exception_message)
if ((self.ucode_major_level >= utils.UCODE_5978) if ((self.ucode_major_level >= utils.UCODE_5978)
and (self.ucode_minor_level > utils.UCODE_5978_ELMSR)): and (self.ucode_minor_level > utils.UCODE_5978_ELMSR)):
# Use Rapid TDEV Deallocation to delete after ELMSR # Use Rapid TDEV Deallocation to delete after ELMSR
try: try:
# Rename volume, removing the OS-<cinderUUID>
self.rename_volume(array, device_id, None)
self.delete_resource(array, SLOPROVISIONING, self.delete_resource(array, SLOPROVISIONING,
"volume", device_id) "volume", device_id)
except Exception as e: except Exception as e:
LOG.warning('Delete volume failed with %(e)s.', {'e': e}) LOG.warning('Delete volume %(dev)s failed with %(e)s.',
{'dev': device_id, 'e': e})
raise raise
else: else:
# Pre-Foxtail, deallocation and delete are separate calls # Pre-Foxtail, deallocation and delete are separate calls
@ -1569,12 +1589,14 @@ class PowerMaxRest(object):
self._modify_volume(array, device_id, payload) self._modify_volume(array, device_id, payload)
pass pass
except Exception as e: except Exception as e:
LOG.warning('Deallocate volume failed with %(e)s.' LOG.warning('Deallocate volume %(dev)s failed with %(e)s.'
'Attempting delete.', {'e': e}) 'Attempting delete.',
{'dev': device_id, 'e': e})
# Try to delete the volume if deallocate failed. # Try to delete the volume if deallocate failed.
self.delete_resource(array, SLOPROVISIONING, self.delete_resource(array, SLOPROVISIONING,
"volume", device_id) "volume", device_id)
@retry(retry_exc_tuple, interval=2, retries=3)
def find_mv_connections_for_vol(self, array, maskingview, device_id): def find_mv_connections_for_vol(self, array, maskingview, device_id):
"""Find the host_lun_id for a volume in a masking view. """Find the host_lun_id for a volume in a masking view.
@ -1595,16 +1617,35 @@ class PowerMaxRest(object):
'for %(mv)s.', {'mv': maskingview}) 'for %(mv)s.', {'mv': maskingview})
else: else:
try: try:
host_lun_id = ( masking_view_conn = connection_info.get(
connection_info[ 'maskingViewConnection')
'maskingViewConnection'][0]['host_lun_address']) if masking_view_conn and isinstance(
masking_view_conn, list):
host_lun_id = masking_view_conn[0].get(
'host_lun_address')
if host_lun_id:
host_lun_id = int(host_lun_id, 16) host_lun_id = int(host_lun_id, 16)
else:
exception_message = (
_('Unable to get host_lun_address for '
'device %(dev)s on masking view %(mv)s. '
'Retrying...')
% {'dev': device_id, 'mv': maskingview})
LOG.warning(exception_message)
raise exception.VolumeBackendAPIException(
message=exception_message)
except Exception as e: except Exception as e:
LOG.error("Unable to retrieve connection information " exception_message = (
_("Unable to retrieve connection information "
"for volume %(vol)s in masking view %(mv)s. " "for volume %(vol)s in masking view %(mv)s. "
"Exception received: %(e)s.", "Exception received: %(e)s. Retrying...")
{'vol': device_id, 'mv': maskingview, % {'vol': device_id, 'mv': maskingview,
'e': e}) 'e': e})
LOG.warning(exception_message)
raise exception.VolumeBackendAPIException(
message=exception_message)
LOG.debug("The hostlunid is %(hli)s for %(dev)s.",
{'hli': host_lun_id, 'dev': device_id})
return host_lun_id return host_lun_id
def get_storage_groups_from_volume(self, array, device_id): def get_storage_groups_from_volume(self, array, device_id):
@ -1615,7 +1656,16 @@ class PowerMaxRest(object):
:returns: storagegroup_list :returns: storagegroup_list
""" """
sg_list = [] sg_list = []
if not device_id:
return sg_list
vol = self.get_volume(array, device_id) vol = self.get_volume(array, device_id)
if vol and isinstance(vol, list):
LOG.warning(
"Device id %(dev_id)s has brought back "
"multiple volume objects.",
{'vol_name': device_id})
return sg_list
if vol and vol.get('storageGroupId'): if vol and vol.get('storageGroupId'):
sg_list = vol['storageGroupId'] sg_list = vol['storageGroupId']
num_storage_groups = len(sg_list) num_storage_groups = len(sg_list)
@ -1647,13 +1697,19 @@ class PowerMaxRest(object):
""" """
device_id = None device_id = None
params = {"volume_identifier": volume_name} params = {"volume_identifier": volume_name}
device_list = self.get_volume_list(array, params)
volume_list = self.get_volume_list(array, params) if not device_list:
if not volume_list: LOG.debug("Cannot find record for volume %(vol_name)s.",
LOG.debug("Cannot find record for volume %(volumeId)s.", {'vol_name': volume_name})
{'volumeId': volume_name})
else: else:
device_id = volume_list[0] LOG.debug("The device id list is %(dev_list)s for %(vol_name)s.",
{'dev_list': device_list,
'vol_name': volume_name})
device_id = device_list[0] if len(device_list) == 1 else (
device_list)
if isinstance(device_id, list):
LOG.warning("More than one devices returned for %(vol_name)s.",
{'vol_name': volume_name})
return device_id return device_id
def find_volume_identifier(self, array, device_id): def find_volume_identifier(self, array, device_id):
@ -1664,7 +1720,7 @@ class PowerMaxRest(object):
:returns: the volume identifier -- string :returns: the volume identifier -- string
""" """
vol = self.get_volume(array, device_id) vol = self.get_volume(array, device_id)
return vol['volume_identifier'] return vol.get('volume_identifier') if vol else None
def get_size_of_device_on_array(self, array, device_id): def get_size_of_device_on_array(self, array, device_id):
"""Get the size of the volume from the array. """Get the size of the volume from the array.