bf96ca7c9a
This patch add share information to the InstancePayload that can be sent if the include_share_mapping configuration parameter is enabled. Manila is the OpenStack Shared Filesystems service. These series of patches implement changes required in Nova to allow the shares provided by Manila to be associated with and attached to instances using virtiofs. Implements: blueprint libvirt-virtiofs-attach-manila-shares Change-Id: I3d5005eab9e3f23be149e955e8cb4608a6ee1312
216 lines
9.0 KiB
Python
216 lines
9.0 KiB
Python
# 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.
|
|
|
|
from unittest import mock
|
|
|
|
from oslo_utils.fixture import uuidsentinel as uuids
|
|
from oslo_utils import timeutils
|
|
|
|
from nova import context as nova_context
|
|
from nova.network import model as network_model
|
|
from nova.notifications import base as notification_base
|
|
from nova.notifications.objects import instance as instance_notification
|
|
from nova import objects
|
|
from nova import test
|
|
from nova.tests.unit import fake_instance
|
|
|
|
|
|
class TestInstanceNotification(test.NoDBTestCase):
|
|
|
|
def setUp(self):
|
|
super(TestInstanceNotification, self).setUp()
|
|
self.test_keys = ['memory_mb', 'vcpus', 'root_gb', 'ephemeral_gb',
|
|
'swap']
|
|
self.flavor_values = {k: 123 for k in self.test_keys}
|
|
instance_values = {k: 456 for k in self.test_keys}
|
|
flavor = objects.Flavor(flavorid='test-flavor', name='test-flavor',
|
|
disabled=False, projects=[], is_public=True,
|
|
extra_specs={}, **self.flavor_values)
|
|
info_cache = objects.InstanceInfoCache(
|
|
network_info=network_model.NetworkInfo())
|
|
self.instance = objects.Instance(
|
|
flavor=flavor,
|
|
info_cache=info_cache,
|
|
metadata={},
|
|
uuid=uuids.instance1,
|
|
locked=False,
|
|
auto_disk_config=False,
|
|
system_metadata={},
|
|
**instance_values)
|
|
self.payload = {
|
|
'bandwidth': {},
|
|
'audit_period_ending': timeutils.utcnow(),
|
|
'audit_period_beginning': timeutils.utcnow(),
|
|
}
|
|
|
|
@mock.patch('nova.notifications.objects.instance.'
|
|
'InstanceUpdateNotification._emit')
|
|
def test_send_version_instance_update_uses_flavor(self, mock_emit):
|
|
# instance.update notification needs some tags value to avoid lazy-load
|
|
self.instance.tags = objects.TagList()
|
|
# Make sure that the notification payload chooses the values in
|
|
# instance.flavor.$value instead of instance.$value
|
|
mock_context = mock.MagicMock()
|
|
mock_context.project_id = 'fake_project_id'
|
|
mock_context.user_id = 'fake_user_id'
|
|
mock_context.request_id = 'fake_req_id'
|
|
notification_base._send_versioned_instance_update(
|
|
mock_context,
|
|
self.instance,
|
|
self.payload,
|
|
'host',
|
|
'compute')
|
|
payload = mock_emit.call_args_list[0][1]['payload']['nova_object.data']
|
|
flavor_payload = payload['flavor']['nova_object.data']
|
|
data = {k: flavor_payload[k] for k in self.test_keys}
|
|
self.assertEqual(self.flavor_values, data)
|
|
|
|
@mock.patch('nova.rpc.NOTIFIER')
|
|
@mock.patch('nova.notifications.objects.instance.'
|
|
'InstanceUpdatePayload.__init__', return_value=None)
|
|
@mock.patch('nova.notifications.objects.instance.'
|
|
'InstanceUpdateNotification.__init__', return_value=None)
|
|
def test_send_versioned_instance_notification_is_not_called_disabled(
|
|
self, mock_notification, mock_payload, mock_notifier):
|
|
mock_notifier.is_enabled.return_value = False
|
|
|
|
notification_base._send_versioned_instance_update(
|
|
mock.MagicMock(),
|
|
self.instance,
|
|
self.payload,
|
|
'host',
|
|
'compute')
|
|
|
|
self.assertFalse(mock_payload.called)
|
|
self.assertFalse(mock_notification.called)
|
|
|
|
@mock.patch('nova.notifications.objects.instance.'
|
|
'InstanceUpdatePayload.__init__', return_value=None)
|
|
@mock.patch('nova.notifications.objects.instance.'
|
|
'InstanceUpdateNotification.__init__', return_value=None)
|
|
def test_send_versioned_instance_notification_is_not_called_unversioned(
|
|
self, mock_notification, mock_payload):
|
|
self.flags(notification_format='unversioned', group='notifications')
|
|
|
|
notification_base._send_versioned_instance_update(
|
|
mock.MagicMock(),
|
|
self.instance,
|
|
self.payload,
|
|
'host',
|
|
'compute')
|
|
|
|
self.assertFalse(mock_payload.called)
|
|
self.assertFalse(mock_notification.called)
|
|
|
|
def test_instance_payload_request_id_periodic_task(self):
|
|
"""Tests that creating an InstancePayload from the type of request
|
|
context used during a periodic task will not populate the
|
|
payload request_id field since it is not an end user request.
|
|
"""
|
|
ctxt = nova_context.get_admin_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt)
|
|
# Set some other fields otherwise populate_schema tries to hit the DB.
|
|
instance.metadata = {}
|
|
instance.system_metadata = {}
|
|
instance.info_cache = objects.InstanceInfoCache(
|
|
network_info=network_model.NetworkInfo([]))
|
|
payload = instance_notification.InstancePayload(ctxt, instance)
|
|
self.assertIsNone(payload.request_id)
|
|
|
|
|
|
class TestBlockDevicePayload(test.NoDBTestCase):
|
|
@mock.patch('nova.objects.instance.Instance.get_bdms')
|
|
def test_payload_contains_volume_bdms_if_requested(self, mock_get_bdms):
|
|
self.flags(bdms_in_notifications='True', group='notifications')
|
|
context = mock.Mock()
|
|
instance = objects.Instance(uuid=uuids.instance_uuid)
|
|
image_bdm = objects.BlockDeviceMapping(
|
|
**{'context': context, 'source_type': 'image',
|
|
'destination_type': 'local',
|
|
'image_id': uuids.image_id,
|
|
'volume_id': None,
|
|
'device_name': '/dev/vda',
|
|
'instance_uuid': instance.uuid})
|
|
|
|
volume_bdm = objects.BlockDeviceMapping(
|
|
**{'context': context, 'source_type': 'volume',
|
|
'destination_type': 'volume',
|
|
'volume_id': uuids.volume_id,
|
|
'device_name': '/dev/vdb',
|
|
'instance_uuid': instance.uuid,
|
|
'boot_index': 0,
|
|
'delete_on_termination': True,
|
|
'tag': 'my-tag'})
|
|
|
|
mock_get_bdms.return_value = [image_bdm, volume_bdm]
|
|
|
|
bdms = instance_notification.BlockDevicePayload.from_instance(
|
|
instance)
|
|
|
|
self.assertEqual(1, len(bdms))
|
|
bdm = bdms[0]
|
|
self.assertIsInstance(bdm, instance_notification.BlockDevicePayload)
|
|
self.assertEqual('/dev/vdb', bdm.device_name)
|
|
self.assertEqual(0, bdm.boot_index)
|
|
self.assertTrue(bdm.delete_on_termination)
|
|
self.assertEqual('my-tag', bdm.tag)
|
|
self.assertEqual(uuids.volume_id, bdm.volume_id)
|
|
|
|
@mock.patch('nova.objects.instance.Instance.get_bdms',
|
|
return_value=mock.NonCallableMock())
|
|
def test_bdms_are_skipped_by_default(self, mock_get_bdms):
|
|
instance = objects.Instance(uuid=uuids.instance_uuid)
|
|
bmds = instance_notification.BlockDevicePayload.from_instance(
|
|
instance)
|
|
self.assertIsNone(bmds)
|
|
|
|
|
|
class TestSharePayload(test.NoDBTestCase):
|
|
@mock.patch('nova.objects.instance.Instance.get_shares')
|
|
def test_payload_contains_shares_if_requested(self, mock_get_shares):
|
|
self.flags(include_share_mapping='True', group='notifications')
|
|
context = mock.Mock()
|
|
instance = objects.Instance(uuid=uuids.instance_uuid)
|
|
|
|
share_mapping = objects.ShareMapping(context)
|
|
share_mapping.uuid = uuids.share_mapping
|
|
share_mapping.instance_uuid = uuids.instance
|
|
share_mapping.share_id = uuids.share
|
|
share_mapping.status = 'inactive'
|
|
share_mapping.tag = 'fake_tag'
|
|
share_mapping.export_location = '192.168.122.152:/manila/share'
|
|
share_mapping.share_proto = 'NFS'
|
|
|
|
mock_get_shares.return_value = [share_mapping]
|
|
|
|
shares = instance_notification.SharePayload.from_instance(
|
|
instance)
|
|
|
|
self.assertEqual(1, len(shares))
|
|
share = shares[0]
|
|
self.assertIsInstance(share, instance_notification.SharePayload)
|
|
self.assertEqual(uuids.share_mapping, share.share_mapping_uuid)
|
|
self.assertEqual(uuids.share, share.share_id)
|
|
self.assertEqual('inactive', share.status)
|
|
self.assertEqual('fake_tag', share.tag)
|
|
# self.assertEqual('192.168.122.152:/manila/share',
|
|
# share.export_location)
|
|
self.assertNotIn('export_location', share)
|
|
|
|
@mock.patch('nova.objects.instance.Instance.get_shares',
|
|
return_value=mock.NonCallableMock())
|
|
def test_shares_are_skipped_by_default(self, mock_get_shares):
|
|
instance = objects.Instance(uuid=uuids.instance_uuid)
|
|
shares = instance_notification.SharePayload.from_instance(
|
|
instance)
|
|
self.assertIsNone(shares)
|