Merge "Driver assisted migration on retype when it's safe"
This commit is contained in:
commit
a7e98dba5b
@ -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)
|
||||||
|
@ -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,
|
||||||
|
@ -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.
|
Loading…
x
Reference in New Issue
Block a user