StorPool: fix the retype volume flow
Change-Id: I786733aae17dea47e3e6f2a393a947bfb46e70a3 Closes-Bug: #2002995
This commit is contained in:
parent
1ecfffafa6
commit
c04d5cb892
@ -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([])
|
||||||
|
@ -338,24 +338,24 @@ 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
|
||||||
if v[0] != v[1]:
|
# call us if there's a serious mismatch between the volume types."
|
||||||
# Retype of a volume backend not supported yet,
|
if diff['extra_specs'].get('volume_backend_name'):
|
||||||
# the volume needs to be migrated.
|
v = diff['extra_specs'].get('volume_backend_name')
|
||||||
return False
|
if v[0] != v[1]:
|
||||||
elif k == 'storpool_template':
|
# Retype of a volume backend not supported yet,
|
||||||
if v[0] != v[1]:
|
# the volume needs to be migrated.
|
||||||
if v[1] is not None:
|
|
||||||
update['template'] = v[1]
|
|
||||||
elif templ is not None:
|
|
||||||
update['template'] = templ
|
|
||||||
else:
|
|
||||||
update['replication'] = repl
|
|
||||||
elif v[0] != v[1]:
|
|
||||||
LOG.error('Retype of extra_specs "%s" not '
|
|
||||||
'supported yet.', k)
|
|
||||||
return False
|
return False
|
||||||
|
if diff['extra_specs'].get('storpool_template'):
|
||||||
|
v = diff['extra_specs'].get('storpool_template')
|
||||||
|
if v[0] != v[1]:
|
||||||
|
if v[1] is not None:
|
||||||
|
update['template'] = v[1]
|
||||||
|
elif templ is not None:
|
||||||
|
update['template'] = templ
|
||||||
|
else:
|
||||||
|
update['replication'] = repl
|
||||||
|
|
||||||
if update:
|
if update:
|
||||||
name = self._attach.volumeName(volume['id'])
|
name = self._attach.volumeName(volume['id'])
|
||||||
@ -380,21 +380,46 @@ 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})
|
||||||
return {'_name_id': new_volume['_name_id'] or new_volume['id']}
|
int_name = temp_name + '--temp--mig'
|
||||||
else:
|
LOG.debug('Trying to swap volume names, intermediate "%(int)s"',
|
||||||
|
{'int': int_name})
|
||||||
try:
|
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,
|
self._attach.api().volumeUpdate(temp_name,
|
||||||
{'rename': orig_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}
|
return {'_name_id': None}
|
||||||
except spapi.ApiError as e:
|
except spapi.ApiError as e:
|
||||||
LOG.error('StorPool update_migrated_volume(): '
|
LOG.error('StorPool update_migrated_volume(): '
|
||||||
'could not rename %(tname)s to %(oname)s: '
|
'could not rename a volume: '
|
||||||
'%(err)s',
|
'%(err)s',
|
||||||
{'tname': temp_name, 'oname': orig_name, 'err': e})
|
{'err': e})
|
||||||
return {'_name_id': new_volume['_name_id'] or new_volume['id']}
|
return {'_name_id': new_volume['_name_id'] or new_volume['id']}
|
||||||
|
|
||||||
|
try:
|
||||||
|
self._attach.api().volumeUpdate(temp_name,
|
||||||
|
{'rename': orig_name})
|
||||||
|
return {'_name_id': None}
|
||||||
|
except spapi.ApiError as e:
|
||||||
|
LOG.error('StorPool update_migrated_volume(): '
|
||||||
|
'could not rename %(tname)s to %(oname)s: '
|
||||||
|
'%(err)s',
|
||||||
|
{'tname': temp_name, 'oname': orig_name, 'err': e})
|
||||||
|
return {'_name_id': new_volume['_name_id'] or new_volume['id']}
|
||||||
|
9
releasenotes/notes/bug2002995-e423f17eaddae22d.yaml
Normal file
9
releasenotes/notes/bug2002995-e423f17eaddae22d.yaml
Normal 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.
|
Loading…
x
Reference in New Issue
Block a user