Delete volume with additional removehostmappings parameter

In some scenarios, User can't delete volume once this volume has host
mapping at Spectrum Virtualize Family storage side.For example: User
destory server instance without detaching disk. With this patch user
can delete this kind of volume directly with cinder driver instead of
deleting this volume manually with storage GUI/TUI. This parameter is
only available while storage code_level is newer than 7.7.0.0.

Change-Id: I8acfdb2110d81a4b3d91060ea2deed122b21b61b
Closes-bug: #1855719
This commit is contained in:
Xuchu Jiang 2020-03-30 13:23:35 +08:00
parent bedcb240e6
commit bdb9e0c5f1
3 changed files with 142 additions and 32 deletions

View File

@ -999,6 +999,11 @@ port_speed!N/A
def _cmd_rmvdisk(self, **kwargs): def _cmd_rmvdisk(self, **kwargs):
force = True if 'force' in kwargs else False force = True if 'force' in kwargs else False
if 'force' not in kwargs and 'force_unmap' in kwargs:
force_unmap = True
else:
force_unmap = False
if 'obj' not in kwargs: if 'obj' not in kwargs:
return self._errors['CMMVC5701E'] return self._errors['CMMVC5701E']
vol_name = kwargs['obj'].strip('\'\"') vol_name = kwargs['obj'].strip('\'\"')
@ -1006,7 +1011,7 @@ port_speed!N/A
if vol_name not in self._volumes_list: if vol_name not in self._volumes_list:
return self._errors['CMMVC5753E'] return self._errors['CMMVC5753E']
if not force: if not force and not force_unmap:
for mapping in self._mappings_list.values(): for mapping in self._mappings_list.values():
if mapping['vol'] == vol_name: if mapping['vol'] == vol_name:
return self._errors['CMMVC5840E'] return self._errors['CMMVC5840E']
@ -4958,7 +4963,7 @@ class StorwizeSVCCommonDriverTestCase(test.TestCase):
'mirror_pool': None, 'mirror_pool': None,
'volume_topology': None, 'volume_topology': None,
'peer_pool': None, 'peer_pool': None,
'cycle_period_seconds': 300, 'cycle_period_seconds': 300
} }
return opt return opt
@ -8075,6 +8080,56 @@ class StorwizeSVCCommonDriverTestCase(test.TestCase):
self.ctxt, self.ctxt,
vol1, snap1) vol1, snap1)
@mock.patch.object(storwize_svc_common.StorwizeHelpers, 'delete_vdisk')
def test_storwize_svc_delete_volume_with_lower_code(self, delete_vdisk):
with mock.patch.object(storwize_svc_common.StorwizeHelpers,
'get_system_info') as get_system_info:
fake_system_info = {'code_level': (7, 6, 0, 0),
'topology': 'standard',
'system_name': 'storwize-svc-sim',
'system_id': '0123456789ABCDEF'}
get_system_info.return_value = fake_system_info
self.driver.do_setup(None)
volume = self._generate_vol_info()
snap = self._generate_snap_info(volume.id)
self.driver.create_volume(volume)
self.driver.create_snapshot(snap)
self.driver.delete_snapshot(snap)
snap_call = [mock.call(snap.name, force_delete=False,
force_unmap=False)]
delete_vdisk.assert_has_calls(snap_call)
self.driver.delete_volume(volume)
vol_call = [mock.call(volume.name, force_delete=False,
force_unmap=False)]
delete_vdisk.assert_has_calls(vol_call)
@mock.patch.object(storwize_svc_common.StorwizeHelpers, 'delete_vdisk')
@mock.patch('cinder.volume.volume_utils.is_group_a_cg_snapshot_type')
@mock.patch('cinder.volume.volume_utils.is_group_a_type')
@mock.patch.object(storwize_svc_common.StorwizeSVCCommonDriver,
'_delete_replication_grp')
def test_storwize_delete_group_with_lower_code(
self, _del_rep_grp, is_grp_a_cg_rep_type,
is_grp_a_cg_snapshot_type, delete_vdisk):
is_grp_a_cg_snapshot_type.return_valume = True
is_grp_a_cg_rep_type.return_value = False
type_ref = volume_types.create(self.ctxt, 'testtype', None)
group = testutils.create_group(self.ctxt,
group_type_id=fake.GROUP_TYPE_ID,
volume_type_ids=[type_ref['id']])
vol1 = self._create_volume(volume_type_id=type_ref['id'],
group_id=group.id)
vol2 = self._create_volume(volume_type_id=type_ref['id'],
group_id=group.id)
volumes = self.db.volume_get_all_by_generic_group(
self.ctxt.elevated(), group.id)
self.driver.delete_group(self.ctxt, group, volumes)
calls = [mock.call(vol1.name, force_unmap=False, force_delete=True),
mock.call(vol2.name, force_unmap=False, force_delete=True)]
delete_vdisk.assert_has_calls(calls, any_order=True)
class CLIResponseTestCase(test.TestCase): class CLIResponseTestCase(test.TestCase):
def test_empty(self): def test_empty(self):
@ -9456,6 +9511,7 @@ class StorwizeSVCReplicationTestCase(test.TestCase):
delete_relationship, delete_relationship,
delete_vdisk): delete_vdisk):
# Set replication target. # Set replication target.
self.driver.configuration.set_override('replication_device', self.driver.configuration.set_override('replication_device',
[self.rep_target]) [self.rep_target])
self.driver.do_setup(self.ctxt) self.driver.do_setup(self.ctxt)
@ -9467,8 +9523,9 @@ class StorwizeSVCReplicationTestCase(test.TestCase):
delete_relationship.assert_called_once_with(fake_name) delete_relationship.assert_called_once_with(fake_name)
master_change_fake_name = ( master_change_fake_name = (
storwize_const.REPLICA_CHG_VOL_PREFIX + fake_name) storwize_const.REPLICA_CHG_VOL_PREFIX + fake_name)
calls = [mock.call(master_change_fake_name, False), calls = [mock.call(master_change_fake_name, force_delete=False,
mock.call(fake_name, False)] force_unmap=True),
mock.call(fake_name, force_delete=False, force_unmap=True)]
delete_vdisk.assert_has_calls(calls, any_order=True) delete_vdisk.assert_has_calls(calls, any_order=True)
self.assertEqual(2, delete_vdisk.call_count) self.assertEqual(2, delete_vdisk.call_count)

View File

@ -501,9 +501,11 @@ class StorwizeSSH(object):
LOG.exception('Failed to create vdisk %(vol)s.', LOG.exception('Failed to create vdisk %(vol)s.',
{'vol': name}) {'vol': name})
def rmvdisk(self, vdisk, force=True): def rmvdisk(self, vdisk, force_unmap=True, force_delete=True):
ssh_cmd = ['svctask', 'rmvdisk'] ssh_cmd = ['svctask', 'rmvdisk']
if force: if force_unmap and not force_delete:
ssh_cmd += ['-removehostmappings']
if force_delete:
ssh_cmd += ['-force'] ssh_cmd += ['-force']
ssh_cmd += ['"%s"' % vdisk] ssh_cmd += ['"%s"' % vdisk]
self.run_ssh_assert_no_output(ssh_cmd) self.run_ssh_assert_no_output(ssh_cmd)
@ -710,11 +712,13 @@ class StorwizeSSH(object):
'"%s"' % pool, '-size', size, '-unit', units] + params '"%s"' % pool, '-size', size, '-unit', units] + params
return self.run_ssh_check_created(ssh_cmd) return self.run_ssh_check_created(ssh_cmd)
def rmvolume(self, volume, force=True): def rmvolume(self, volume, force_unmap=True, force_delete=True):
ssh_cmd = ['svctask', 'rmvolume'] ssh_cmd = ['svctask', 'rmvolume']
if force: if force_delete:
ssh_cmd += ['-removehostmappings', '-removefcmaps', ssh_cmd += ['-removehostmappings', '-removefcmaps',
'-removercrelationships'] '-removercrelationships']
elif force_unmap:
ssh_cmd += ['-removehostmappings']
ssh_cmd += ['"%s"' % volume] ssh_cmd += ['"%s"' % volume]
self.run_ssh_assert_no_output(ssh_cmd) self.run_ssh_assert_no_output(ssh_cmd)
@ -796,6 +800,7 @@ class StorwizeHelpers(object):
msg = _('Failed to get code level (%s).') % level msg = _('Failed to get code level (%s).') % level
raise exception.VolumeBackendAPIException(data=msg) raise exception.VolumeBackendAPIException(data=msg)
code_level = match_obj.group().split('.') code_level = match_obj.group().split('.')
LOG.info("code_level is: %s.", level)
return {'code_level': tuple([int(x) for x in code_level]), return {'code_level': tuple([int(x) for x in code_level]),
'topology': resp['topology'], 'topology': resp['topology'],
'system_name': resp['name'], 'system_name': resp['name'],
@ -1699,14 +1704,16 @@ class StorwizeHelpers(object):
hyper_pool = '%s' % peer_pool hyper_pool = '%s' % peer_pool
self.ssh.rmvolumecopy(vol_name, hyper_pool) self.ssh.rmvolumecopy(vol_name, hyper_pool)
def delete_hyperswap_volume(self, volume, force): def delete_hyperswap_volume(self, volume, force_unmap, force_delete):
"""Ensures that vdisk is not part of FC mapping and deletes it.""" """Ensures that vdisk is not part of FC mapping and deletes it."""
if not self.is_vdisk_defined(volume): if not self.is_vdisk_defined(volume):
LOG.warning('Tried to delete non-existent volume %s.', volume) LOG.warning('Tried to delete non-existent volume %s.', volume)
return return
self.ensure_vdisk_no_fc_mappings(volume, allow_snaps=True, self.ensure_vdisk_no_fc_mappings(volume, allow_snaps=True,
allow_fctgt=True) allow_fctgt=True)
self.ssh.rmvolume(volume, force=force) self.ssh.rmvolume(volume,
force_unmap=force_unmap,
force_delete=force_delete)
def get_vdisk_attributes(self, vdisk): def get_vdisk_attributes(self, vdisk):
attrs = self.ssh.lsvdisk(vdisk) attrs = self.ssh.lsvdisk(vdisk)
@ -1844,7 +1851,9 @@ class StorwizeHelpers(object):
try: try:
for snapshot in snapshots: for snapshot in snapshots:
self.delete_vdisk(snapshot['name'], True) self.delete_vdisk(snapshot['name'],
force_unmap=False,
force_delete=True)
except exception.VolumeBackendAPIException as err: except exception.VolumeBackendAPIException as err:
model_update['status'] = ( model_update['status'] = (
fields.GroupSnapshotStatus.ERROR_DELETING) fields.GroupSnapshotStatus.ERROR_DELETING)
@ -2185,7 +2194,8 @@ class StorwizeHelpers(object):
relationship = self.ssh.lsrcrelationship(vol_attrs['RC_name']) relationship = self.ssh.lsrcrelationship(vol_attrs['RC_name'])
return relationship[0] if len(relationship) > 0 else None return relationship[0] if len(relationship) > 0 else None
def delete_rc_volume(self, volume_name, target_vol=False): def delete_rc_volume(self, volume_name, target_vol=False,
force_unmap=True):
vol_name = volume_name vol_name = volume_name
if target_vol: if target_vol:
vol_name = storwize_const.REPLICA_AUX_VOL_PREFIX + volume_name vol_name = storwize_const.REPLICA_AUX_VOL_PREFIX + volume_name
@ -2196,8 +2206,12 @@ class StorwizeHelpers(object):
self.delete_relationship(vol_name) self.delete_relationship(vol_name)
# Delete change volume # Delete change volume
self.delete_vdisk( self.delete_vdisk(
storwize_const.REPLICA_CHG_VOL_PREFIX + vol_name, False) storwize_const.REPLICA_CHG_VOL_PREFIX + vol_name,
self.delete_vdisk(vol_name, False) force_unmap=force_unmap,
force_delete=False)
self.delete_vdisk(vol_name,
force_unmap=force_unmap,
force_delete=False)
except Exception as e: except Exception as e:
msg = (_('Unable to delete the volume for ' msg = (_('Unable to delete the volume for '
'volume %(vol)s. Exception: %(err)s.'), 'volume %(vol)s. Exception: %(err)s.'),
@ -2273,7 +2287,7 @@ class StorwizeHelpers(object):
def chpartnership(self, partnership_id): def chpartnership(self, partnership_id):
self.ssh.chpartnership(partnership_id) self.ssh.chpartnership(partnership_id)
def delete_vdisk(self, vdisk, force): def delete_vdisk(self, vdisk, force_unmap, force_delete):
"""Ensures that vdisk is not part of FC mapping and deletes it.""" """Ensures that vdisk is not part of FC mapping and deletes it."""
LOG.debug('Enter: delete_vdisk: vdisk %s.', vdisk) LOG.debug('Enter: delete_vdisk: vdisk %s.', vdisk)
if not self.is_vdisk_defined(vdisk): if not self.is_vdisk_defined(vdisk):
@ -2281,7 +2295,9 @@ class StorwizeHelpers(object):
return return
self.ensure_vdisk_no_fc_mappings(vdisk, allow_snaps=True, self.ensure_vdisk_no_fc_mappings(vdisk, allow_snaps=True,
allow_fctgt=True) allow_fctgt=True)
self.ssh.rmvdisk(vdisk, force=force) self.ssh.rmvdisk(vdisk,
force_unmap=force_unmap,
force_delete=force_delete)
LOG.debug('Leave: delete_vdisk: vdisk %s.', vdisk) LOG.debug('Leave: delete_vdisk: vdisk %s.', vdisk)
def create_copy(self, src, tgt, src_id, config, opts, def create_copy(self, src, tgt, src_id, config, opts,
@ -2311,7 +2327,7 @@ class StorwizeHelpers(object):
full_copy=full_copy) full_copy=full_copy)
except Exception: except Exception:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
self.delete_vdisk(tgt, True) self.delete_vdisk(tgt, force_unmap=False, force_delete=True)
LOG.debug('Leave: _create_copy: snapshot %(tgt)s from ' LOG.debug('Leave: _create_copy: snapshot %(tgt)s from '
'vdisk %(src)s.', 'vdisk %(src)s.',
@ -3098,27 +3114,33 @@ class StorwizeSVCCommonDriver(san.SanDriver,
def delete_volume(self, volume): def delete_volume(self, volume):
LOG.debug('enter: delete_volume: volume %s', volume['name']) LOG.debug('enter: delete_volume: volume %s', volume['name'])
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
if self._state['code_level'] < (7, 7, 0, 0):
force_unmap = False
else:
force_unmap = True
hyper_volume = self.is_volume_hyperswap(volume) hyper_volume = self.is_volume_hyperswap(volume)
if hyper_volume: if hyper_volume:
LOG.debug('Volume %s to be deleted is a hyperswap ' LOG.debug('Volume %s to be deleted is a hyperswap '
'volume.', volume.name) 'volume.', volume.name)
self._helpers.delete_hyperswap_volume(volume.name, False) self._helpers.delete_hyperswap_volume(volume.name,
force_unmap=force_unmap,
force_delete=False)
return return
rep_type = self._get_volume_replicated_type(ctxt, volume) rep_type = self._get_volume_replicated_type(ctxt, volume)
if rep_type: if rep_type:
if self._aux_backend_helpers: if self._aux_backend_helpers:
self._aux_backend_helpers.delete_rc_volume(volume['name'], self._aux_backend_helpers.delete_rc_volume(
target_vol=True) volume['name'], target_vol=True, force_unmap=force_unmap)
if not self._active_backend_id: if not self._active_backend_id:
self._master_backend_helpers.delete_rc_volume(volume['name']) self._master_backend_helpers.delete_rc_volume(
volume['name'], force_unmap=force_unmap)
else: else:
# If it's in fail over state, also try to delete the volume # If it's in fail over state, also try to delete the volume
# in master backend # in master backend
try: try:
self._master_backend_helpers.delete_rc_volume( self._master_backend_helpers.delete_rc_volume(
volume['name']) volume['name'], force_unmap=force_unmap)
except Exception as ex: except Exception as ex:
LOG.error('Failed to get delete volume %(volume)s in ' LOG.error('Failed to get delete volume %(volume)s in '
'master backend. Exception: %(err)s.', 'master backend. Exception: %(err)s.',
@ -3131,7 +3153,10 @@ class StorwizeSVCCommonDriver(san.SanDriver,
LOG.error(msg) LOG.error(msg)
raise exception.VolumeDriverException(message=msg) raise exception.VolumeDriverException(message=msg)
else: else:
self._helpers.delete_vdisk(volume['name'], False) self._helpers.delete_vdisk(
volume['name'],
force_unmap=force_unmap,
force_delete=False)
if volume['id'] in self._vdiskcopyops: if volume['id'] in self._vdiskcopyops:
del self._vdiskcopyops[volume['id']] del self._vdiskcopyops[volume['id']]
@ -3175,7 +3200,12 @@ class StorwizeSVCCommonDriver(san.SanDriver,
opts, False, self._state, pool=pool) opts, False, self._state, pool=pool)
def delete_snapshot(self, snapshot): def delete_snapshot(self, snapshot):
self._helpers.delete_vdisk(snapshot['name'], False) if self._state['code_level'] < (7, 7, 0, 0):
force_unmap = False
else:
force_unmap = True
self._helpers.delete_vdisk(
snapshot['name'], force_unmap=force_unmap, force_delete=False)
def create_volume_from_snapshot(self, volume, snapshot): def create_volume_from_snapshot(self, volume, snapshot):
# Create volume from snapshot with a replication or hyperswap group_id # Create volume from snapshot with a replication or hyperswap group_id
@ -4853,13 +4883,19 @@ class StorwizeSVCCommonDriver(san.SanDriver,
new_opts['flashcopy_rate']) new_opts['flashcopy_rate'])
# Delete replica if needed # Delete replica if needed
if self._state['code_level'] < (7, 7, 0, 0):
force_unmap = False
else:
force_unmap = True
if old_rep_type and not new_rep_type: if old_rep_type and not new_rep_type:
self._aux_backend_helpers.delete_rc_volume(volume['name'], self._aux_backend_helpers.delete_rc_volume(volume['name'],
target_vol=True) target_vol=True,
force_unmap=force_unmap)
if storwize_const.GMCV == old_rep_type: if storwize_const.GMCV == old_rep_type:
self._helpers.delete_vdisk( self._helpers.delete_vdisk(
storwize_const.REPLICA_CHG_VOL_PREFIX + volume['name'], storwize_const.REPLICA_CHG_VOL_PREFIX + volume['name'],
False) force_unmap=force_unmap, force_delete=False)
model_update = {'replication_status': model_update = {'replication_status':
fields.ReplicationStatus.DISABLED, fields.ReplicationStatus.DISABLED,
'replication_driver_data': None, 'replication_driver_data': None,
@ -5327,7 +5363,10 @@ class StorwizeSVCCommonDriver(san.SanDriver,
else: else:
for volume in volumes: for volume in volumes:
try: try:
self._helpers.delete_vdisk(volume.name, True) self._helpers.delete_vdisk(
volume['name'],
force_unmap=False,
force_delete=True)
volumes_model_update.append( volumes_model_update.append(
{'id': volume.id, 'status': 'deleted'}) {'id': volume.id, 'status': 'deleted'})
except exception.VolumeBackendAPIException as err: except exception.VolumeBackendAPIException as err:
@ -5661,6 +5700,10 @@ class StorwizeSVCCommonDriver(san.SanDriver,
return vdisk return vdisk
def _delete_replication_grp(self, group, volumes): def _delete_replication_grp(self, group, volumes):
if self._state['code_level'] < (7, 7, 0, 0):
force_unmap = False
else:
force_unmap = True
model_update = {'status': fields.GroupStatus.DELETED} model_update = {'status': fields.GroupStatus.DELETED}
volumes_model_update = [] volumes_model_update = []
rccg_name = self._get_rccg_name(group) rccg_name = self._get_rccg_name(group)
@ -5674,9 +5717,10 @@ class StorwizeSVCCommonDriver(san.SanDriver,
for volume in volumes: for volume in volumes:
try: try:
self._master_backend_helpers.delete_rc_volume(volume.name) self._master_backend_helpers.delete_rc_volume(
self._aux_backend_helpers.delete_rc_volume(volume.name, volume.name, force_unmap=force_unmap)
target_vol=True) self._aux_backend_helpers.delete_rc_volume(
volume.name, target_vol=True, force_unmap=force_unmap)
volumes_model_update.append( volumes_model_update.append(
{'id': volume.id, 'status': 'deleted'}) {'id': volume.id, 'status': 'deleted'})
except exception.VolumeDriverException as err: except exception.VolumeDriverException as err:
@ -5796,7 +5840,9 @@ class StorwizeSVCCommonDriver(san.SanDriver,
for volume in volumes: for volume in volumes:
try: try:
self._helpers.delete_hyperswap_volume(volume.name, True) self._helpers.delete_hyperswap_volume(volume.name,
force_unmap=False,
force_delete=True)
volumes_model_update.append( volumes_model_update.append(
{'id': volume.id, 'status': 'deleted'}) {'id': volume.id, 'status': 'deleted'})
except exception.VolumeDriverException as err: except exception.VolumeDriverException as err:

View File

@ -0,0 +1,7 @@
---
upgrade:
- |
IBM Spectrum Virtualize Family (previously known as Storwize) driver
cannot delete volume which has host mapping in some rare cases while
code_level of IBM Spectrum Virtualize Family storage lower than
7.7.0.0. Please upgrade to latest code to avoid this kind of issue.