Merge "Driver assisted migration on retype when it's safe"

This commit is contained in:
Zuul 2021-07-02 22:36:26 +00:00 committed by Gerrit Code Review
commit a7e98dba5b
3 changed files with 99 additions and 3 deletions

View File

@ -105,6 +105,52 @@ class VolumeMigrationTestCase(base.BaseVolumeTestCase):
self.assertEqual('newhost', volume.host) self.assertEqual('newhost', volume.host)
self.assertEqual('success', volume.migration_status) self.assertEqual('success', volume.migration_status)
@mock.patch('cinder.volume.manager.VolumeManager.'
'_can_use_driver_migration')
def test_migrate_volume_driver_for_retype(self, mock_can_use):
"""Test volume migration done by driver on a retype."""
# Mock driver and rpc functions
mock_driver = self.mock_object(self.volume.driver, 'migrate_volume',
return_value=(True, {}))
volume = tests_utils.create_volume(self.context, size=0,
host=CONF.host,
migration_status='migrating')
host_obj = {'host': 'newhost', 'capabilities': {}}
self.volume.migrate_volume(self.context, volume, host_obj, False,
fake.VOLUME_TYPE2_ID, mock.sentinel.diff)
mock_can_use.assert_called_once_with(mock.sentinel.diff)
mock_driver.assert_called_once_with(self.context, volume, host_obj)
# check volume properties
volume = objects.Volume.get_by_id(context.get_admin_context(),
volume.id)
self.assertEqual('newhost', volume.host)
self.assertEqual('success', volume.migration_status)
self.assertEqual(fake.VOLUME_TYPE2_ID, volume.volume_type_id)
@mock.patch('cinder.volume.manager.VolumeManager._migrate_volume_generic')
@mock.patch('cinder.volume.manager.VolumeManager.'
'_can_use_driver_migration')
def test_migrate_volume_driver_for_retype_generic(self, mock_can_use,
mock_generic):
"""Test generic volume migration on a retype after driver can't."""
# Mock driver and rpc functions
mock_driver = self.mock_object(self.volume.driver, 'migrate_volume',
return_value=(False, None))
volume = tests_utils.create_volume(self.context, size=0,
host=CONF.host,
migration_status='migrating')
host_obj = {'host': 'newhost', 'capabilities': {}}
self.volume.migrate_volume(self.context, volume, host_obj, False,
fake.VOLUME_TYPE2_ID, mock.sentinel.diff)
mock_can_use.assert_called_once_with(mock.sentinel.diff)
mock_driver.assert_called_once_with(self.context, volume, host_obj)
mock_generic.assert_called_once_with(self.context, volume, host_obj,
fake.VOLUME_TYPE2_ID)
def test_migrate_volume_driver_cross_az(self): def test_migrate_volume_driver_cross_az(self):
"""Test volume migration done by driver.""" """Test volume migration done by driver."""
# Mock driver and rpc functions # Mock driver and rpc functions
@ -1015,3 +1061,21 @@ class VolumeMigrationTestCase(base.BaseVolumeTestCase):
self.volume.retype(self.context, volume, new_vol_type.id, self.volume.retype(self.context, volume, new_vol_type.id,
host_obj, migration_policy='on-demand') host_obj, migration_policy='on-demand')
vt_get.assert_not_called() vt_get.assert_not_called()
@ddt.data(
(None, True),
({'encryption': {'cipher': ('v1', 'v2')}}, False),
({'qos_specs': {'key1': ('v1', 'v2')}}, False),
({'encryption': {}, 'qos_specs': {}, 'extra_specs': {}}, True),
({'encryption': {}, 'qos_specs': {},
'extra_specs': {'volume_backend_name': ('ceph1', 'ceph2'),
'RESKEY:availability_zones': ('nova', 'nova2')}},
True),
({'encryption': {}, 'qos_specs': {},
'extra_specs': {'thin_provisioning_support': ('<is> True', None)}},
False),
)
@ddt.unpack
def test__can_use_driver_migration(self, diff, expected):
res = self.volume._can_use_driver_migration(diff)
self.assertEqual(expected, res)

View File

@ -2553,12 +2553,35 @@ class VolumeManager(manager.CleanableManager,
resource=volume) resource=volume)
return volume.id return volume.id
def _can_use_driver_migration(self, diff):
"""Return when we can use driver assisted migration on a retype."""
# We can if there's no retype or there are no difference in the types
if not diff:
return True
extra_specs = diff.get('extra_specs')
qos = diff.get('qos_specs')
enc = diff.get('encryption')
# We cant' if QoS or Encryption changes and we can if there are no
# extra specs changes.
if qos or enc or not extra_specs:
return not (qos or enc)
# We can use driver assisted migration if we only change the backend
# name, and the AZ.
extra_specs = extra_specs.copy()
extra_specs.pop('volume_backend_name', None)
extra_specs.pop('RESKEY:availability_zones', None)
return not extra_specs
def migrate_volume(self, def migrate_volume(self,
ctxt: context.RequestContext, ctxt: context.RequestContext,
volume, volume,
host, host,
force_host_copy: bool = False, force_host_copy: bool = False,
new_type_id=None) -> None: new_type_id=None,
diff=None) -> None:
"""Migrate the volume to the specified host (called on source host).""" """Migrate the volume to the specified host (called on source host)."""
try: try:
# NOTE(flaper87): Verify the driver is enabled # NOTE(flaper87): Verify the driver is enabled
@ -2579,7 +2602,7 @@ class VolumeManager(manager.CleanableManager,
volume.migration_status = 'migrating' volume.migration_status = 'migrating'
volume.save() volume.save()
if not force_host_copy and new_type_id is None: if not force_host_copy and self._can_use_driver_migration(diff):
try: try:
LOG.debug("Issue driver.migrate_volume.", resource=volume) LOG.debug("Issue driver.migrate_volume.", resource=volume)
moved, model_update = self.driver.migrate_volume(ctxt, moved, model_update = self.driver.migrate_volume(ctxt,
@ -2598,6 +2621,8 @@ class VolumeManager(manager.CleanableManager,
updates.update(status_update) updates.update(status_update)
if model_update: if model_update:
updates.update(model_update) updates.update(model_update)
if new_type_id:
updates['volume_type_id'] = new_type_id
volume.update(updates) volume.update(updates)
volume.save() volume.save()
except Exception: except Exception:
@ -3047,7 +3072,7 @@ class VolumeManager(manager.CleanableManager,
try: try:
self.migrate_volume(context, volume, host, self.migrate_volume(context, volume, host,
new_type_id=new_type_id) new_type_id=new_type_id, diff=diff)
except Exception: except Exception:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
_retype_error(context, volume, old_reservations, _retype_error(context, volume, old_reservations,

View File

@ -0,0 +1,7 @@
---
fixes:
- |
`Bug #1886543 <https://bugs.launchpad.net/cinder/+bug/1886543>`_:
On retypes requiring a migration, try to use the driver assisted mechanism
when moving from one backend to another when we know it's safe from the
volume type perspective.