StorPool: fix the retype volume flow

Change-Id: I786733aae17dea47e3e6f2a393a947bfb46e70a3
Closes-Bug: #2002995
This commit is contained in:
Peter Penchev 2022-12-14 17:55:56 +02:00 committed by Kiran Pawar
parent 1ecfffafa6
commit c04d5cb892
3 changed files with 66 additions and 29 deletions

View File

@ -356,12 +356,6 @@ class StorPoolTestCase(test.TestCase):
'available') 'available')
self.assertDictEqual({'_name_id': '1'}, res) self.assertDictEqual({'_name_id': '1'}, res)
# Failure: a volume with the original volume's name already exists
res = self.driver.update_migrated_volume(None, {'id': '1'},
{'id': '2', '_name_id': '1'},
'available')
self.assertDictEqual({'_name_id': '1'}, res)
# Success: rename the migrated volume to match the original # Success: rename the migrated volume to match the original
res = self.driver.update_migrated_volume(None, {'id': '3'}, res = self.driver.update_migrated_volume(None, {'id': '3'},
{'id': '2', '_name_id': '3'}, {'id': '2', '_name_id': '3'},
@ -371,6 +365,15 @@ class StorPoolTestCase(test.TestCase):
volumes.keys()) volumes.keys())
self.assertVolumeNames(('1', '3',)) self.assertVolumeNames(('1', '3',))
# Success: swap volume names with an existing volume
res = self.driver.update_migrated_volume(None, {'id': '1'},
{'id': '3', '_name_id': '1'},
'available')
self.assertDictEqual({'_name_id': None}, res)
self.assertCountEqual([volumeName('1'), volumeName('3')],
volumes.keys())
self.assertVolumeNames(('1', '3',))
for vid in ('1', '3'): for vid in ('1', '3'):
self.driver.delete_volume({'id': vid}) self.driver.delete_volume({'id': vid})
self.assertVolumeNames([]) self.assertVolumeNames([])

View File

@ -338,13 +338,17 @@ class StorPoolDriver(driver.VolumeDriver):
templ = self.configuration.storpool_template templ = self.configuration.storpool_template
repl = self.configuration.storpool_replication repl = self.configuration.storpool_replication
if diff['extra_specs']: if diff['extra_specs']:
for (k, v) in diff['extra_specs'].items(): # Check for the StorPool extra specs. We intentionally ignore any
if k == 'volume_backend_name': # other extra_specs because the cinder scheduler should not even
# call us if there's a serious mismatch between the volume types."
if diff['extra_specs'].get('volume_backend_name'):
v = diff['extra_specs'].get('volume_backend_name')
if v[0] != v[1]: if v[0] != v[1]:
# Retype of a volume backend not supported yet, # Retype of a volume backend not supported yet,
# the volume needs to be migrated. # the volume needs to be migrated.
return False return False
elif k == 'storpool_template': if diff['extra_specs'].get('storpool_template'):
v = diff['extra_specs'].get('storpool_template')
if v[0] != v[1]: if v[0] != v[1]:
if v[1] is not None: if v[1] is not None:
update['template'] = v[1] update['template'] = v[1]
@ -352,10 +356,6 @@ class StorPoolDriver(driver.VolumeDriver):
update['template'] = templ update['template'] = templ
else: else:
update['replication'] = repl update['replication'] = repl
elif v[0] != v[1]:
LOG.error('Retype of extra_specs "%s" not '
'supported yet.', k)
return False
if update: if update:
name = self._attach.volumeName(volume['id']) name = self._attach.volumeName(volume['id'])
@ -380,14 +380,39 @@ class StorPoolDriver(driver.VolumeDriver):
'created as part of the migration from ' 'created as part of the migration from '
'"%(oid)s".', {'tid': temp_id, 'oid': orig_id}) '"%(oid)s".', {'tid': temp_id, 'oid': orig_id})
return {'_name_id': new_volume['_name_id'] or new_volume['id']} return {'_name_id': new_volume['_name_id'] or new_volume['id']}
elif orig_name in vols:
LOG.error('StorPool update_migrated_volume(): both ' if orig_name in vols:
LOG.debug('StorPool update_migrated_volume(): both '
'the original volume "%(oid)s" and the migrated ' 'the original volume "%(oid)s" and the migrated '
'StorPool volume "%(tid)s" seem to exist on ' 'StorPool volume "%(tid)s" seem to exist on '
'the StorPool cluster.', 'the StorPool cluster.',
{'oid': orig_id, 'tid': temp_id}) {'oid': orig_id, 'tid': temp_id})
int_name = temp_name + '--temp--mig'
LOG.debug('Trying to swap volume names, intermediate "%(int)s"',
{'int': int_name})
try:
LOG.debug('- rename "%(orig)s" to "%(int)s"',
{'orig': orig_name, 'int': int_name})
self._attach.api().volumeUpdate(orig_name,
{'rename': int_name})
LOG.debug('- rename "%(temp)s" to "%(orig)s"',
{'temp': temp_name, 'orig': orig_name})
self._attach.api().volumeUpdate(temp_name,
{'rename': orig_name})
LOG.debug('- rename "%(int)s" to "%(temp)s"',
{'int': int_name, 'temp': temp_name})
self._attach.api().volumeUpdate(int_name,
{'rename': temp_name})
return {'_name_id': None}
except spapi.ApiError as e:
LOG.error('StorPool update_migrated_volume(): '
'could not rename a volume: '
'%(err)s',
{'err': e})
return {'_name_id': new_volume['_name_id'] or new_volume['id']} return {'_name_id': new_volume['_name_id'] or new_volume['id']}
else:
try: try:
self._attach.api().volumeUpdate(temp_name, self._attach.api().volumeUpdate(temp_name,
{'rename': orig_name}) {'rename': orig_name})

View File

@ -0,0 +1,9 @@
---
fixes:
- |
StorPool driver `bug #2002995
<https://bugs.launchpad.net/cinder/+bug/2002995>`_: When retyping a
volume on a StorPool backend to a different volume type also on that
StorPool backend but using a different StorPool template, occasionally
the retype operation would fail or the old volume could be left attached
to a StorPool client. This issue has been fixed in this release.