Merge "Add dependency check in RBD delete_snapshot"
This commit is contained in:
commit
0330f251ce
@ -372,9 +372,48 @@ class RBDTestCase(test.TestCase):
|
|||||||
|
|
||||||
self.driver.delete_snapshot(self.snapshot)
|
self.driver.delete_snapshot(self.snapshot)
|
||||||
|
|
||||||
args = [str(self.snapshot_name)]
|
proxy.remove_snap.assert_called_with(self.snapshot_name)
|
||||||
proxy.remove_snap.assert_called_with(*args)
|
proxy.unprotect_snap.assert_called_with(self.snapshot_name)
|
||||||
proxy.unprotect_snap.assert_called_with(*args)
|
|
||||||
|
@common_mocks
|
||||||
|
def test_delete_busy_snapshot(self):
|
||||||
|
proxy = self.mock_proxy.return_value
|
||||||
|
proxy.__enter__.return_value = proxy
|
||||||
|
|
||||||
|
proxy.unprotect_snap.side_effect = (
|
||||||
|
self.mock_rbd.ImageBusy)
|
||||||
|
|
||||||
|
with mock.patch.object(self.driver, '_get_children_info') as \
|
||||||
|
mock_get_children_info:
|
||||||
|
mock_get_children_info.return_value = [('pool', 'volume2')]
|
||||||
|
|
||||||
|
with mock.patch.object(driver, 'LOG') as \
|
||||||
|
mock_log:
|
||||||
|
|
||||||
|
self.assertRaises(exception.SnapshotIsBusy,
|
||||||
|
self.driver.delete_snapshot,
|
||||||
|
self.snapshot)
|
||||||
|
|
||||||
|
mock_get_children_info.assert_called_once_with(
|
||||||
|
proxy,
|
||||||
|
self.snapshot_name)
|
||||||
|
|
||||||
|
self.assertTrue(mock_log.info.called)
|
||||||
|
self.assertTrue(proxy.unprotect_snap.called)
|
||||||
|
self.assertFalse(proxy.remove_snap.called)
|
||||||
|
|
||||||
|
@common_mocks
|
||||||
|
def test_get_children_info(self):
|
||||||
|
volume = self.mock_proxy
|
||||||
|
volume.set_snap = mock.Mock()
|
||||||
|
volume.list_children = mock.Mock()
|
||||||
|
list_children = [('pool', 'volume2')]
|
||||||
|
volume.list_children.return_value = list_children
|
||||||
|
|
||||||
|
info = self.driver._get_children_info(volume,
|
||||||
|
self.snapshot_name)
|
||||||
|
|
||||||
|
self.assertEqual(list_children, info)
|
||||||
|
|
||||||
@common_mocks
|
@common_mocks
|
||||||
def test_get_clone_info(self):
|
def test_get_clone_info(self):
|
||||||
|
@ -24,7 +24,6 @@ from eventlet import tpool
|
|||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
from oslo_utils import units
|
from oslo_utils import units
|
||||||
import six
|
|
||||||
from six.moves import urllib
|
from six.moves import urllib
|
||||||
|
|
||||||
from cinder import exception
|
from cinder import exception
|
||||||
@ -614,11 +613,26 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
if parent_snap == "%s.clone_snap" % volume_name:
|
if parent_snap == "%s.clone_snap" % volume_name:
|
||||||
return pool, parent, parent_snap
|
return pool, parent, parent_snap
|
||||||
except self.rbd.ImageNotFound:
|
except self.rbd.ImageNotFound:
|
||||||
LOG.debug("volume %s is not a clone", volume_name)
|
LOG.debug("Volume %s is not a clone.", volume_name)
|
||||||
volume.set_snap(None)
|
volume.set_snap(None)
|
||||||
|
|
||||||
return (None, None, None)
|
return (None, None, None)
|
||||||
|
|
||||||
|
def _get_children_info(self, volume, snap):
|
||||||
|
"""List children for the given snapshot of an volume(image).
|
||||||
|
|
||||||
|
Returns a list of (pool, image).
|
||||||
|
"""
|
||||||
|
|
||||||
|
children_list = []
|
||||||
|
|
||||||
|
if snap:
|
||||||
|
volume.set_snap(snap)
|
||||||
|
children_list = volume.list_children()
|
||||||
|
volume.set_snap(None)
|
||||||
|
|
||||||
|
return children_list
|
||||||
|
|
||||||
def _delete_clone_parent_refs(self, client, parent_name, parent_snap):
|
def _delete_clone_parent_refs(self, client, parent_name, parent_snap):
|
||||||
"""Walk back up the clone chain and delete references.
|
"""Walk back up the clone chain and delete references.
|
||||||
|
|
||||||
@ -734,10 +748,21 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
# utf-8 otherwise librbd will barf.
|
# utf-8 otherwise librbd will barf.
|
||||||
volume_name = utils.convert_str(snapshot['volume_name'])
|
volume_name = utils.convert_str(snapshot['volume_name'])
|
||||||
snap_name = utils.convert_str(snapshot['name'])
|
snap_name = utils.convert_str(snapshot['name'])
|
||||||
|
|
||||||
with RBDVolumeProxy(self, volume_name) as volume:
|
with RBDVolumeProxy(self, volume_name) as volume:
|
||||||
try:
|
try:
|
||||||
volume.unprotect_snap(snap_name)
|
volume.unprotect_snap(snap_name)
|
||||||
except self.rbd.ImageBusy:
|
except self.rbd.ImageBusy:
|
||||||
|
children_list = self._get_children_info(volume, snap_name)
|
||||||
|
|
||||||
|
if children_list:
|
||||||
|
for (pool, image) in children_list:
|
||||||
|
LOG.info(_LI('Image %(pool)s/%(image)s is dependent '
|
||||||
|
'on the snapshot %(snap)s.'),
|
||||||
|
{'pool': pool,
|
||||||
|
'image': image,
|
||||||
|
'snap': snap_name})
|
||||||
|
|
||||||
raise exception.SnapshotIsBusy(snapshot_name=snap_name)
|
raise exception.SnapshotIsBusy(snapshot_name=snap_name)
|
||||||
volume.remove_snap(snap_name)
|
volume.remove_snap(snap_name)
|
||||||
|
|
||||||
@ -753,15 +778,15 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
})
|
})
|
||||||
|
|
||||||
if volume['host'] != host['host']:
|
if volume['host'] != host['host']:
|
||||||
LOG.error(_LE('Retype with host migration not supported'))
|
LOG.error(_LE('Retype with host migration not supported.'))
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if diff['encryption']:
|
if diff['encryption']:
|
||||||
LOG.error(_LE('Retype of encryption type not supported'))
|
LOG.error(_LE('Retype of encryption type not supported.'))
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if diff['extra_specs']:
|
if diff['extra_specs']:
|
||||||
LOG.error(_LE('Retype of extra_specs not supported'))
|
LOG.error(_LE('Retype of extra_specs not supported.'))
|
||||||
return False
|
return False
|
||||||
|
|
||||||
return True
|
return True
|
||||||
@ -823,17 +848,18 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
try:
|
try:
|
||||||
fsid, pool, image, snapshot = self._parse_location(image_location)
|
fsid, pool, image, snapshot = self._parse_location(image_location)
|
||||||
except exception.ImageUnacceptable as e:
|
except exception.ImageUnacceptable as e:
|
||||||
LOG.debug('not cloneable: %s', six.text_type(e))
|
LOG.debug('not cloneable: %s.', e)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if self._get_fsid() != fsid:
|
if self._get_fsid() != fsid:
|
||||||
LOG.debug('%s is in a different ceph cluster', image_location)
|
LOG.debug('%s is in a different ceph cluster.', image_location)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if image_meta['disk_format'] != 'raw':
|
if image_meta['disk_format'] != 'raw':
|
||||||
LOG.debug(("rbd image clone requires image format to be "
|
LOG.debug("rbd image clone requires image format to be "
|
||||||
"'raw' but image {0} is '{1}'").format(
|
"'raw' but image %(image)s is '%(format)s'",
|
||||||
image_location, image_meta['disk_format']))
|
{"image", image_location,
|
||||||
|
"format", image_meta['disk_format']})
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# check that we can read the image
|
# check that we can read the image
|
||||||
@ -844,7 +870,7 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
read_only=True):
|
read_only=True):
|
||||||
return True
|
return True
|
||||||
except self.rbd.Error as e:
|
except self.rbd.Error as e:
|
||||||
LOG.debug('Unable to open image %(loc)s: %(err)s',
|
LOG.debug('Unable to open image %(loc)s: %(err)s.',
|
||||||
dict(loc=image_location, err=e))
|
dict(loc=image_location, err=e))
|
||||||
return False
|
return False
|
||||||
|
|
||||||
@ -880,7 +906,7 @@ class RBDDriver(driver.RetypeVD, driver.TransferVD, driver.ExtendVD,
|
|||||||
|
|
||||||
if tmpdir == self.configuration.volume_tmp_dir:
|
if tmpdir == self.configuration.volume_tmp_dir:
|
||||||
LOG.warning(_LW('volume_tmp_dir is now deprecated, please use '
|
LOG.warning(_LW('volume_tmp_dir is now deprecated, please use '
|
||||||
'image_conversion_dir'))
|
'image_conversion_dir.'))
|
||||||
|
|
||||||
# ensure temporary directory exists
|
# ensure temporary directory exists
|
||||||
if not os.path.exists(tmpdir):
|
if not os.path.exists(tmpdir):
|
||||||
|
Loading…
Reference in New Issue
Block a user