RBD: Fix _show_msg_check_clone_v2_api

There might be a case where the exception raised by `op_features`
could be other than AttributeError. When any exception other than
AttributeError is raised, we should log and avoid raising an
exception.

Closes-Bug: #1942210
Co-Authored-By: Gorka Eguileor <geguileo@redhat.com>
Change-Id: I513abe980b73d7e7b1a3cd9c7ff89490f7fd6b08
This commit is contained in:
Sofia Enriquez 2021-02-23 00:24:55 +00:00
parent ef9a3d6299
commit 98acb82221
3 changed files with 47 additions and 10 deletions

View File

@ -117,6 +117,9 @@ def common_mocks(f):
inst.mock_rbd.PermissionError = MockPermissionError inst.mock_rbd.PermissionError = MockPermissionError
inst.driver.rbd = inst.mock_rbd inst.driver.rbd = inst.mock_rbd
aux = inst.driver.rbd
aux.Image.return_value.stripe_unit.return_value = 4194304
inst.driver.rados = inst.mock_rados inst.driver.rados = inst.mock_rados
return f(inst, *args, **kwargs) return f(inst, *args, **kwargs)
@ -1102,6 +1105,29 @@ class RBDTestCase(test.TestCase):
flatten_mock.assert_called_once_with(self.cfg.rbd_pool, flatten_mock.assert_called_once_with(self.cfg.rbd_pool,
self.volume_a.name) self.volume_a.name)
@common_mocks
@mock.patch('cinder.objects.Volume.get_by_id')
@mock.patch.object(driver.RBDDriver, '_resize', mock.Mock())
def test_log_create_vol_from_snap_raise(self, volume_get_by_id):
volume_get_by_id.return_value = self.volume_a
self.mock_proxy().__enter__().volume.op_features.side_effect = \
Exception
self.mock_rbd.RBD_OPERATION_FEATURE_CLONE_PARENT = 1
snapshot = self.snapshot
self.cfg.rbd_flatten_volume_from_snapshot = False
with mock.patch.object(driver, 'LOG') as mock_log:
# Fist call
self.driver.create_volume_from_snapshot(self.volume_a, snapshot)
self.assertTrue(self.driver._clone_v2_api_checked)
# Second call
self.driver.create_volume_from_snapshot(self.volume_a, snapshot)
# Check that that the second call to create_volume_from_snapshot
# doesn't log anything
mock_log.warning.assert_called_once_with(mock.ANY)
@common_mocks @common_mocks
@mock.patch('cinder.objects.Volume.get_by_id') @mock.patch('cinder.objects.Volume.get_by_id')
def test_delete_snapshot(self, volume_get_by_id): def test_delete_snapshot(self, volume_get_by_id):

View File

@ -328,12 +328,13 @@ class RBDDriver(driver.CloneableImageVD, driver.MigrateVD,
self._clone_v2_api_checked = True self._clone_v2_api_checked = True
with RBDVolumeProxy(self, volume_name, read_only=True) as volume: with RBDVolumeProxy(self, volume_name, read_only=True) as volume:
try: try:
if (volume.volume.op_features() & enabled = (volume.volume.op_features() &
self.rbd.RBD_OPERATION_FEATURE_CLONE_PARENT): self.rbd.RBD_OPERATION_FEATURE_CLONE_PARENT)
except Exception:
enabled = False
if enabled:
LOG.info('Using v2 Clone API') LOG.info('Using v2 Clone API')
return else:
except AttributeError:
pass
LOG.warning('Not using v2 clone API, please upgrade to' LOG.warning('Not using v2 clone API, please upgrade to'
' mimic+ and set the OSD minimum client' ' mimic+ and set the OSD minimum client'
' compat version to mimic for better' ' compat version to mimic for better'

View File

@ -0,0 +1,10 @@
---
fixes:
- |
RBD driver `bug #1942210
<https://bugs.launchpad.net/cinder/+bug/1942210>`_: When
creating a volume from a snapshot, the operation could
fail due to an uncaught exception being raised during a
check to see if the backend Ceph installation supported
the clone v2 API. The driver now handles this situation
gracefully.