Add user messages for some volume snapshot actions

Also adds tests to help keep the message_field module
consistent.

Co-authored-by: TommyLike <tommylikehu@gmail.com>
Co-authored-by: Brian Rosmaita <rosmaita.fossdev@gmail.com>

Change-Id: I8a16c9742cadfee1a75a6d738945476a9fc7c19b
This commit is contained in:
TommyLike 2017-03-21 17:24:38 +08:00 committed by Brian Rosmaita
parent 2c14226392
commit 46bec90f03
4 changed files with 223 additions and 5 deletions

View File

@ -27,6 +27,7 @@ from cinder.i18n import _
class Resource(object): class Resource(object):
VOLUME = 'VOLUME' VOLUME = 'VOLUME'
VOLUME_SNAPSHOT = 'VOLUME_SNAPSHOT'
class Action(object): class Action(object):
@ -40,6 +41,10 @@ class Action(object):
EXTEND_VOLUME = ('007', _('extend volume')) EXTEND_VOLUME = ('007', _('extend volume'))
CREATE_VOLUME_FROM_BACKEND = ('008', CREATE_VOLUME_FROM_BACKEND = ('008',
_('create volume from backend storage')) _('create volume from backend storage'))
SNAPSHOT_CREATE = ('009', _('create snapshot'))
SNAPSHOT_DELETE = ('010', _('delete snapshot'))
SNAPSHOT_UPDATE = ('011', _('update snapshot'))
SNAPSHOT_METADATA_UPDATE = ('012', _('update snapshot metadata'))
ALL = (SCHEDULE_ALLOCATE_VOLUME, ALL = (SCHEDULE_ALLOCATE_VOLUME,
ATTACH_VOLUME, ATTACH_VOLUME,
@ -48,7 +53,11 @@ class Action(object):
COPY_IMAGE_TO_VOLUME, COPY_IMAGE_TO_VOLUME,
UNMANAGE_VOLUME, UNMANAGE_VOLUME,
EXTEND_VOLUME, EXTEND_VOLUME,
CREATE_VOLUME_FROM_BACKEND CREATE_VOLUME_FROM_BACKEND,
SNAPSHOT_CREATE,
SNAPSHOT_DELETE,
SNAPSHOT_UPDATE,
SNAPSHOT_METADATA_UPDATE,
) )
@ -85,6 +94,12 @@ class Detail(object):
DRIVER_FAILED_CREATE = ( DRIVER_FAILED_CREATE = (
'012', '012',
_('Driver failed to create the volume.')) _('Driver failed to create the volume.'))
SNAPSHOT_CREATE_ERROR = ('013', _("Snapshot failed to create."))
SNAPSHOT_UPDATE_METADATA_FAILED = (
'014',
_("Volume snapshot update metadata failed."))
SNAPSHOT_IS_BUSY = ('015', _("Snapshot is busy."))
SNAPSHOT_DELETE_ERROR = ('016', _("Snapshot failed to delete."))
ALL = (UNKNOWN_ERROR, ALL = (UNKNOWN_ERROR,
DRIVER_NOT_INITIALIZED, DRIVER_NOT_INITIALIZED,
@ -97,7 +112,12 @@ class Detail(object):
NOTIFY_COMPUTE_SERVICE_FAILED, NOTIFY_COMPUTE_SERVICE_FAILED,
DRIVER_FAILED_EXTEND, DRIVER_FAILED_EXTEND,
SIGNATURE_VERIFICATION_FAILED, SIGNATURE_VERIFICATION_FAILED,
DRIVER_FAILED_CREATE) DRIVER_FAILED_CREATE,
SNAPSHOT_CREATE_ERROR,
SNAPSHOT_UPDATE_METADATA_FAILED,
SNAPSHOT_IS_BUSY,
SNAPSHOT_DELETE_ERROR,
)
# Exception and detail mappings # Exception and detail mappings
EXCEPTION_DETAIL_MAPPINGS = { EXCEPTION_DETAIL_MAPPINGS = {
@ -108,6 +128,7 @@ class Detail(object):
'BackupLimitExceeded', 'BackupLimitExceeded',
'SnapshotLimitExceeded'], 'SnapshotLimitExceeded'],
NOT_ENOUGH_SPACE_FOR_IMAGE: ['ImageTooBig'], NOT_ENOUGH_SPACE_FOR_IMAGE: ['ImageTooBig'],
SNAPSHOT_IS_BUSY: ['SnapshotIsBusy'],
} }

View File

@ -30,11 +30,32 @@ class MessageFieldTest(test.TestCase):
action_ids = [x[0] for x in message_field.Action.ALL] action_ids = [x[0] for x in message_field.Action.ALL]
self.assertEqual(len(action_ids), len(set(action_ids))) self.assertEqual(len(action_ids), len(set(action_ids)))
def test_all_action_fields_in_ALL(self):
"""Assert that all and only defined fields are in the ALL tuple"""
defined_fields = [k for k in message_field.Action.__dict__.keys()
if k != 'ALL' and not k.startswith('__')]
for d in defined_fields:
self.assertIn(getattr(message_field.Action, d),
message_field.Action.ALL)
self.assertEqual(len(message_field.Action.ALL),
len(defined_fields))
def test_unique_detail_ids(self): def test_unique_detail_ids(self):
"""Assert that no detail_id is duplicated.""" """Assert that no detail_id is duplicated."""
detail_ids = [x[0] for x in message_field.Detail.ALL] detail_ids = [x[0] for x in message_field.Detail.ALL]
self.assertEqual(len(detail_ids), len(set(detail_ids))) self.assertEqual(len(detail_ids), len(set(detail_ids)))
def test_all_detail_fields_in_ALL(self):
"""Assert that all and only defined fields are in the ALL tuple"""
defined_fields = [k for k in message_field.Detail.__dict__.keys()
if k != 'ALL' and not k.startswith('__')
and k != 'EXCEPTION_DETAIL_MAPPINGS']
for d in defined_fields:
self.assertIn(getattr(message_field.Detail, d),
message_field.Detail.ALL)
self.assertEqual(len(message_field.Detail.ALL),
len(defined_fields))
known_exceptions = [ known_exceptions = [
name for name, _ in name for name, _ in
inspect.getmembers(exception, inspect.isclass)] inspect.getmembers(exception, inspect.isclass)]

View File

@ -0,0 +1,148 @@
# Copyright 2019, Red Hat Inc.
# All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
"""Tests for Volume Manager Code."""
import mock
from cinder import exception
from cinder.message import message_field
from cinder.tests.unit import volume as base
from cinder.volume import manager as vol_manager
class VolumeManagerTestCase(base.BaseVolumeTestCase):
@mock.patch('cinder.message.api.API.create')
@mock.patch('cinder.utils.require_driver_initialized')
@mock.patch('cinder.volume.manager.VolumeManager.'
'_notify_about_snapshot_usage')
def test_create_snapshot_driver_not_initialized_generates_user_message(
self, fake_notify, fake_init, fake_msg_create):
manager = vol_manager.VolumeManager()
fake_init.side_effect = exception.CinderException()
fake_snapshot = mock.MagicMock(id='22')
fake_context = mock.MagicMock()
fake_context.elevated.return_value = fake_context
ex = self.assertRaises(exception.CinderException,
manager.create_snapshot,
fake_context,
fake_snapshot)
# make sure a user message was generated
fake_msg_create.assert_called_once_with(
fake_context,
action=message_field.Action.SNAPSHOT_CREATE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=fake_snapshot['id'],
exception=ex,
detail=message_field.Detail.SNAPSHOT_CREATE_ERROR)
@mock.patch('cinder.message.api.API.create')
@mock.patch('cinder.utils.require_driver_initialized')
@mock.patch('cinder.volume.manager.VolumeManager.'
'_notify_about_snapshot_usage')
def test_create_snapshot_metadata_update_failure_generates_user_message(
self, fake_notify, fake_init, fake_msg_create):
manager = vol_manager.VolumeManager()
fake_driver = mock.MagicMock()
fake_driver.create_snapshot.return_value = False
manager.driver = fake_driver
fake_vol_ref = mock.MagicMock()
fake_vol_ref.bootable.return_value = True
fake_db = mock.MagicMock()
fake_db.volume_get.return_value = fake_vol_ref
fake_exp = exception.CinderException()
fake_db.volume_glance_metadata_copy_to_snapshot.side_effect = fake_exp
manager.db = fake_db
fake_snapshot = mock.MagicMock(id='86')
fake_context = mock.MagicMock()
fake_context.elevated.return_value = fake_context
self.assertRaises(exception.CinderException,
manager.create_snapshot,
fake_context,
fake_snapshot)
# make sure a user message was generated
fake_msg_create.assert_called_once_with(
fake_context,
action=message_field.Action.SNAPSHOT_CREATE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=fake_snapshot['id'],
exception=fake_exp,
detail=message_field.Detail.SNAPSHOT_UPDATE_METADATA_FAILED)
@mock.patch('cinder.message.api.API.create')
@mock.patch('cinder.utils.require_driver_initialized')
@mock.patch('cinder.volume.manager.VolumeManager.'
'_notify_about_snapshot_usage')
def test_delete_snapshot_when_busy_generates_user_message(
self, fake_notify, fake_init, fake_msg_create):
manager = vol_manager.VolumeManager()
fake_snapshot = mock.MagicMock(id='0', project_id='1')
fake_context = mock.MagicMock()
fake_context.elevated.return_value = fake_context
fake_exp = exception.SnapshotIsBusy(snapshot_name='Fred')
fake_init.side_effect = fake_exp
manager.delete_snapshot(fake_context, fake_snapshot)
# make sure a user message was generated
fake_msg_create.assert_called_once_with(
fake_context,
action=message_field.Action.SNAPSHOT_DELETE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=fake_snapshot['id'],
exception=fake_exp)
@mock.patch('cinder.message.api.API.create')
@mock.patch('cinder.utils.require_driver_initialized')
@mock.patch('cinder.volume.manager.VolumeManager.'
'_notify_about_snapshot_usage')
def test_delete_snapshot_general_exception_generates_user_message(
self, fake_notify, fake_init, fake_msg_create):
manager = vol_manager.VolumeManager()
fake_snapshot = mock.MagicMock(id='0', project_id='1')
fake_context = mock.MagicMock()
fake_context.elevated.return_value = fake_context
class LocalException(Exception):
pass
fake_exp = LocalException()
# yeah, this isn't where it would be coming from in real life,
# but it saves mocking out a bunch more stuff
fake_init.side_effect = fake_exp
self.assertRaises(LocalException,
manager.delete_snapshot,
fake_context,
fake_snapshot)
# make sure a user message was generated
fake_msg_create.assert_called_once_with(
fake_context,
action=message_field.Action.SNAPSHOT_DELETE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=fake_snapshot['id'],
exception=fake_exp,
detail=message_field.Detail.SNAPSHOT_DELETE_ERROR)

View File

@ -1149,10 +1149,17 @@ class VolumeManager(manager.CleanableManager,
snapshot.update(model_update) snapshot.update(model_update)
snapshot.save() snapshot.save()
except Exception: except Exception as create_error:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
snapshot.status = fields.SnapshotStatus.ERROR snapshot.status = fields.SnapshotStatus.ERROR
snapshot.save() snapshot.save()
self.message_api.create(
context,
action=message_field.Action.SNAPSHOT_CREATE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=snapshot['id'],
exception=create_error,
detail=message_field.Detail.SNAPSHOT_CREATE_ERROR)
vol_ref = self.db.volume_get(context, snapshot.volume_id) vol_ref = self.db.volume_get(context, snapshot.volume_id)
if vol_ref.bootable: if vol_ref.bootable:
@ -1172,6 +1179,14 @@ class VolumeManager(manager.CleanableManager,
resource=snapshot) resource=snapshot)
snapshot.status = fields.SnapshotStatus.ERROR snapshot.status = fields.SnapshotStatus.ERROR
snapshot.save() snapshot.save()
self.message_api.create(
context,
action=message_field.Action.SNAPSHOT_CREATE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=snapshot['id'],
exception=ex,
detail=message_field.Detail.SNAPSHOT_UPDATE_METADATA_FAILED
)
raise exception.MetadataCopyFailure(reason=six.text_type(ex)) raise exception.MetadataCopyFailure(reason=six.text_type(ex))
snapshot.status = fields.SnapshotStatus.AVAILABLE snapshot.status = fields.SnapshotStatus.AVAILABLE
@ -1213,16 +1228,29 @@ class VolumeManager(manager.CleanableManager,
self.driver.unmanage_snapshot(snapshot) self.driver.unmanage_snapshot(snapshot)
else: else:
self.driver.delete_snapshot(snapshot) self.driver.delete_snapshot(snapshot)
except exception.SnapshotIsBusy: except exception.SnapshotIsBusy as busy_error:
LOG.error("Delete snapshot failed, due to snapshot busy.", LOG.error("Delete snapshot failed, due to snapshot busy.",
resource=snapshot) resource=snapshot)
snapshot.status = fields.SnapshotStatus.AVAILABLE snapshot.status = fields.SnapshotStatus.AVAILABLE
snapshot.save() snapshot.save()
self.message_api.create(
context,
action=message_field.Action.SNAPSHOT_DELETE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=snapshot['id'],
exception=busy_error)
return return
except Exception: except Exception as delete_error:
with excutils.save_and_reraise_exception(): with excutils.save_and_reraise_exception():
snapshot.status = fields.SnapshotStatus.ERROR_DELETING snapshot.status = fields.SnapshotStatus.ERROR_DELETING
snapshot.save() snapshot.save()
self.message_api.create(
context,
action=message_field.Action.SNAPSHOT_DELETE,
resource_type=message_field.Resource.VOLUME_SNAPSHOT,
resource_uuid=snapshot['id'],
exception=delete_error,
detail=message_field.Detail.SNAPSHOT_DELETE_ERROR)
# Get reservations # Get reservations
reservations = None reservations = None