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:
parent
bedcb240e6
commit
bdb9e0c5f1
@ -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)
|
||||||
|
|
||||||
|
@ -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:
|
||||||
|
@ -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.
|
Loading…
Reference in New Issue
Block a user