Modify default/delete volume type logic

This patch modifies the delete volume type logic such that a volume type
cannot be deleted if:
1) It is the default volume type
2) The default type configured is wrong/doesn't exist

This also implies that there will exist atleast 1 volume type in the
deployment and that will be the default volume type.

This also includes following 2 changes on the default_volume_type conf option:
1) It is a mandatory field
2) default value of this config option is '__DEFAULT__'

All these changes ensure that we don't allow creating untyped volumes.

Also it is now possible to delete the '__DEFAULT__' type as it acts
as a normal type.

Change-Id: Ifa3d22305060b5913332cad89ea696bf7fd84ce1
Closes-Bug: #1886632
This commit is contained in:
Rajat Dhasmana 2020-07-16 16:51:30 +00:00
parent 641762befc
commit e5d842eb1b
14 changed files with 228 additions and 56 deletions

View File

@ -158,11 +158,15 @@ class VolumeTypesManageController(wsgi.Controller):
context, 'volume_type.delete', err, id=id) context, 'volume_type.delete', err, id=id)
# Not found exception will be handled at the wsgi level # Not found exception will be handled at the wsgi level
raise raise
except exception.VolumeTypeDefault as err: except (exception.VolumeTypeDeletionError,
exception.VolumeTypeDefaultDeletionError) as err:
self._notify_volume_type_error( self._notify_volume_type_error(
context, 'volume_type.delete', err, volume_type=vol_type) context, 'volume_type.delete', err, volume_type=vol_type)
msg = _('Target volume type is default and cannot be deleted.') raise webob.exc.HTTPBadRequest(explanation=err.msg)
raise webob.exc.HTTPBadRequest(explanation=msg) except exception.VolumeTypeDefaultMisconfiguredError as err:
self._notify_volume_type_error(
context, 'volume_type.delete', err, volume_type=vol_type)
raise webob.exc.HTTPInternalServerError(explanation=err.msg)
return webob.Response(status_int=http_client.ACCEPTED) return webob.Response(status_int=http_client.ACCEPTED)

View File

@ -264,11 +264,12 @@ class VolumeController(wsgi.Controller):
"be to specify multiattach enabled volume types.") "be to specify multiattach enabled volume types.")
versionutils.report_deprecated_feature(LOG, msg) versionutils.report_deprecated_feature(LOG, msg)
new_volume = self.volume_api.create(context, try:
size, new_volume = self.volume_api.create(
volume.get('display_name'), context, size, volume.get('display_name'),
volume.get('display_description'), volume.get('display_description'), **kwargs)
**kwargs) except exception.VolumeTypeDefaultMisconfiguredError as err:
raise webob.exc.HTTPInternalServerError(explanation=err.msg)
retval = self._view_builder.detail(req, new_volume) retval = self._view_builder.detail(req, new_volume)

View File

@ -389,15 +389,14 @@ class VolumeController(volumes_v2.VolumeController):
"release. The default behavior going forward will " "release. The default behavior going forward will "
"be to specify multiattach enabled volume types.") "be to specify multiattach enabled volume types.")
versionutils.report_deprecated_feature(LOG, msg) versionutils.report_deprecated_feature(LOG, msg)
try:
new_volume = self.volume_api.create(context, new_volume = self.volume_api.create(
size, context, size, volume.get('display_name'),
volume.get('display_name'), volume.get('display_description'), **kwargs)
volume.get('display_description'), except exception.VolumeTypeDefaultMisconfiguredError as err:
**kwargs) raise exc.HTTPInternalServerError(explanation=err.msg)
retval = self._view_builder.detail(req, new_volume) retval = self._view_builder.detail(req, new_volume)
return retval return retval

View File

@ -107,6 +107,8 @@ global_opts = [
'default_availability_zone, then ' 'default_availability_zone, then '
'storage_availability_zone, instead of failing.'), 'storage_availability_zone, instead of failing.'),
cfg.StrOpt('default_volume_type', cfg.StrOpt('default_volume_type',
default='__DEFAULT__',
required=True,
help='Default volume type to use'), help='Default volume type to use'),
cfg.StrOpt('default_group_type', cfg.StrOpt('default_group_type',
help='Default group type to use'), help='Default group type to use'),

View File

@ -4157,6 +4157,9 @@ def volume_type_destroy(context, id):
utcnow = timeutils.utcnow() utcnow = timeutils.utcnow()
session = get_session() session = get_session()
with session.begin(): with session.begin():
vol_types = volume_type_get_all(context)
if len(vol_types) <= 1:
raise exception.VolumeTypeDeletionError(volume_type_id=id)
_volume_type_get(context, id, session) _volume_type_get(context, id, session)
results = model_query(context, models.Volume, session=session). \ results = model_query(context, models.Volume, session=session). \
filter_by(volume_type_id=id).all() filter_by(volume_type_id=id).all()

View File

@ -399,9 +399,19 @@ class VolumeTypeInUse(CinderException):
"volumes present with the type.") "volumes present with the type.")
class VolumeTypeDefault(CinderException): class VolumeTypeDeletionError(Invalid):
message = _("The volume type %(volume_type_name)s " message = _("The volume type %(volume_type_id)s is the only currently "
"is the default volume type and cannot be deleted.") "defined volume type and cannot be deleted.")
class VolumeTypeDefaultDeletionError(Invalid):
message = _("The volume type %(volume_type_id)s is the default volume "
"type and cannot be deleted.")
class VolumeTypeDefaultMisconfiguredError(CinderException):
message = _("The request cannot be fulfilled as the default volume type "
"%(volume_type_name)s cannot be found.")
class GroupTypeNotFound(NotFound): class GroupTypeNotFound(NotFound):

View File

@ -56,6 +56,10 @@ class OpenStackApiException400(OpenStackApiException):
message = _("400 Bad Request") message = _("400 Bad Request")
class OpenStackApiException500(OpenStackApiException):
message = _("500 Internal Server Error")
class TestOpenStackClient(object): class TestOpenStackClient(object):
"""Simple OpenStack API Client. """Simple OpenStack API Client.

View File

@ -15,9 +15,9 @@
from oslo_utils import uuidutils from oslo_utils import uuidutils
from cinder.tests.functional.api import client
from cinder.tests.functional import functional_helpers from cinder.tests.functional import functional_helpers
from cinder.volume import configuration from cinder.volume import configuration
from cinder.volume import volume_types
class VolumesTest(functional_helpers._FunctionalTestBase): class VolumesTest(functional_helpers._FunctionalTestBase):
@ -79,10 +79,7 @@ class VolumesTest(functional_helpers._FunctionalTestBase):
self.assertIsNone(found_volume) self.assertIsNone(found_volume)
def test_create_no_volume_type(self): def test_create_no_volume_type(self):
"""Verify volume_type is not None (should be system default type.)""" """Verify volume_type is not None"""
# un-configure operator default volume type
self.flags(default_volume_type=None)
# Create volume # Create volume
created_volume = self.api.post_volume({'volume': {'size': 1}}) created_volume = self.api.post_volume({'volume': {'size': 1}})
@ -93,15 +90,58 @@ class VolumesTest(functional_helpers._FunctionalTestBase):
found_volume = self._poll_volume_while(created_volume_id, ['creating']) found_volume = self._poll_volume_while(created_volume_id, ['creating'])
self.assertEqual('available', found_volume['status']) self.assertEqual('available', found_volume['status'])
# It should have the system default volume_type # It should have a volume_type
self.assertEqual(volume_types.DEFAULT_VOLUME_TYPE, self.assertIsNotNone(found_volume['volume_type'])
found_volume['volume_type'])
# Delete the volume # Delete the volume
self.api.delete_volume(created_volume_id) self.api.delete_volume(created_volume_id)
found_volume = self._poll_volume_while(created_volume_id, ['deleting']) found_volume = self._poll_volume_while(created_volume_id, ['deleting'])
self.assertIsNone(found_volume) self.assertIsNone(found_volume)
def test_create_volume_default_type(self):
"""Verify that the configured default_volume_type is used"""
my_vol_type_name = 'default_type'
self.api.create_type(my_vol_type_name)
self.flags(default_volume_type=my_vol_type_name)
# Create volume
created_volume = self.api.post_volume({'volume': {'size': 1}})
self.assertTrue(uuidutils.is_uuid_like(created_volume['id']))
created_volume_id = created_volume['id']
# Wait (briefly) for creation. Delay is due to the 'message queue'
found_volume = self._poll_volume_while(created_volume_id, ['creating'])
self.assertEqual('available', found_volume['status'])
# It should have the default volume_type
self.assertEqual(my_vol_type_name, found_volume['volume_type'])
# Delete the volume
self.api.delete_volume(created_volume_id)
found_volume = self._poll_volume_while(created_volume_id, ['deleting'])
self.assertIsNone(found_volume)
def test_create_volume_bad_default_type(self):
"""Verify non-existent default volume type errors out."""
# configure a non-existent default type
self.flags(default_volume_type='non-existent-type')
# Create volume and verify it errors out with 500 status
self.assertRaises(client.OpenStackApiException500,
self.api.post_volume, {'volume': {'size': 1}})
def test_create_volume_default_type_set_none(self):
"""Verify None default volume type errors out."""
# configure None default type
self.flags(default_volume_type=None)
# Create volume and verify it errors out with 500 status
self.assertRaises(client.OpenStackApiException500,
self.api.post_volume, {'volume': {'size': 1}})
def test_create_volume_specified_type(self): def test_create_volume_specified_type(self):
"""Verify volume_type is not default.""" """Verify volume_type is not default."""

View File

@ -40,6 +40,12 @@ class VolumeTypeTestCase(test.TestCase):
volume_types.get_by_name_or_id, self.ctxt, volume_types.get_by_name_or_id, self.ctxt,
volume_type['id']) volume_type['id'])
def test_volume_db_delete_last_type(self):
default = volume_types.get_default_volume_type()
self.assertRaises(exception.VolumeTypeDeletionError,
db.volume_type_destroy, self.ctxt,
default['id'])
def test_volume_type_delete_with_volume_in_use(self): def test_volume_type_delete_with_volume_in_use(self):
volume_type = db.volume_type_create(self.ctxt, {'name': volume_type = db.volume_type_create(self.ctxt, {'name':
'fake volume type'}) 'fake volume type'})

View File

@ -226,6 +226,30 @@ class VolumeTypeTestCase(test.TestCase):
self.assertEqual(conf_fixture.def_vol_type, self.assertEqual(conf_fixture.def_vol_type,
default_vol_type.get('name')) default_vol_type.get('name'))
def test_get_default_volume_type_not_found(self):
"""Ensure setting non-existent default type raises error."""
self.flags(default_volume_type='fake_type')
self.assertRaises(exception.VolumeTypeDefaultMisconfiguredError,
volume_types.get_default_volume_type)
def test_delete_default_volume_type(self):
"""Ensures default volume type cannot be deleted."""
default = volume_types.create(self.ctxt, 'default_type')
self.flags(default_volume_type='default_type')
self.assertRaises(exception.VolumeTypeDefaultDeletionError,
volume_types.destroy, self.ctxt, default['id'])
def test_delete_when_default_volume_type_not_found(self):
"""Ensures volume types cannot be deleted until valid default is set.
"""
default = volume_types.create(self.ctxt, 'default_type')
self.flags(default_volume_type='fake_default')
self.assertRaises(exception.VolumeTypeDefaultMisconfiguredError,
volume_types.destroy, self.ctxt, default['id'])
def test_default_volume_type_missing_in_db(self): def test_default_volume_type_missing_in_db(self):
"""Test default volume type is missing in database. """Test default volume type is missing in database.
@ -248,6 +272,9 @@ class VolumeTypeTestCase(test.TestCase):
def test_non_existent_vol_type_shouldnt_delete(self): def test_non_existent_vol_type_shouldnt_delete(self):
"""Ensures that volume type creation fails with invalid args.""" """Ensures that volume type creation fails with invalid args."""
# create a dummy type as DB requires at least 1 type to perform the
# delete operation
volume_types.create(self.ctxt, self.vol_type1_name)
self.assertRaises(exception.VolumeTypeNotFound, self.assertRaises(exception.VolumeTypeNotFound,
volume_types.destroy, self.ctxt, "sfsfsdfdfs") volume_types.destroy, self.ctxt, "sfsfsdfdfs")

View File

@ -53,14 +53,14 @@ class ExtractVolumeRequestTaskValidationsTestCase(test.TestCase):
'param_source_vol': None, 'param_source_vol': None,
'param_snap': None, 'param_snap': None,
'param_img_vol_type_id': None, 'param_img_vol_type_id': None,
'config_value': None, 'config_value': volume_types.DEFAULT_VOLUME_TYPE,
'expected_vol_type': volume_types.DEFAULT_VOLUME_TYPE}, 'expected_vol_type': volume_types.DEFAULT_VOLUME_TYPE},
# case set 1: if a volume_type is passed, should always be selected # case set 1: if a volume_type is passed, should always be selected
{'param_vol_type': fake_vol_type, {'param_vol_type': fake_vol_type,
'param_source_vol': None, 'param_source_vol': None,
'param_snap': None, 'param_snap': None,
'param_img_vol_type_id': None, 'param_img_vol_type_id': None,
'config_value': None, 'config_value': volume_types.DEFAULT_VOLUME_TYPE,
'expected_vol_type': 'vt-from-volume_type'}, 'expected_vol_type': 'vt-from-volume_type'},
{'param_vol_type': fake_vol_type, {'param_vol_type': fake_vol_type,
'param_source_vol': fake_source_vol, 'param_source_vol': fake_source_vol,
@ -74,7 +74,7 @@ class ExtractVolumeRequestTaskValidationsTestCase(test.TestCase):
'param_source_vol': fake_source_vol, 'param_source_vol': fake_source_vol,
'param_snap': None, 'param_snap': None,
'param_img_vol_type_id': None, 'param_img_vol_type_id': None,
'config_value': None, 'config_value': volume_types.DEFAULT_VOLUME_TYPE,
'expected_vol_type': 'vt-from-source_vol'}, 'expected_vol_type': 'vt-from-source_vol'},
{'param_vol_type': None, {'param_vol_type': None,
'param_source_vol': fake_source_vol, 'param_source_vol': fake_source_vol,
@ -88,7 +88,7 @@ class ExtractVolumeRequestTaskValidationsTestCase(test.TestCase):
'param_source_vol': None, 'param_source_vol': None,
'param_snap': fake_snapshot, 'param_snap': fake_snapshot,
'param_img_vol_type_id': None, 'param_img_vol_type_id': None,
'config_value': None, 'config_value': volume_types.DEFAULT_VOLUME_TYPE,
'expected_vol_type': 'vt-from-snapshot'}, 'expected_vol_type': 'vt-from-snapshot'},
{'param_vol_type': None, {'param_vol_type': None,
'param_source_vol': None, 'param_source_vol': None,
@ -102,7 +102,7 @@ class ExtractVolumeRequestTaskValidationsTestCase(test.TestCase):
'param_source_vol': None, 'param_source_vol': None,
'param_snap': None, 'param_snap': None,
'param_img_vol_type_id': fake_img_vol_type_id, 'param_img_vol_type_id': fake_img_vol_type_id,
'config_value': None, 'config_value': volume_types.DEFAULT_VOLUME_TYPE,
'expected_vol_type': 'vt-from-image_volume_type_id'}, 'expected_vol_type': 'vt-from-image_volume_type_id'},
{'param_vol_type': None, {'param_vol_type': None,
'param_source_vol': None, 'param_source_vol': None,
@ -190,10 +190,19 @@ class ExtractVolumeRequestTaskValidationsTestCase(test.TestCase):
test_fn = create_volume.ExtractVolumeRequestTask._get_volume_type test_fn = create_volume.ExtractVolumeRequestTask._get_volume_type
self.assertRaises(exception.VolumeTypeNotFoundByName, if config_value:
test_fn, self.assertRaises(exception.VolumeTypeDefaultMisconfiguredError,
self.context, test_fn,
None, self.context,
param_source_vol, None,
param_snap, param_source_vol,
param_img_vol_type_id) param_snap,
param_img_vol_type_id)
else:
self.assertRaises(exception.VolumeTypeNotFoundByName,
test_fn,
self.context,
None,
param_source_vol,
param_snap,
param_img_vol_type_id)

View File

@ -605,6 +605,17 @@ class VolumeTestCase(base.BaseVolumeTestCase):
volume_type=self.vol_type) volume_type=self.vol_type)
self.assertEqual('default-az', volume['availability_zone']) self.assertEqual('default-az', volume['availability_zone'])
def test_create_volume_with_default_type_misconfigured(self):
"""Test volume creation with non-existent default volume type."""
volume_api = cinder.volume.api.API()
self.flags(default_volume_type='fake_type')
# Create volume with default volume type while default
# volume type doesn't exist
self.assertRaises(exception.VolumeTypeDefaultMisconfiguredError,
volume_api.create, self.context, 1,
'name', 'description')
@mock.patch('cinder.quota.QUOTAS.rollback', new=mock.MagicMock()) @mock.patch('cinder.quota.QUOTAS.rollback', new=mock.MagicMock())
@mock.patch('cinder.quota.QUOTAS.commit', new=mock.MagicMock()) @mock.patch('cinder.quota.QUOTAS.commit', new=mock.MagicMock())
@mock.patch('cinder.quota.QUOTAS.reserve', return_value=["RESERVATION"]) @mock.patch('cinder.quota.QUOTAS.reserve', return_value=["RESERVATION"])

View File

@ -102,14 +102,31 @@ def update(context, id, name, description, is_public=None):
def destroy(context, id): def destroy(context, id):
"""Marks volume types as deleted.""" """Marks volume types as deleted.
There must exist at least one volume type (i.e. the default type) in
the deployment.
This method achieves that by ensuring:
1) the default_volume_type is set and is a valid one
2) the type requested to delete isn't the default type
:raises VolumeTypeDefaultDeletionError: when the type requested to
delete is the default type
"""
if id is None: if id is None:
msg = _("id cannot be None") msg = _("id cannot be None")
raise exception.InvalidVolumeType(reason=msg) raise exception.InvalidVolumeType(reason=msg)
vol_type = get_volume_type(context, id)
if vol_type['name'] == DEFAULT_VOLUME_TYPE:
raise exception.VolumeTypeDefault(vol_type['name'])
elevated = context if context.is_admin else context.elevated() elevated = context if context.is_admin else context.elevated()
# Default type *must* be set in order to delete any volume type.
# If the default isn't set, the following call will raise
# VolumeTypeDefaultMisconfiguredError exception which will error out the
# delete operation.
default_type = get_default_volume_type()
# don't allow delete if the type requested is the default type
if id == default_type.get('id'):
raise exception.VolumeTypeDefaultDeletionError(volume_type_id=id)
return db.volume_type_destroy(elevated, id) return db.volume_type_destroy(elevated, id)
@ -173,22 +190,23 @@ def get_volume_type_by_name(context, name):
def get_default_volume_type(): def get_default_volume_type():
"""Get the default volume type.""" """Get the default volume type.
:raises VolumeTypeDefaultMisconfiguredError: when the configured default
is not found
"""
name = CONF.default_volume_type name = CONF.default_volume_type
vol_type = {}
ctxt = context.get_admin_context() ctxt = context.get_admin_context()
if name: vol_type = {}
try: try:
vol_type = get_volume_type_by_name(ctxt, name) vol_type = get_volume_type_by_name(ctxt, name)
except exception.VolumeTypeNotFoundByName: except (exception.VolumeTypeNotFoundByName, exception.InvalidVolumeType):
# Couldn't find volume type with the name in default_volume_type # Couldn't find volume type with the name in default_volume_type
# flag, record this issue and raise exception # flag, record this issue and raise exception
# TODO(zhiteng) consider add notification to warn admin LOG.exception('Default volume type is not found. '
LOG.exception('Default volume type is not found. ' 'Please check default_volume_type config:')
'Please check default_volume_type config:') raise exception.VolumeTypeDefaultMisconfiguredError(
raise exception.VolumeTypeNotFoundByName(volume_type_name=name) volume_type_name=name)
else:
vol_type = get_volume_type_by_name(ctxt, DEFAULT_VOLUME_TYPE)
return vol_type return vol_type

View File

@ -0,0 +1,38 @@
---
upgrade:
- |
The ``default_volume_type`` configuration option is now required
to have a value. The default value is ``__DEFAULT__``, so you
should see no change in behavior whether or not you have set a
value for ``default_volume_type``. See
`Bug #1886632 <https://bugs.launchpad.net/cinder/+bug/1886632>`_
for more information about this change.
fixes:
- |
`Bug #1886632 <https://bugs.launchpad.net/cinder/+bug/1886632>`_:
The system defined ``__DEFAULT__`` volume type is now treated as
a regular volume-type and may be updated or deleted. Since the
configured ``default_volume_type`` cannot be deleted, however,
the ``__DEFAULT__`` volume type may not be deleted if it is the
value of that configuration option.
other:
- |
Beginning with the Train release, untyped volumes (that is, volumes with
no volume-type) have been disallowed. To facilitate this, a
``__DEFAULT__`` volume-type was included as part of the Train database
migration. In this release, handling of the default volume-type has been
improved:
* The ``default_volume_type`` configuration option is required to have a
value. The default value is ``__DEFAULT__``.
* A request to delete the currently configured ``default_volume_type``
will fail. (You can delete that volume-type, but you cannot do it
while it is the value of the configuration option.)
* There must always be at least one volume-type defined in a Cinder
installation. This is enforced by the type-delete call.
* If the ``default_volume_type`` is misconfigured (that is, if the value
refers to a non-existent volume-type), requests that rely on the
default volume-type (for example, a volume-create request that does
not specify a volume-type) will result in a HTTP 500 response.