e43bf900dc
If the vnic_type of a bound port changes from "direct" to "macvtap" and then the compute service is restarted then during _init_instance nova tries to plug the vif of the changed port. However as it now has macvtap vnic_type nova tries to look up the netdev of the parent VF. Still that VF is consumed by the instance so there is no such netdev on the host OS. This error killed the compute service at startup due to unhandled exception. This patch adds the exception handler, logs an ERROR and continue initializing other instances on the host. Also this patch adds a detailed ERROR log when nova detects that the vnic_type changed during _heal_instance_info_cache periodic. Closes-Bug: #1981813 Change-Id: I1719f8eda04e8d15a3b01f0612977164c4e55e85
9921 lines
447 KiB
Python
9921 lines
447 KiB
Python
# Copyright 2012 OpenStack Foundation
|
|
# 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.
|
|
#
|
|
|
|
import collections
|
|
import copy
|
|
from unittest import mock
|
|
|
|
from keystoneauth1.fixture import V2Token
|
|
from keystoneauth1 import loading as ks_loading
|
|
from keystoneauth1 import service_token
|
|
from neutronclient.common import exceptions
|
|
from neutronclient.v2_0 import client
|
|
from oslo_config import cfg
|
|
from oslo_config import fixture as config_fixture
|
|
from oslo_policy import policy as oslo_policy
|
|
from oslo_serialization import jsonutils
|
|
from oslo_utils.fixture import uuidsentinel as uuids
|
|
from oslo_utils import timeutils
|
|
from oslo_utils import uuidutils
|
|
import requests_mock
|
|
|
|
from nova import context
|
|
from nova.db.main import api as db_api
|
|
from nova import exception
|
|
from nova.network import constants
|
|
from nova.network import model
|
|
from nova.network import neutron as neutronapi
|
|
from nova import objects
|
|
from nova.objects import fields as obj_fields
|
|
from nova.objects import network_request as net_req_obj
|
|
from nova.objects import virtual_interface as obj_vif
|
|
from nova.pci import request as pci_request
|
|
from nova.pci import utils as pci_utils
|
|
from nova.pci import whitelist as pci_whitelist
|
|
from nova import policy
|
|
from nova import service_auth
|
|
from nova import test
|
|
from nova.tests.unit import fake_instance
|
|
|
|
|
|
CONF = cfg.CONF
|
|
|
|
# NOTE: Neutron client raises Exception which is discouraged by HACKING.
|
|
# We set this variable here and use it for assertions below to avoid
|
|
# the hacking checks until we can make neutron client throw a custom
|
|
# exception class instead.
|
|
NEUTRON_CLIENT_EXCEPTION = Exception
|
|
|
|
fake_info_cache = {
|
|
'created_at': None,
|
|
'updated_at': None,
|
|
'deleted_at': None,
|
|
'deleted': False,
|
|
'instance_uuid': uuids.instance,
|
|
'network_info': '[]',
|
|
}
|
|
|
|
|
|
class TestNeutronClient(test.NoDBTestCase):
|
|
|
|
def setUp(self):
|
|
super(TestNeutronClient, self).setUp()
|
|
neutronapi.reset_state()
|
|
self.addCleanup(service_auth.reset_globals)
|
|
|
|
def test_ksa_adapter_loading_defaults(self):
|
|
"""No 'url' triggers ksa loading path with defaults."""
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
auth_token='token')
|
|
cl = neutronapi.get_client(my_context)
|
|
self.assertEqual('network', cl.httpclient.service_type)
|
|
self.assertIsNone(cl.httpclient.service_name)
|
|
self.assertEqual(['internal', 'public'], cl.httpclient.interface)
|
|
self.assertIsNone(cl.httpclient.region_name)
|
|
self.assertIsNone(cl.httpclient.endpoint_override)
|
|
self.assertIsNotNone(cl.httpclient.global_request_id)
|
|
self.assertEqual(my_context.global_id,
|
|
cl.httpclient.global_request_id)
|
|
|
|
def test_ksa_adapter_loading(self):
|
|
"""Test ksa loading path with specified values."""
|
|
self.flags(group='neutron',
|
|
service_type='st',
|
|
service_name='sn',
|
|
valid_interfaces='admin',
|
|
region_name='RegionTwo',
|
|
endpoint_override='eo')
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
auth_token='token')
|
|
cl = neutronapi.get_client(my_context)
|
|
self.assertEqual('st', cl.httpclient.service_type)
|
|
self.assertEqual('sn', cl.httpclient.service_name)
|
|
self.assertEqual(['admin'], cl.httpclient.interface)
|
|
self.assertEqual('RegionTwo', cl.httpclient.region_name)
|
|
self.assertEqual('eo', cl.httpclient.endpoint_override)
|
|
|
|
def test_withtoken(self):
|
|
self.flags(endpoint_override='http://anyhost/', group='neutron')
|
|
self.flags(timeout=30, group='neutron')
|
|
# Will use the token rather than load auth from config.
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
auth_token='token')
|
|
cl = neutronapi.get_client(my_context)
|
|
|
|
self.assertEqual(CONF.neutron.endpoint_override,
|
|
cl.httpclient.endpoint_override)
|
|
self.assertEqual(CONF.neutron.region_name, cl.httpclient.region_name)
|
|
self.assertEqual(my_context.auth_token, cl.httpclient.auth.auth_token)
|
|
self.assertEqual(CONF.neutron.timeout, cl.httpclient.session.timeout)
|
|
|
|
def test_withouttoken(self):
|
|
my_context = context.RequestContext('userid', uuids.my_tenant)
|
|
self.assertRaises(exception.Unauthorized,
|
|
neutronapi.get_client,
|
|
my_context)
|
|
|
|
@mock.patch.object(ks_loading, 'load_auth_from_conf_options')
|
|
def test_non_admin_with_service_token(self, mock_load):
|
|
self.flags(send_service_user_token=True, group='service_user')
|
|
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
auth_token='token')
|
|
|
|
cl = neutronapi.get_client(my_context)
|
|
self.assertIsInstance(cl.httpclient.auth,
|
|
service_token.ServiceTokenAuthWrapper)
|
|
|
|
@mock.patch.object(client.Client, "list_networks",
|
|
side_effect=exceptions.Unauthorized())
|
|
def test_Unauthorized_user(self, mock_list_networks):
|
|
my_context = context.RequestContext('userid', uuids.my_tenant,
|
|
auth_token='token',
|
|
is_admin=False)
|
|
client = neutronapi.get_client(my_context)
|
|
self.assertRaises(
|
|
exception.Unauthorized,
|
|
client.list_networks)
|
|
|
|
@mock.patch.object(client.Client, "list_networks",
|
|
side_effect=exceptions.Unauthorized())
|
|
def test_Unauthorized_admin(self, mock_list_networks):
|
|
my_context = context.RequestContext('userid', uuids.my_tenant,
|
|
auth_token='token',
|
|
is_admin=True)
|
|
client = neutronapi.get_client(my_context)
|
|
self.assertRaises(
|
|
exception.NeutronAdminCredentialConfigurationInvalid,
|
|
client.list_networks)
|
|
|
|
@mock.patch.object(client.Client, "create_port",
|
|
side_effect=exceptions.Forbidden())
|
|
def test_Forbidden(self, mock_create_port):
|
|
my_context = context.RequestContext('userid', uuids.my_tenant,
|
|
auth_token='token',
|
|
is_admin=False)
|
|
client = neutronapi.get_client(my_context)
|
|
exc = self.assertRaises(
|
|
exception.Forbidden,
|
|
client.create_port)
|
|
self.assertIsInstance(exc.format_message(), str)
|
|
|
|
def test_withtoken_context_is_admin(self):
|
|
self.flags(endpoint_override='http://anyhost/', group='neutron')
|
|
self.flags(timeout=30, group='neutron')
|
|
# No auth_token set but is_admin will load auth from config.
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
is_admin=True)
|
|
with mock.patch.object(neutronapi, '_load_auth_plugin') as mock_auth:
|
|
cl = neutronapi.get_client(my_context)
|
|
|
|
self.assertEqual(CONF.neutron.endpoint_override,
|
|
cl.httpclient.endpoint_override)
|
|
self.assertEqual(mock_auth.return_value.auth_token,
|
|
cl.httpclient.auth.auth_token)
|
|
self.assertEqual(CONF.neutron.timeout, cl.httpclient.session.timeout)
|
|
|
|
def test_withouttoken_keystone_connection_error(self):
|
|
self.flags(endpoint_override='http://anyhost/', group='neutron')
|
|
my_context = context.RequestContext('userid', uuids.my_tenant)
|
|
self.assertRaises(NEUTRON_CLIENT_EXCEPTION,
|
|
neutronapi.get_client,
|
|
my_context)
|
|
|
|
@mock.patch('nova.network.neutron._ADMIN_AUTH')
|
|
@mock.patch.object(client.Client, "list_networks", new=mock.Mock())
|
|
def test_reuse_admin_token(self, m):
|
|
self.flags(endpoint_override='http://anyhost/', group='neutron')
|
|
my_context = context.RequestContext('userid', uuids.my_tenant,
|
|
auth_token='token')
|
|
|
|
tokens = ['new_token2', 'new_token1']
|
|
|
|
def token_vals(*args, **kwargs):
|
|
return tokens.pop()
|
|
|
|
m.get_token.side_effect = token_vals
|
|
|
|
client1 = neutronapi.get_client(my_context, True)
|
|
client1.list_networks(retrieve_all=False)
|
|
self.assertEqual('new_token1', client1.httpclient.auth.get_token(None))
|
|
|
|
client1 = neutronapi.get_client(my_context, True)
|
|
client1.list_networks(retrieve_all=False)
|
|
self.assertEqual('new_token2', client1.httpclient.auth.get_token(None))
|
|
|
|
@mock.patch('nova.network.neutron.LOG.error')
|
|
@mock.patch.object(ks_loading, 'load_auth_from_conf_options')
|
|
def test_load_auth_plugin_failed(self, mock_load_from_conf, mock_log_err):
|
|
mock_load_from_conf.return_value = None
|
|
from neutronclient.common import exceptions as neutron_client_exc
|
|
self.assertRaises(neutron_client_exc.Unauthorized,
|
|
neutronapi._load_auth_plugin, CONF)
|
|
mock_log_err.assert_called()
|
|
self.assertIn('The [neutron] section of your nova configuration file',
|
|
mock_log_err.call_args[0][0])
|
|
|
|
@mock.patch.object(client.Client, "list_networks",
|
|
side_effect=exceptions.Unauthorized())
|
|
def test_wrapper_exception_translation(self, m):
|
|
my_context = context.RequestContext('userid', 'my_tenantid',
|
|
auth_token='token')
|
|
client = neutronapi.get_client(my_context)
|
|
self.assertRaises(
|
|
exception.Unauthorized,
|
|
client.list_networks)
|
|
|
|
def test_neutron_http_retries(self):
|
|
retries = 42
|
|
self.flags(http_retries=retries, group='neutron')
|
|
my_context = context.RequestContext('userid',
|
|
uuids.my_tenant,
|
|
auth_token='token')
|
|
cl = neutronapi.get_client(my_context)
|
|
self.assertEqual(retries, cl.httpclient.connect_retries)
|
|
|
|
|
|
class TestAPIBase(test.TestCase):
|
|
|
|
def setUp(self):
|
|
super(TestAPIBase, self).setUp()
|
|
self.api = neutronapi.API()
|
|
self.context = context.RequestContext(
|
|
'userid', uuids.my_tenant,
|
|
auth_token='bff4a5a6b9eb4ea2a6efec6eefb77936')
|
|
self.tenant_id = '9d049e4b60b64716978ab415e6fbd5c0'
|
|
self.instance = {'project_id': self.tenant_id,
|
|
'uuid': uuids.fake,
|
|
'display_name': 'test_instance',
|
|
'hostname': 'test-instance',
|
|
'availability_zone': 'nova',
|
|
'host': 'some_host',
|
|
'info_cache': {'network_info': []},
|
|
'security_groups': []}
|
|
self.instance2 = {'project_id': self.tenant_id,
|
|
'uuid': uuids.fake,
|
|
'display_name': 'test_instance2',
|
|
'availability_zone': 'nova',
|
|
'info_cache': {'network_info': []},
|
|
'security_groups': []}
|
|
self.nets1 = [{'id': uuids.my_netid1,
|
|
'name': 'my_netname1',
|
|
'subnets': ['mysubnid1'],
|
|
'tenant_id': uuids.my_tenant}]
|
|
self.nets2 = []
|
|
self.nets2.append(self.nets1[0])
|
|
self.nets2.append({'id': uuids.my_netid2,
|
|
'name': 'my_netname2',
|
|
'subnets': ['mysubnid2'],
|
|
'tenant_id': uuids.my_tenant})
|
|
self.nets3 = self.nets2 + [{'id': uuids.my_netid3,
|
|
'name': 'my_netname3',
|
|
'subnets': ['mysubnid3'],
|
|
'tenant_id': uuids.my_tenant}]
|
|
self.nets4 = [{'id': 'his_netid4',
|
|
'name': 'his_netname4',
|
|
'tenant_id': 'his_tenantid'}]
|
|
# A network request with external networks
|
|
self.nets5 = self.nets1 + [{'id': 'the-external-one',
|
|
'name': 'out-of-this-world',
|
|
'subnets': ['mysubnid5'],
|
|
'router:external': True,
|
|
'tenant_id': 'should-be-an-admin'}]
|
|
# A network request with a duplicate
|
|
self.nets6 = []
|
|
self.nets6.append(self.nets1[0])
|
|
self.nets6.append(self.nets1[0])
|
|
# A network request with a combo
|
|
self.nets7 = []
|
|
self.nets7.append(self.nets2[1])
|
|
self.nets7.append(self.nets1[0])
|
|
self.nets7.append(self.nets2[1])
|
|
self.nets7.append(self.nets1[0])
|
|
# A network request with only external network
|
|
self.nets8 = [self.nets5[1]]
|
|
# An empty network
|
|
self.nets9 = []
|
|
# A network that is both shared and external
|
|
self.nets10 = [{'id': 'net_id', 'name': 'net_name',
|
|
'router:external': True, 'shared': True,
|
|
'subnets': ['mysubnid10']}]
|
|
# A network with non-blank dns_domain to test _update_port_dns_name
|
|
self.nets11 = [{'id': uuids.my_netid1,
|
|
'name': 'my_netname1',
|
|
'subnets': ['mysubnid1'],
|
|
'tenant_id': uuids.my_tenant,
|
|
'dns_domain': 'my-domain.org.'}]
|
|
|
|
self.nets = [self.nets1, self.nets2, self.nets3, self.nets4,
|
|
self.nets5, self.nets6, self.nets7, self.nets8,
|
|
self.nets9, self.nets10, self.nets11]
|
|
|
|
self.port_address = '10.0.1.2'
|
|
self.port_data1 = [{'network_id': uuids.my_netid1,
|
|
'device_id': self.instance2['uuid'],
|
|
'tenant_id': self.tenant_id,
|
|
'device_owner': 'compute:nova',
|
|
'id': uuids.portid_1,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'status': 'DOWN',
|
|
'admin_state_up': True,
|
|
'fixed_ips': [{'ip_address': self.port_address,
|
|
'subnet_id': 'my_subid1'}],
|
|
'mac_address': 'my_mac1', }]
|
|
self.float_data1 = [{'port_id': uuids.portid_1,
|
|
'fixed_ip_address': self.port_address,
|
|
'floating_ip_address': '172.0.1.2'}]
|
|
self.dhcp_port_data1 = [{'fixed_ips': [{'ip_address': '10.0.1.9',
|
|
'subnet_id': 'my_subid1'}],
|
|
'status': 'ACTIVE',
|
|
'admin_state_up': True}]
|
|
self.port_address2 = '10.0.2.2'
|
|
self.port_data2 = []
|
|
self.port_data2.append(self.port_data1[0])
|
|
self.port_data2.append({'network_id': uuids.my_netid2,
|
|
'device_id': self.instance['uuid'],
|
|
'tenant_id': self.tenant_id,
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'device_owner': 'compute:nova',
|
|
'id': uuids.portid_2,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'fixed_ips':
|
|
[{'ip_address': self.port_address2,
|
|
'subnet_id': 'my_subid2'}],
|
|
'mac_address': 'my_mac2', })
|
|
self.float_data2 = []
|
|
self.float_data2.append(self.float_data1[0])
|
|
self.float_data2.append({'port_id': uuids.portid_2,
|
|
'fixed_ip_address': '10.0.2.2',
|
|
'floating_ip_address': '172.0.2.2'})
|
|
self.port_data3 = [{'network_id': uuids.my_netid1,
|
|
'device_id': 'device_id3',
|
|
'tenant_id': self.tenant_id,
|
|
'status': 'DOWN',
|
|
'admin_state_up': True,
|
|
'device_owner': 'compute:nova',
|
|
'id': uuids.portid_3,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'fixed_ips': [], # no fixed ip
|
|
'mac_address': 'my_mac3', }]
|
|
self.subnet_data1 = [{'id': 'my_subid1',
|
|
'cidr': '10.0.1.0/24',
|
|
'network_id': uuids.my_netid1,
|
|
'gateway_ip': '10.0.1.1',
|
|
'dns_nameservers': ['8.8.1.1', '8.8.1.2']}]
|
|
self.subnet_data2 = []
|
|
self.subnet_data_n = [{'id': 'my_subid1',
|
|
'cidr': '10.0.1.0/24',
|
|
'network_id': uuids.my_netid1,
|
|
'gateway_ip': '10.0.1.1',
|
|
'dns_nameservers': ['8.8.1.1', '8.8.1.2']},
|
|
{'id': 'my_subid2',
|
|
'cidr': '20.0.1.0/24',
|
|
'network_id': uuids.my_netid2,
|
|
'gateway_ip': '20.0.1.1',
|
|
'dns_nameservers': ['8.8.1.1', '8.8.1.2']}]
|
|
self.subnet_data2.append({'id': 'my_subid2',
|
|
'cidr': '10.0.2.0/24',
|
|
'network_id': uuids.my_netid2,
|
|
'gateway_ip': '10.0.2.1',
|
|
'dns_nameservers': ['8.8.2.1', '8.8.2.2']})
|
|
|
|
self.fip_pool = {'id': '4fdbfd74-eaf8-4884-90d9-00bd6f10c2d3',
|
|
'name': 'ext_net',
|
|
'router:external': True,
|
|
'tenant_id': 'admin_tenantid'}
|
|
self.fip_pool_nova = {'id': '435e20c3-d9f1-4f1b-bee5-4611a1dd07db',
|
|
'name': 'nova',
|
|
'router:external': True,
|
|
'tenant_id': 'admin_tenantid'}
|
|
self.fip_unassociated = {'tenant_id': uuids.my_tenant,
|
|
'id': uuids.fip_id1,
|
|
'floating_ip_address': '172.24.4.227',
|
|
'floating_network_id': self.fip_pool['id'],
|
|
'port_id': None,
|
|
'fixed_ip_address': None,
|
|
'router_id': None}
|
|
fixed_ip_address = self.port_data2[1]['fixed_ips'][0]['ip_address']
|
|
self.fip_associated = {'tenant_id': uuids.my_tenant,
|
|
'id': uuids.fip_id2,
|
|
'floating_ip_address': '172.24.4.228',
|
|
'floating_network_id': self.fip_pool['id'],
|
|
'port_id': self.port_data2[1]['id'],
|
|
'fixed_ip_address': fixed_ip_address,
|
|
'router_id': 'router_id1'}
|
|
self._returned_nw_info = []
|
|
|
|
self.arqs = [
|
|
{'uuid': uuids.arq_uuid1,
|
|
'device_profile_name': "smart_nic",
|
|
'device_profile_group_id': '5',
|
|
'state': 'Bound',
|
|
'device_rp_uuid': uuids.resource_provider_uuid1,
|
|
'hostname': "host_nodename",
|
|
'instance_uuid': uuids.instance_uuid,
|
|
'attach_handle_info': {
|
|
'bus': '0c', 'device': '0',
|
|
'domain': '0000', 'function': '0',
|
|
'physical_network': 'physicalnet1'
|
|
},
|
|
'attach_handle_type': 'PCI'},
|
|
{'uuid': uuids.arq_uuid2,
|
|
'device_profile_name': "smart_nic",
|
|
'device_profile_group_id': '5',
|
|
'state': 'Bound',
|
|
'device_rp_uuid': uuids.resource_provider_uuid2,
|
|
'hostname': "host_nodename",
|
|
'instance_uuid': uuids.instance_uuid,
|
|
'attach_handle_info': {
|
|
'bus': '0c', 'device': '1',
|
|
'domain': '0000', 'function': '0',
|
|
'physical_network': 'physicalnet1'},
|
|
'attach_handle_type': 'PCI'}
|
|
]
|
|
|
|
def _fake_instance_object(self, instance):
|
|
return fake_instance.fake_instance_obj(self.context, **instance)
|
|
|
|
def _fake_instance_info_cache(self, nw_info, instance_uuid=None):
|
|
info_cache = {}
|
|
if instance_uuid is None:
|
|
info_cache['instance_uuid'] = uuids.fake
|
|
else:
|
|
info_cache['instance_uuid'] = instance_uuid
|
|
info_cache['deleted'] = False
|
|
info_cache['created_at'] = timeutils.utcnow()
|
|
info_cache['deleted_at'] = timeutils.utcnow()
|
|
info_cache['updated_at'] = timeutils.utcnow()
|
|
info_cache['network_info'] = model.NetworkInfo.hydrate(
|
|
str(jsonutils.dumps(nw_info)))
|
|
return info_cache
|
|
|
|
def _fake_instance_object_with_info_cache(self, instance):
|
|
expected_attrs = ['info_cache']
|
|
instance = objects.Instance._from_db_object(self.context,
|
|
objects.Instance(), fake_instance.fake_db_instance(**instance),
|
|
expected_attrs=expected_attrs)
|
|
return instance
|
|
|
|
def _test_allocate_for_instance_with_virtual_interface(
|
|
self, net_idx=1, **kwargs):
|
|
self._vifs_created = []
|
|
|
|
def _new_vif(*args):
|
|
m = mock.MagicMock()
|
|
self._vifs_created.append(m)
|
|
return m
|
|
|
|
with mock.patch('nova.objects.VirtualInterface') as mock_vif:
|
|
mock_vif.side_effect = _new_vif
|
|
requested_networks = kwargs.pop('requested_networks', None)
|
|
|
|
return self._test_allocate_for_instance(
|
|
net_idx=net_idx, requested_networks=requested_networks,
|
|
**kwargs)
|
|
|
|
@mock.patch.object(neutronapi.API, '_populate_neutron_extension_values')
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info',
|
|
return_value=None)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_allocate_for_instance(self, mock_get_client, mock_get_nw,
|
|
mock_populate, net_idx=1,
|
|
requested_networks=None,
|
|
exception=None,
|
|
context=None,
|
|
**kwargs):
|
|
ctxt = context or self.context
|
|
self.instance = self._fake_instance_object(self.instance)
|
|
self.instance2 = self._fake_instance_object(self.instance2)
|
|
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
|
|
bind_host_id = kwargs.get('bind_host_id')
|
|
|
|
has_dns_extension = False
|
|
if kwargs.get('dns_extension'):
|
|
has_dns_extension = True
|
|
self.api.extensions = {
|
|
constants.DNS_INTEGRATION: {
|
|
'alias': constants.DNS_INTEGRATION,
|
|
},
|
|
}
|
|
|
|
# Net idx is 1-based for compatibility with existing unit tests
|
|
nets = self.nets[net_idx - 1]
|
|
ports = {}
|
|
fixed_ips = {}
|
|
|
|
req_net_ids = []
|
|
ordered_networks = []
|
|
|
|
expected_show_port_calls = self._stub_allocate_for_instance_show_port(
|
|
nets, ports, fixed_ips, req_net_ids, ordered_networks,
|
|
requested_networks, mocked_client, **kwargs)
|
|
|
|
populate_values = []
|
|
update_port_values = []
|
|
try:
|
|
if kwargs.get('_break') == 'pre_list_networks':
|
|
raise test.TestingException()
|
|
|
|
expected_list_networks_calls = (
|
|
self._stub_allocate_for_instance_list_networks(
|
|
req_net_ids, nets, mocked_client))
|
|
|
|
pre_create_port = (
|
|
kwargs.get('_break') == 'post_list_networks' or
|
|
((requested_networks is None or
|
|
requested_networks.as_tuples() == [(None, None, None)]) and
|
|
len(nets) > 1) or
|
|
kwargs.get('_break') == 'post_list_extensions')
|
|
|
|
if pre_create_port:
|
|
raise test.TestingException()
|
|
|
|
expected_create_port_calls = (
|
|
self._stub_allocate_for_instance_create_port(
|
|
ordered_networks, fixed_ips, nets, mocked_client))
|
|
|
|
preexisting_port_ids = []
|
|
ports_in_requested_net_order = []
|
|
nets_in_requested_net_order = []
|
|
index = 0
|
|
expected_populate_calls = []
|
|
expected_update_port_calls = []
|
|
for request in ordered_networks:
|
|
index += 1
|
|
port_req_body = {
|
|
'port': {
|
|
'device_id': self.instance.uuid,
|
|
'device_owner': 'compute:nova',
|
|
},
|
|
}
|
|
# Network lookup for available network_id
|
|
network = None
|
|
for net in nets:
|
|
if net['id'] == request.network_id:
|
|
network = net
|
|
break
|
|
# if net_id did not pass validate_networks() and not available
|
|
# here then skip it safely not continuing with a None Network
|
|
else:
|
|
continue
|
|
|
|
populate_values.append(None)
|
|
expected_populate_calls.append(
|
|
mock.call(mock.ANY, self.instance, mock.ANY, mock.ANY,
|
|
network=network, neutron=mocked_client,
|
|
bind_host_id=bind_host_id,
|
|
port_arq=mock.ANY))
|
|
|
|
if not request.port_id:
|
|
port_id = uuids.fake
|
|
update_port_res = {'port': {
|
|
'id': port_id,
|
|
'mac_address': 'fakemac%i' % index}}
|
|
ports_in_requested_net_order.append(port_id)
|
|
if kwargs.get('_break') == 'mac' + request.network_id:
|
|
if populate_values:
|
|
mock_populate.side_effect = populate_values
|
|
if update_port_values:
|
|
mocked_client.update_port.side_effect = (
|
|
update_port_values)
|
|
raise test.TestingException()
|
|
else:
|
|
ports_in_requested_net_order.append(request.port_id)
|
|
preexisting_port_ids.append(request.port_id)
|
|
port_id = request.port_id
|
|
update_port_res = {'port': ports[port_id]}
|
|
|
|
new_mac = port_req_body['port'].get('mac_address')
|
|
if new_mac:
|
|
update_port_res['port']['mac_address'] = new_mac
|
|
|
|
update_port_values.append(update_port_res)
|
|
expected_update_port_calls.append(
|
|
mock.call(port_id, port_req_body))
|
|
|
|
if has_dns_extension:
|
|
if net_idx == 11:
|
|
port_req_body_dns = {
|
|
'port': {
|
|
'dns_name': self.instance.hostname
|
|
}
|
|
}
|
|
res_port_dns = {
|
|
'port': {
|
|
'id': ports_in_requested_net_order[-1]
|
|
}
|
|
}
|
|
update_port_values.append(res_port_dns)
|
|
expected_update_port_calls.append(
|
|
mock.call(ports_in_requested_net_order[-1],
|
|
port_req_body_dns))
|
|
nets_in_requested_net_order.append(network)
|
|
mock_get_nw.return_value = self._returned_nw_info
|
|
except test.TestingException:
|
|
pass
|
|
|
|
mock_populate.side_effect = populate_values
|
|
mocked_client.update_port.side_effect = update_port_values
|
|
|
|
# Call the allocate_for_instance method
|
|
nw_info = None
|
|
if exception:
|
|
self.assertRaises(exception, self.api.allocate_for_instance,
|
|
ctxt, self.instance,
|
|
requested_networks, bind_host_id=bind_host_id)
|
|
else:
|
|
nw_info = self.api.allocate_for_instance(
|
|
ctxt, self.instance, requested_networks,
|
|
bind_host_id=bind_host_id)
|
|
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(ctxt), mock.call(ctxt, admin=True)],
|
|
any_order=True)
|
|
|
|
if requested_networks:
|
|
mocked_client.show_port.assert_has_calls(expected_show_port_calls)
|
|
self.assertEqual(len(expected_show_port_calls),
|
|
mocked_client.show_port.call_count)
|
|
|
|
if kwargs.get('_break') == 'pre_list_networks':
|
|
return nw_info, mocked_client
|
|
|
|
mocked_client.list_networks.assert_has_calls(
|
|
expected_list_networks_calls)
|
|
self.assertEqual(len(expected_list_networks_calls),
|
|
mocked_client.list_networks.call_count)
|
|
|
|
if pre_create_port:
|
|
return nw_info, mocked_client
|
|
|
|
mocked_client.create_port.assert_has_calls(
|
|
expected_create_port_calls)
|
|
self.assertEqual(len(expected_create_port_calls),
|
|
mocked_client.create_port.call_count)
|
|
|
|
mocked_client.update_port.assert_has_calls(
|
|
expected_update_port_calls)
|
|
self.assertEqual(len(expected_update_port_calls),
|
|
mocked_client.update_port.call_count)
|
|
|
|
mock_populate.assert_has_calls(expected_populate_calls)
|
|
self.assertEqual(len(expected_populate_calls),
|
|
mock_populate.call_count)
|
|
|
|
if mock_get_nw.return_value is None:
|
|
mock_get_nw.assert_not_called()
|
|
else:
|
|
mock_get_nw.assert_called_once_with(
|
|
mock.ANY, self.instance, networks=nets_in_requested_net_order,
|
|
port_ids=ports_in_requested_net_order,
|
|
admin_client=mocked_client,
|
|
preexisting_port_ids=preexisting_port_ids)
|
|
|
|
return nw_info, mocked_client
|
|
|
|
def _stub_allocate_for_instance_show_port(self, nets, ports, fixed_ips,
|
|
req_net_ids, ordered_networks, requested_networks,
|
|
mocked_client, **kwargs):
|
|
expected_show_port_calls = []
|
|
|
|
if requested_networks:
|
|
show_port_values = []
|
|
for request in requested_networks:
|
|
if request.port_id:
|
|
if request.port_id == uuids.portid_3:
|
|
show_port_values.append(
|
|
{'port': {'id': uuids.portid_3,
|
|
'network_id': uuids.my_netid1,
|
|
'tenant_id': self.tenant_id,
|
|
'mac_address': 'my_mac1',
|
|
'device_id': kwargs.get('_device') and
|
|
self.instance2.uuid or
|
|
''}})
|
|
ports[uuids.my_netid1] = [self.port_data1[0],
|
|
self.port_data3[0]]
|
|
ports[request.port_id] = self.port_data3[0]
|
|
request.network_id = uuids.my_netid1
|
|
elif request.port_id == uuids.non_existent_uuid:
|
|
show_port_values.append(
|
|
exceptions.PortNotFoundClient(status_code=404))
|
|
else:
|
|
show_port_values.append(
|
|
{'port': {'id': uuids.portid_1,
|
|
'network_id': uuids.my_netid1,
|
|
'tenant_id': self.tenant_id,
|
|
'mac_address': 'my_mac1',
|
|
'device_id': kwargs.get('_device') and
|
|
self.instance2.uuid or
|
|
'',
|
|
'dns_name': kwargs.get('_dns_name') or
|
|
''}})
|
|
ports[request.port_id] = self.port_data1[0]
|
|
request.network_id = uuids.my_netid1
|
|
expected_show_port_calls.append(mock.call(request.port_id))
|
|
else:
|
|
fixed_ips[request.network_id] = request.address
|
|
req_net_ids.append(request.network_id)
|
|
ordered_networks.append(request)
|
|
if show_port_values:
|
|
mocked_client.show_port.side_effect = show_port_values
|
|
else:
|
|
for n in nets:
|
|
ordered_networks.append(
|
|
objects.NetworkRequest(network_id=n['id']))
|
|
|
|
return expected_show_port_calls
|
|
|
|
def _stub_allocate_for_instance_list_networks(self, req_net_ids, nets,
|
|
mocked_client):
|
|
if req_net_ids:
|
|
expected_list_networks_calls = [mock.call(id=req_net_ids)]
|
|
mocked_client.list_networks.return_value = {'networks': nets}
|
|
else:
|
|
expected_list_networks_calls = [
|
|
mock.call(tenant_id=self.instance.project_id,
|
|
shared=False),
|
|
mock.call(shared=True)]
|
|
mocked_client.list_networks.side_effect = [
|
|
{'networks': nets}, {'networks': []}]
|
|
|
|
return expected_list_networks_calls
|
|
|
|
def _stub_allocate_for_instance_create_port(self, ordered_networks,
|
|
fixed_ips, nets, mocked_client):
|
|
create_port_values = []
|
|
expected_create_port_calls = []
|
|
for request in ordered_networks:
|
|
if not request.port_id:
|
|
# Check network is available, skip if not
|
|
network = None
|
|
for net in nets:
|
|
if net['id'] == request.network_id:
|
|
network = net
|
|
break
|
|
if network is None:
|
|
continue
|
|
|
|
port_req_body_create = {'port': {'device_id':
|
|
self.instance.uuid}}
|
|
request.address = fixed_ips.get(request.network_id)
|
|
if request.address:
|
|
port_req_body_create['port']['fixed_ips'] = [
|
|
{'ip_address': str(request.address)}]
|
|
port_req_body_create['port']['network_id'] = (
|
|
request.network_id)
|
|
port_req_body_create['port']['admin_state_up'] = True
|
|
port_req_body_create['port']['tenant_id'] = (
|
|
self.instance.project_id)
|
|
res_port = {'port': {'id': uuids.fake}}
|
|
expected_create_port_calls.append(
|
|
mock.call(port_req_body_create))
|
|
create_port_values.append(res_port)
|
|
mocked_client.create_port.side_effect = create_port_values
|
|
|
|
return expected_create_port_calls
|
|
|
|
|
|
class TestAPI(TestAPIBase):
|
|
"""Used to test Neutron V2 API."""
|
|
|
|
@mock.patch.object(db_api, 'instance_info_cache_get')
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_instance_nw_info(self, number, mock_get_client,
|
|
mock_cache_update, mock_cache_get):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_data = number == 1 and self.port_data1 or self.port_data2
|
|
net_info_cache = []
|
|
for port in port_data:
|
|
net_info_cache.append({"network": {"id": port['network_id']},
|
|
"id": port['id']})
|
|
|
|
list_ports_values = [{'ports': port_data}]
|
|
expected_list_ports_calls = [mock.call(
|
|
tenant_id=self.instance['project_id'],
|
|
device_id=self.instance['uuid'])]
|
|
|
|
net_ids = [port['network_id'] for port in port_data]
|
|
nets = number == 1 and self.nets1 or self.nets2
|
|
mocked_client.list_networks.return_value = {'networks': nets}
|
|
|
|
list_subnets_values = []
|
|
expected_list_subnets_calls = []
|
|
list_floatingips_values = []
|
|
expected_list_floatingips_calls = []
|
|
|
|
for i in range(1, number + 1):
|
|
float_data = number == 1 and self.float_data1 or self.float_data2
|
|
for ip in port_data[i - 1]['fixed_ips']:
|
|
float_data = [x for x in float_data
|
|
if x['fixed_ip_address'] == ip['ip_address']]
|
|
list_floatingips_values.append({'floatingips': float_data})
|
|
expected_list_floatingips_calls.append(
|
|
mock.call(fixed_ip_address=ip['ip_address'],
|
|
port_id=port_data[i - 1]['id']))
|
|
subnet_data = i == 1 and self.subnet_data1 or self.subnet_data2
|
|
list_subnets_values.append({'subnets': subnet_data})
|
|
expected_list_subnets_calls.append(
|
|
mock.call(id=['my_subid%s' % i]))
|
|
list_ports_values.append({'ports': []})
|
|
expected_list_ports_calls.append(mock.call(
|
|
network_id=subnet_data[0]['network_id'],
|
|
device_owner='network:dhcp'))
|
|
|
|
mocked_client.list_ports.side_effect = list_ports_values
|
|
mocked_client.list_subnets.side_effect = list_subnets_values
|
|
mocked_client.list_floatingips.side_effect = list_floatingips_values
|
|
|
|
self.instance['info_cache'] = self._fake_instance_info_cache(
|
|
net_info_cache, self.instance['uuid'])
|
|
mock_cache_get.return_value = self.instance['info_cache']
|
|
instance = self._fake_instance_object_with_info_cache(self.instance)
|
|
|
|
nw_inf = self.api.get_instance_nw_info(self.context, instance)
|
|
|
|
mock_get_client.assert_any_call(mock.ANY, admin=True)
|
|
mock_cache_update.assert_called_once_with(
|
|
mock.ANY, self.instance['uuid'], mock.ANY)
|
|
mock_cache_get.assert_called_once_with(mock.ANY, self.instance['uuid'])
|
|
mocked_client.list_ports.assert_has_calls(expected_list_ports_calls)
|
|
self.assertEqual(len(expected_list_ports_calls),
|
|
mocked_client.list_ports.call_count)
|
|
mocked_client.list_subnets.assert_has_calls(
|
|
expected_list_subnets_calls)
|
|
self.assertEqual(len(expected_list_subnets_calls),
|
|
mocked_client.list_subnets.call_count)
|
|
mocked_client.list_floatingips.assert_has_calls(
|
|
expected_list_floatingips_calls)
|
|
self.assertEqual(len(expected_list_floatingips_calls),
|
|
mocked_client.list_floatingips.call_count)
|
|
mocked_client.list_networks.assert_called_once_with(id=net_ids)
|
|
|
|
for i in range(0, number):
|
|
self._verify_nw_info(nw_inf, i)
|
|
|
|
def _verify_nw_info(self, nw_inf, index=0):
|
|
id_suffix = index + 1
|
|
self.assertEqual('10.0.%s.2' % id_suffix,
|
|
nw_inf.fixed_ips()[index]['address'])
|
|
self.assertEqual('172.0.%s.2' % id_suffix,
|
|
nw_inf.fixed_ips()[index].floating_ip_addresses()[0])
|
|
self.assertEqual('my_netname%s' % id_suffix,
|
|
nw_inf[index]['network']['label'])
|
|
self.assertEqual(getattr(uuids, 'portid_%s' % id_suffix),
|
|
nw_inf[index]['id'])
|
|
self.assertEqual('my_mac%s' % id_suffix, nw_inf[index]['address'])
|
|
self.assertEqual('10.0.%s.0/24' % id_suffix,
|
|
nw_inf[index]['network']['subnets'][0]['cidr'])
|
|
|
|
ip_addr = model.IP(address='8.8.%s.1' % id_suffix,
|
|
version=4, type='dns')
|
|
self.assertIn(ip_addr, nw_inf[index]['network']['subnets'][0]['dns'])
|
|
|
|
def test_get_instance_nw_info_1(self):
|
|
# Test to get one port in one network and subnet.
|
|
self._test_get_instance_nw_info(1)
|
|
|
|
def test_get_instance_nw_info_2(self):
|
|
# Test to get one port in each of two networks and subnets.
|
|
self._test_get_instance_nw_info(2)
|
|
|
|
def test_get_instance_nw_info_with_nets_add_interface(self):
|
|
# This tests that adding an interface to an instance does not
|
|
# remove the first instance from the instance.
|
|
network_model = model.Network(id='network_id',
|
|
bridge='br-int',
|
|
injected='injected',
|
|
label='fake_network',
|
|
tenant_id='fake_tenant')
|
|
network_cache = {'info_cache': {
|
|
'network_info': [{'id': self.port_data2[0]['id'],
|
|
'address': 'mac_address',
|
|
'network': network_model,
|
|
'type': 'ovs',
|
|
'ovs_interfaceid': 'ovs_interfaceid',
|
|
'devname': 'devname'}]}}
|
|
|
|
self._test_get_instance_nw_info_helper(
|
|
network_cache, self.port_data2, networks=self.nets2,
|
|
port_ids=[self.port_data2[1]['id']])
|
|
|
|
def test_get_instance_nw_info_remove_ports_from_neutron(self):
|
|
# This tests that when a port is removed in neutron it
|
|
# is also removed from the nova.
|
|
network_model = model.Network(id=self.port_data2[0]['network_id'],
|
|
bridge='br-int',
|
|
injected='injected',
|
|
label='fake_network',
|
|
tenant_id='fake_tenant')
|
|
network_cache = {'info_cache': {
|
|
'network_info': [{'id': 'network_id',
|
|
'address': 'mac_address',
|
|
'network': network_model,
|
|
'type': 'ovs',
|
|
'ovs_interfaceid': 'ovs_interfaceid',
|
|
'devname': 'devname'}]}}
|
|
|
|
self._test_get_instance_nw_info_helper(network_cache,
|
|
self.port_data2)
|
|
|
|
def test_get_instance_nw_info_ignores_neutron_ports(self):
|
|
# Tests that only ports in the network_cache are updated
|
|
# and ports returned from neutron that match the same
|
|
# instance_id/device_id are ignored.
|
|
port_data2 = copy.copy(self.port_data2)
|
|
|
|
# set device_id on the ports to be the same.
|
|
port_data2[1]['device_id'] = port_data2[0]['device_id']
|
|
network_model = model.Network(id='network_id',
|
|
bridge='br-int',
|
|
injected='injected',
|
|
label='fake_network',
|
|
tenant_id='fake_tenant')
|
|
network_cache = {'info_cache': {
|
|
'network_info': [{'id': 'network_id',
|
|
'address': 'mac_address',
|
|
'network': network_model,
|
|
'type': 'ovs',
|
|
'ovs_interfaceid': 'ovs_interfaceid',
|
|
'devname': 'devname'}]}}
|
|
|
|
self._test_get_instance_nw_info_helper(network_cache,
|
|
port_data2)
|
|
|
|
def test_get_instance_nw_info_ignores_neutron_ports_empty_cache(self):
|
|
# Tests that ports returned from neutron that match the same
|
|
# instance_id/device_id are ignored when the instance info cache is
|
|
# empty.
|
|
port_data2 = copy.copy(self.port_data2)
|
|
|
|
# set device_id on the ports to be the same.
|
|
port_data2[1]['device_id'] = port_data2[0]['device_id']
|
|
network_cache = {'info_cache': {'network_info': []}}
|
|
|
|
self._test_get_instance_nw_info_helper(network_cache,
|
|
port_data2)
|
|
|
|
@mock.patch.object(db_api, 'instance_info_cache_get')
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_instance_nw_info_helper(self, network_cache,
|
|
current_neutron_ports,
|
|
mock_get_client,
|
|
mock_cache_update, mock_cache_get,
|
|
networks=None, port_ids=None):
|
|
"""Helper function to test get_instance_nw_info.
|
|
|
|
:param network_cache - data already in the nova network cache.
|
|
:param current_neutron_ports - updated list of ports from neutron.
|
|
:param networks - networks of ports being added to instance.
|
|
:param port_ids - new ports being added to instance.
|
|
"""
|
|
|
|
# keep a copy of the original ports/networks to pass to
|
|
# get_instance_nw_info() as the code below changes them.
|
|
original_port_ids = copy.copy(port_ids)
|
|
original_networks = copy.copy(networks)
|
|
|
|
api = neutronapi.API()
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
|
|
list_ports_values = [{'ports': current_neutron_ports}]
|
|
expected_list_ports_calls = [
|
|
mock.call(tenant_id=self.instance['project_id'],
|
|
device_id=self.instance['uuid'])]
|
|
|
|
ifaces = network_cache['info_cache']['network_info']
|
|
|
|
if port_ids is None:
|
|
port_ids = [iface['id'] for iface in ifaces]
|
|
net_ids = [iface['network']['id'] for iface in ifaces]
|
|
nets = [{'id': iface['network']['id'],
|
|
'name': iface['network']['label'],
|
|
'tenant_id': iface['network']['meta']['tenant_id']}
|
|
for iface in ifaces]
|
|
|
|
if networks is None:
|
|
list_networks_values = []
|
|
expected_list_networks_calls = []
|
|
if ifaces:
|
|
list_networks_values.append({'networks': nets})
|
|
expected_list_networks_calls.append(mock.call(id=net_ids))
|
|
else:
|
|
non_shared_nets = [
|
|
{'id': iface['network']['id'],
|
|
'name': iface['network']['label'],
|
|
'tenant_id': iface['network']['meta']['tenant_id']}
|
|
for iface in ifaces if not iface['shared']]
|
|
shared_nets = [
|
|
{'id': iface['network']['id'],
|
|
'name': iface['network']['label'],
|
|
'tenant_id': iface['network']['meta']['tenant_id']}
|
|
for iface in ifaces if iface['shared']]
|
|
list_networks_values.extend([
|
|
{'networks': non_shared_nets}, {'networks': shared_nets}])
|
|
expected_list_networks_calls.extend([
|
|
mock.call(shared=False,
|
|
tenant_id=self.instance['project_id']),
|
|
mock.call(shared=True)])
|
|
mocked_client.list_networks.side_effect = list_networks_values
|
|
else:
|
|
port_ids = [iface['id'] for iface in ifaces] + port_ids
|
|
|
|
index = 0
|
|
|
|
current_neutron_port_map = {}
|
|
for current_neutron_port in current_neutron_ports:
|
|
current_neutron_port_map[current_neutron_port['id']] = (
|
|
current_neutron_port)
|
|
|
|
list_floatingips_values = []
|
|
expected_list_floatingips_calls = []
|
|
list_subnets_values = []
|
|
expected_list_subnets_calls = []
|
|
|
|
for port_id in port_ids:
|
|
current_neutron_port = current_neutron_port_map.get(port_id)
|
|
if current_neutron_port:
|
|
for ip in current_neutron_port['fixed_ips']:
|
|
list_floatingips_values.append(
|
|
{'floatingips': [self.float_data2[index]]})
|
|
expected_list_floatingips_calls.append(
|
|
mock.call(fixed_ip_address=ip['ip_address'],
|
|
port_id=current_neutron_port['id']))
|
|
list_subnets_values.append(
|
|
{'subnets': [self.subnet_data_n[index]]})
|
|
expected_list_subnets_calls.append(
|
|
mock.call(id=[ip['subnet_id']]))
|
|
list_ports_values.append({'ports': self.dhcp_port_data1})
|
|
expected_list_ports_calls.append(
|
|
mock.call(
|
|
network_id=current_neutron_port['network_id'],
|
|
device_owner='network:dhcp'))
|
|
index += 1
|
|
|
|
mocked_client.list_floatingips.side_effect = list_floatingips_values
|
|
mocked_client.list_subnets.side_effect = list_subnets_values
|
|
mocked_client.list_ports.side_effect = list_ports_values
|
|
|
|
self.instance['info_cache'] = self._fake_instance_info_cache(
|
|
network_cache['info_cache']['network_info'], self.instance['uuid'])
|
|
mock_cache_get.return_value = self.instance['info_cache']
|
|
|
|
instance = self._fake_instance_object_with_info_cache(self.instance)
|
|
|
|
nw_infs = api.get_instance_nw_info(self.context,
|
|
instance,
|
|
networks=original_networks,
|
|
port_ids=original_port_ids)
|
|
|
|
self.assertEqual(index, len(nw_infs))
|
|
# ensure that nic ordering is preserved
|
|
for iface_index in range(index):
|
|
self.assertEqual(port_ids[iface_index],
|
|
nw_infs[iface_index]['id'])
|
|
|
|
mock_get_client.assert_any_call(mock.ANY, admin=True)
|
|
mock_cache_update.assert_called_once_with(
|
|
mock.ANY, self.instance['uuid'], mock.ANY)
|
|
mock_cache_get.assert_called_once_with(mock.ANY, self.instance['uuid'])
|
|
|
|
if networks is None:
|
|
mocked_client.list_networks.assert_has_calls(
|
|
expected_list_networks_calls)
|
|
self.assertEqual(len(expected_list_networks_calls),
|
|
mocked_client.list_networks.call_count)
|
|
|
|
mocked_client.list_floatingips.assert_has_calls(
|
|
expected_list_floatingips_calls)
|
|
self.assertEqual(len(expected_list_floatingips_calls),
|
|
mocked_client.list_floatingips.call_count)
|
|
mocked_client.list_subnets.assert_has_calls(
|
|
expected_list_subnets_calls)
|
|
self.assertEqual(len(expected_list_subnets_calls),
|
|
mocked_client.list_subnets.call_count)
|
|
mocked_client.list_ports.assert_has_calls(
|
|
expected_list_ports_calls)
|
|
self.assertEqual(len(expected_list_ports_calls),
|
|
mocked_client.list_ports.call_count)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(db_api, 'instance_info_cache_get')
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_instance_nw_info_without_subnet(
|
|
self, mock_get_client, mock_cache_update, mock_cache_get,
|
|
mock_get_physnet):
|
|
# Test get instance_nw_info for a port without subnet.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data3}
|
|
mocked_client.list_networks.return_value = {'networks': self.nets1}
|
|
|
|
net_info_cache = []
|
|
for port in self.port_data3:
|
|
net_info_cache.append({"network": {"id": port['network_id']},
|
|
"id": port['id']})
|
|
self.instance['info_cache'] = self._fake_instance_info_cache(
|
|
net_info_cache, self.instance['uuid'])
|
|
mock_cache_get.return_value = self.instance['info_cache']
|
|
|
|
instance = self._fake_instance_object_with_info_cache(self.instance)
|
|
nw_inf = self.api.get_instance_nw_info(self.context, instance)
|
|
|
|
id_suffix = 3
|
|
self.assertEqual(0, len(nw_inf.fixed_ips()))
|
|
self.assertEqual('my_netname1', nw_inf[0]['network']['label'])
|
|
self.assertEqual(uuids.portid_3, nw_inf[0]['id'])
|
|
self.assertEqual('my_mac%s' % id_suffix, nw_inf[0]['address'])
|
|
self.assertEqual(0, len(nw_inf[0]['network']['subnets']))
|
|
|
|
mock_get_client.assert_has_calls([mock.call(mock.ANY, admin=True)] * 2,
|
|
any_order=True)
|
|
mock_cache_update.assert_called_once_with(
|
|
mock.ANY, self.instance['uuid'], mock.ANY)
|
|
mock_cache_get.assert_called_once_with(mock.ANY, self.instance['uuid'])
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=self.instance['project_id'],
|
|
device_id=self.instance['uuid'])
|
|
mocked_client.list_networks.assert_called_once_with(
|
|
id=[self.port_data1[0]['network_id']])
|
|
mock_get_physnet.assert_called_once_with(
|
|
mock.ANY, mock.ANY, self.port_data1[0]['network_id'])
|
|
|
|
def test_refresh_neutron_extensions_cache(self):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mocked_client.list_extensions.return_value = {
|
|
'extensions': [{'alias': constants.DNS_INTEGRATION}]}
|
|
self.api._refresh_neutron_extensions_cache(mocked_client)
|
|
self.assertEqual(
|
|
{constants.DNS_INTEGRATION: {'alias': constants.DNS_INTEGRATION}},
|
|
self.api.extensions)
|
|
mocked_client.list_extensions.assert_called_once_with()
|
|
|
|
def test_allocate_for_instance_1(self):
|
|
# Allocate one port in one network env.
|
|
self._test_allocate_for_instance_with_virtual_interface(1)
|
|
|
|
def test_allocate_for_instance_2(self):
|
|
# Allocate one port in two networks env.
|
|
self._test_allocate_for_instance(
|
|
net_idx=2, exception=exception.NetworkAmbiguous)
|
|
|
|
def test_allocate_for_instance_accepts_only_portid(self):
|
|
# Make sure allocate_for_instance works when only a portid is provided
|
|
self._returned_nw_info = self.port_data1
|
|
result, _ = self._test_allocate_for_instance_with_virtual_interface(
|
|
requested_networks=objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1,
|
|
tag='test')]))
|
|
self.assertEqual(self.port_data1, result)
|
|
self.assertEqual(1, len(self._vifs_created))
|
|
self.assertEqual('test', self._vifs_created[0].tag)
|
|
self.assertEqual(self.instance.uuid,
|
|
self._vifs_created[0].instance_uuid)
|
|
self.assertEqual(uuids.portid_1, self._vifs_created[0].uuid)
|
|
self.assertEqual('%s/%s' % (self.port_data1[0]['mac_address'],
|
|
self.port_data1[0]['id']),
|
|
self._vifs_created[0].address)
|
|
|
|
def test_allocate_for_instance_without_requested_networks(self):
|
|
self._test_allocate_for_instance(
|
|
net_idx=3, exception=exception.NetworkAmbiguous)
|
|
|
|
def test_allocate_for_instance_with_requested_non_available_network(self):
|
|
"""verify that a non available network is ignored.
|
|
self.nets2 (net_idx=2) is composed of self.nets3[0] and self.nets3[1]
|
|
Do not create a port on a non available network self.nets3[2].
|
|
"""
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=net['id'])
|
|
for net in (self.nets3[0], self.nets3[2], self.nets3[1])])
|
|
requested_networks[0].tag = 'foo'
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=2, requested_networks=requested_networks)
|
|
self.assertEqual(2, len(self._vifs_created))
|
|
# NOTE(danms) nets3[2] is chosen above as one that won't validate,
|
|
# so we never actually run create() on the VIF.
|
|
vifs_really_created = [vif for vif in self._vifs_created
|
|
if vif.create.called]
|
|
self.assertEqual(2, len(vifs_really_created))
|
|
self.assertEqual([('foo', 'fakemac1/%s' % uuids.fake),
|
|
(None, 'fakemac3/%s' % uuids.fake)],
|
|
[(vif.tag, vif.address)
|
|
for vif in vifs_really_created])
|
|
|
|
def test_allocate_for_instance_with_requested_networks(self):
|
|
# specify only first and last network
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=net['id'])
|
|
for net in (self.nets3[1], self.nets3[0], self.nets3[2])])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=3, requested_networks=requested_networks)
|
|
|
|
def test_allocate_for_instance_with_no_subnet_defined(self):
|
|
# net_id=4 does not specify subnet and does not set the option
|
|
# port_security_disabled to True, so Neutron will not been
|
|
# able to associate the default security group to the port
|
|
# requested to be created. We expect an exception to be
|
|
# raised.
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=4, exception=exception.SecurityGroupCannotBeApplied,
|
|
_break='post_list_extensions')
|
|
|
|
def test_allocate_for_instance_with_invalid_network_id(self):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(
|
|
network_id=uuids.non_existent_uuid)])
|
|
self._test_allocate_for_instance(net_idx=9,
|
|
requested_networks=requested_networks,
|
|
exception=exception.NetworkNotFound,
|
|
_break='post_list_networks')
|
|
|
|
def test_allocate_for_instance_with_requested_networks_with_fixedip(self):
|
|
# specify only first and last network
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=self.nets1[0]['id'],
|
|
address='10.0.1.0')])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=1, requested_networks=requested_networks)
|
|
|
|
def test_allocate_for_instance_with_requested_networks_with_port(self):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=1, requested_networks=requested_networks)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_no_networks(self, mock_get_client):
|
|
"""verify the exception thrown when there are no networks defined."""
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': model.NetworkInfo([])}
|
|
nwinfo = self.api.allocate_for_instance(self.context, self.instance,
|
|
None)
|
|
self.assertEqual(0, len(nwinfo))
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)])
|
|
mocked_client.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=self.instance.project_id, shared=False),
|
|
mock.call(shared=True)])
|
|
self.assertEqual(2, mocked_client.list_networks.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.network.neutron.API._populate_neutron_extension_values')
|
|
@mock.patch('nova.network.neutron.API._create_ports_for_instance')
|
|
@mock.patch('nova.network.neutron.API._unbind_ports')
|
|
def test_allocate_for_instance_ex1(self, mock_unbind, mock_create_ports,
|
|
mock_populate, mock_get_client):
|
|
"""Verify we will delete created ports if we fail to allocate all net
|
|
resources.
|
|
|
|
We mock to raise an exception when creating a second port. In this
|
|
case, the code should delete the first created port.
|
|
"""
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=net['id'])
|
|
for net in (self.nets2[0], self.nets2[1])])
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mock_create_ports.return_value = [
|
|
(request, (getattr(uuids, 'portid_%s' % request.network_id)))
|
|
for request in requested_networks
|
|
]
|
|
index = 0
|
|
update_port_values = []
|
|
expected_update_port_calls = []
|
|
for network in self.nets2:
|
|
binding_port_req_body = {
|
|
'port': {
|
|
'device_id': self.instance.uuid,
|
|
'device_owner': 'compute:nova',
|
|
},
|
|
}
|
|
port_req_body = {
|
|
'port': {
|
|
'network_id': network['id'],
|
|
'admin_state_up': True,
|
|
'tenant_id': self.instance.project_id,
|
|
},
|
|
}
|
|
port_req_body['port'].update(binding_port_req_body['port'])
|
|
port_id = getattr(uuids, 'portid_%s' % network['id'])
|
|
port = {'id': port_id, 'mac_address': 'foo'}
|
|
|
|
if index == 0:
|
|
update_port_values.append({'port': port})
|
|
else:
|
|
update_port_values.append(exceptions.MacAddressInUseClient())
|
|
expected_update_port_calls.append(mock.call(
|
|
port_id, binding_port_req_body))
|
|
index += 1
|
|
mocked_client.update_port.side_effect = update_port_values
|
|
|
|
self.assertRaises(exception.PortInUse,
|
|
self.api.allocate_for_instance,
|
|
self.context, self.instance,
|
|
requested_networks=requested_networks)
|
|
mock_unbind.assert_called_once_with(self.context, [],
|
|
mocked_client, mock.ANY)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)], any_order=True)
|
|
mocked_client.list_networks.assert_called_once_with(
|
|
id=[uuids.my_netid1, uuids.my_netid2])
|
|
mocked_client.update_port.assert_has_calls(expected_update_port_calls)
|
|
self.assertEqual(len(expected_update_port_calls),
|
|
mocked_client.update_port.call_count)
|
|
mocked_client.delete_port.assert_has_calls([
|
|
mock.call(getattr(uuids, 'portid_%s' % self.nets2[0]['id'])),
|
|
mock.call(getattr(uuids, 'portid_%s' % self.nets2[1]['id']))])
|
|
self.assertEqual(2, mocked_client.delete_port.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_ex2(self, mock_get_client):
|
|
"""verify we have no port to delete
|
|
if we fail to allocate the first net resource.
|
|
|
|
Mock to raise exception when creating the first port.
|
|
In this case, the code should not delete any ports.
|
|
"""
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=net['id'])
|
|
for net in (self.nets2[0], self.nets2[1])])
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
port_req_body = {
|
|
'port': {
|
|
'network_id': self.nets2[0]['id'],
|
|
'admin_state_up': True,
|
|
'device_id': self.instance.uuid,
|
|
'tenant_id': self.instance.project_id,
|
|
},
|
|
}
|
|
mocked_client.create_port.side_effect = Exception(
|
|
"fail to create port")
|
|
self.assertRaises(NEUTRON_CLIENT_EXCEPTION,
|
|
self.api.allocate_for_instance,
|
|
self.context, self.instance,
|
|
requested_networks=requested_networks)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)])
|
|
mocked_client.list_networks.assert_called_once_with(
|
|
id=[uuids.my_netid1, uuids.my_netid2])
|
|
mocked_client.create_port.assert_called_once_with(port_req_body)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_available_networks')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_no_port_or_network(
|
|
self, mock_get_client, mock_get_available):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
# Make sure we get an empty list and then bail out of the rest
|
|
# of the function
|
|
mock_get_available.side_effect = test.TestingException
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest()])
|
|
self.assertRaises(test.TestingException,
|
|
self.api.allocate_for_instance, self.context,
|
|
self.instance, requested_networks)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)])
|
|
mock_get_available.assert_called_once_with(
|
|
self.context, self.instance.project_id, [],
|
|
neutron=mocked_client, auto_allocate=False)
|
|
|
|
def test_allocate_for_instance_second_time(self):
|
|
# Make sure that allocate_for_instance only returns ports that it
|
|
# allocated during _that_ run.
|
|
new_port = {'id': uuids.fake}
|
|
self._returned_nw_info = self.port_data1 + [new_port]
|
|
nw_info, _ = self._test_allocate_for_instance_with_virtual_interface()
|
|
self.assertEqual([new_port], nw_info)
|
|
|
|
def test_allocate_for_instance_port_in_use(self):
|
|
# If a port is already in use, an exception should be raised.
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance(
|
|
requested_networks=requested_networks,
|
|
exception=exception.PortInUse, _break='pre_list_networks',
|
|
_device=True)
|
|
|
|
def test_allocate_for_instance_port_not_found(self):
|
|
# If a port is not found, an exception should be raised.
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.non_existent_uuid)])
|
|
self._test_allocate_for_instance(
|
|
requested_networks=requested_networks,
|
|
exception=exception.PortNotFound,
|
|
_break='pre_list_networks')
|
|
|
|
def test_allocate_for_instance_port_invalid_tenantid(self):
|
|
self.tenant_id = 'invalid_id'
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance(
|
|
requested_networks=requested_networks,
|
|
exception=exception.PortNotUsable,
|
|
_break='pre_list_networks')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_with_externalnet_forbidden(
|
|
self, mock_get_client):
|
|
"""Only one network is available, it's external, and the client
|
|
is unauthorized to use it.
|
|
"""
|
|
rules = {'network:attach_external_network': 'is_admin:True'}
|
|
policy.set_rules(oslo_policy.Rules.from_dict(rules))
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
mocked_client.list_networks.side_effect = [
|
|
# no networks in the tenant
|
|
{'networks': model.NetworkInfo([])},
|
|
# external network is shared
|
|
{'networks': self.nets8}]
|
|
self.assertRaises(exception.ExternalNetworkAttachForbidden,
|
|
self.api.allocate_for_instance, self.context,
|
|
self.instance, None)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)])
|
|
mocked_client.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=self.instance.project_id, shared=False),
|
|
mock.call(shared=True)])
|
|
self.assertEqual(2, mocked_client.list_networks.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_with_externalnet_multiple(
|
|
self, mock_get_client):
|
|
"""Multiple networks are available, one the client is authorized
|
|
to use, and an external one the client is unauthorized to use.
|
|
"""
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
mocked_client.list_networks.side_effect = [
|
|
# network found in the tenant
|
|
{'networks': self.nets1},
|
|
# external network is shared
|
|
{'networks': self.nets8}]
|
|
self.assertRaises(
|
|
exception.NetworkAmbiguous,
|
|
self.api.allocate_for_instance,
|
|
self.context, self.instance, None)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context),
|
|
mock.call(self.context, admin=True)])
|
|
mocked_client.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=self.instance.project_id, shared=False),
|
|
mock.call(shared=True)])
|
|
self.assertEqual(2, mocked_client.list_networks.call_count)
|
|
|
|
def test_allocate_for_instance_with_externalnet_admin_ctx(self):
|
|
"""Only one network is available, it's external, and the client
|
|
is authorized.
|
|
"""
|
|
admin_ctx = context.RequestContext('userid', uuids.my_tenant,
|
|
is_admin=True)
|
|
self._test_allocate_for_instance(net_idx=8, context=admin_ctx)
|
|
|
|
def test_allocate_for_instance_with_external_shared_net(self):
|
|
"""Only one network is available, it's external and shared."""
|
|
ctx = context.RequestContext('userid', uuids.my_tenant)
|
|
self._test_allocate_for_instance(net_idx=10, context=ctx)
|
|
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_deallocate_for_instance(self, number, mock_get_client,
|
|
mock_cache_update,
|
|
requested_networks=None):
|
|
# TODO(mriedem): Remove this conversion when all neutronv2 APIs are
|
|
# converted to handling instance objects.
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_data = number == 1 and self.port_data1 or self.port_data2
|
|
ports = {port['id'] for port in port_data}
|
|
ret_data = copy.deepcopy(port_data)
|
|
if requested_networks:
|
|
if isinstance(requested_networks, objects.NetworkRequestList):
|
|
requested_networks = requested_networks.as_tuples()
|
|
for net, fip, port, request_id, _, _ in requested_networks:
|
|
ret_data.append({'network_id': net,
|
|
'device_id': self.instance.uuid,
|
|
'device_owner': 'compute:nova',
|
|
'id': port,
|
|
'status': 'DOWN',
|
|
'admin_state_up': True,
|
|
'fixed_ips': [],
|
|
'mac_address': 'fake_mac', })
|
|
mocked_client.list_ports.return_value = {'ports': ret_data}
|
|
|
|
show_port_values = []
|
|
expected_show_port_calls = []
|
|
show_network_values = []
|
|
expected_show_network_calls = []
|
|
expected_update_port_calls = []
|
|
if requested_networks:
|
|
for net, fip, port, request_id, _, _ in requested_networks:
|
|
expected_show_port_calls.append(mock.call(
|
|
port, fields=['binding:profile', 'network_id']))
|
|
show_port_values.append({'port': ret_data[0]})
|
|
expected_show_network_calls.append(mock.call(
|
|
ret_data[0]['network_id'], fields=['dns_domain']))
|
|
show_network_values.append(
|
|
{'network': {'id': ret_data[0]['network_id']}})
|
|
expected_update_port_calls.append(mock.call(
|
|
port,
|
|
{'port': {
|
|
'device_owner': '',
|
|
'device_id': '',
|
|
'binding:host_id': None,
|
|
'binding:profile': {}}}))
|
|
mocked_client.show_port.side_effect = show_port_values
|
|
mocked_client.show_network.side_effect = show_network_values
|
|
|
|
expected_delete_port_calls = []
|
|
for port in ports:
|
|
expected_delete_port_calls.append(mock.call(port))
|
|
|
|
self.api.deallocate_for_instance(self.context, self.instance,
|
|
requested_networks=requested_networks)
|
|
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context), mock.call(self.context, admin=True)],
|
|
any_order=True)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
device_id=self.instance.uuid)
|
|
mocked_client.show_port.assert_has_calls(expected_show_port_calls)
|
|
self.assertEqual(len(expected_show_port_calls),
|
|
mocked_client.show_port.call_count)
|
|
mocked_client.show_network.assert_has_calls(
|
|
expected_show_network_calls)
|
|
self.assertEqual(len(expected_show_network_calls),
|
|
mocked_client.show_network.call_count)
|
|
mocked_client.update_port.assert_has_calls(expected_update_port_calls)
|
|
self.assertEqual(len(expected_update_port_calls),
|
|
mocked_client.update_port.call_count)
|
|
mocked_client.delete_port.assert_has_calls(expected_delete_port_calls,
|
|
any_order=True)
|
|
self.assertEqual(len(expected_delete_port_calls),
|
|
mocked_client.delete_port.call_count)
|
|
mock_cache_update.assert_called_once_with(
|
|
self.context, self.instance.uuid, {'network_info': '[]'})
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_for_instance_1_with_requested(self, mock_preexisting):
|
|
mock_preexisting.return_value = []
|
|
requested = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id='fake-net',
|
|
address='1.2.3.4',
|
|
port_id=uuids.portid_5)])
|
|
# Test to deallocate in one port env.
|
|
self._test_deallocate_for_instance(1, requested_networks=requested)
|
|
mock_preexisting.assert_called_once_with(self.instance)
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_for_instance_2_with_requested(self, mock_preexisting):
|
|
mock_preexisting.return_value = []
|
|
requested = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id='fake-net',
|
|
address='1.2.3.4',
|
|
port_id=uuids.portid_6)])
|
|
# Test to deallocate in one port env.
|
|
self._test_deallocate_for_instance(2, requested_networks=requested)
|
|
mock_preexisting.assert_called_once_with(self.instance)
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_for_instance_1(self, mock_preexisting):
|
|
mock_preexisting.return_value = []
|
|
# Test to deallocate in one port env.
|
|
self._test_deallocate_for_instance(1)
|
|
mock_preexisting.assert_called_once_with(self.instance)
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_for_instance_2(self, mock_preexisting):
|
|
mock_preexisting.return_value = []
|
|
# Test to deallocate in two ports env.
|
|
self._test_deallocate_for_instance(2)
|
|
mock_preexisting.assert_called_once_with(self.instance)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_for_instance_port_not_found(self,
|
|
mock_preexisting,
|
|
mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
# TODO(mriedem): Remove this conversion when all neutronv2 APIs are
|
|
# converted to handling instance objects.
|
|
self.instance = fake_instance.fake_instance_obj(self.context,
|
|
**self.instance)
|
|
mock_preexisting.return_value = []
|
|
port_data = self.port_data1
|
|
mocked_client.list_ports.return_value = {'ports': port_data}
|
|
|
|
NeutronNotFound = exceptions.NeutronClientException(status_code=404)
|
|
delete_port_values = []
|
|
expected_delete_port_calls = []
|
|
for port in reversed(port_data):
|
|
delete_port_values.append(NeutronNotFound)
|
|
expected_delete_port_calls.append(mock.call(port['id']))
|
|
mocked_client.delete_port.side_effect = expected_delete_port_calls
|
|
|
|
self.api.deallocate_for_instance(self.context, self.instance)
|
|
|
|
mock_preexisting.assert_called_once_with(self.instance)
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context), mock.call(self.context, admin=True)],
|
|
any_order=True)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
device_id=self.instance.uuid)
|
|
mocked_client.delete_port.assert_has_calls(expected_delete_port_calls)
|
|
self.assertEqual(len(expected_delete_port_calls),
|
|
mocked_client.delete_port.call_count)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(db_api, 'instance_info_cache_get')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_deallocate_port_for_instance(self, number, mock_get_client,
|
|
mock_cache_get, mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
|
|
port_data = number == 1 and self.port_data1 or self.port_data2
|
|
nets = number == 1 and self.nets1 or self.nets2
|
|
mocked_client.show_port.return_value = {'port': port_data[0]}
|
|
|
|
net_info_cache = []
|
|
for port in port_data:
|
|
net_info_cache.append({"network": {"id": port['network_id']},
|
|
"id": port['id']})
|
|
self.instance['info_cache'] = self._fake_instance_info_cache(
|
|
net_info_cache, self.instance['uuid'])
|
|
mocked_client.list_ports.return_value = {'ports': port_data[1:]}
|
|
net_ids = [port['network_id'] for port in port_data]
|
|
mocked_client.list_networks.return_value = {'networks': nets}
|
|
|
|
list_floatingips_values = []
|
|
expected_list_floatingips_calls = []
|
|
float_data = number == 1 and self.float_data1 or self.float_data2
|
|
for data in port_data[1:]:
|
|
for ip in data['fixed_ips']:
|
|
list_floatingips_values.append({'floatingips': float_data[1:]})
|
|
expected_list_floatingips_calls.append(
|
|
mock.call(fixed_ip_address=ip['ip_address'],
|
|
port_id=data['id']))
|
|
mocked_client.list_floatingips.side_effect = list_floatingips_values
|
|
|
|
mocked_client.list_subnets.return_value = {}
|
|
expected_list_subnets_calls = []
|
|
for port in port_data[1:]:
|
|
expected_list_subnets_calls.append(mock.call(id=['my_subid2']))
|
|
|
|
mock_cache_get.return_value = self.instance['info_cache']
|
|
|
|
instance = self._fake_instance_object_with_info_cache(self.instance)
|
|
nwinfo, port_allocation = self.api.deallocate_port_for_instance(
|
|
self.context, instance, port_data[0]['id'])
|
|
self.assertEqual(len(port_data[1:]), len(nwinfo))
|
|
if len(port_data) > 1:
|
|
self.assertEqual(uuids.my_netid2, nwinfo[0]['network']['id'])
|
|
|
|
mocked_client.delete_port.assert_called_once_with(port_data[0]['id'])
|
|
mocked_client.show_port.assert_called_once_with(port_data[0]['id'])
|
|
expected_get_client_calls = [
|
|
mock.call(self.context, admin=True),
|
|
mock.call(self.context, admin=True),
|
|
]
|
|
if number == 2:
|
|
expected_get_client_calls.append(mock.call(self.context,
|
|
admin=True))
|
|
mock_get_client.assert_has_calls(expected_get_client_calls,
|
|
any_order=True)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=self.instance['project_id'],
|
|
device_id=self.instance['uuid'])
|
|
mocked_client.list_networks.assert_called_once_with(id=net_ids)
|
|
mocked_client.list_floatingips.assert_has_calls(
|
|
expected_list_floatingips_calls)
|
|
self.assertEqual(len(expected_list_floatingips_calls),
|
|
mocked_client.list_floatingips.call_count)
|
|
mock_cache_get.assert_called_once_with(mock.ANY, self.instance['uuid'])
|
|
mocked_client.list_subnets.assert_has_calls(
|
|
expected_list_subnets_calls)
|
|
self.assertEqual(len(expected_list_subnets_calls),
|
|
mocked_client.list_subnets.call_count)
|
|
if number == 2:
|
|
mock_get_physnet.assert_called_once_with(mock.ANY, mock.ANY,
|
|
mock.ANY)
|
|
else:
|
|
mock_get_physnet.assert_not_called()
|
|
|
|
def test_deallocate_port_for_instance_1(self):
|
|
# Test to deallocate the first and only port
|
|
self._test_deallocate_port_for_instance(1)
|
|
|
|
def test_deallocate_port_for_instance_2(self):
|
|
# Test to deallocate the first port of two
|
|
self._test_deallocate_port_for_instance(2)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_list_ports(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
search_opts = {'parm': 'value'}
|
|
|
|
self.api.list_ports(self.context, **search_opts)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(**search_opts)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_show_port(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
mocked_client.show_port.return_value = {'port': self.port_data1[0]}
|
|
|
|
self.api.show_port(self.context, 'foo')
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with('foo')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = [
|
|
(uuids.my_netid1, None, None, None, None, None),
|
|
(uuids.my_netid2, None, None, None, None, None)]
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': 50}}
|
|
mocked_client.list_ports.return_value = {'ports': []}
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.my_tenant, fields=['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_without_port_quota_on_network_side(
|
|
self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = [
|
|
(uuids.my_netid1, None, None, None, None, None),
|
|
(uuids.my_netid2, None, None, None, None, None)]
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mocked_client.show_quota.return_value = {'quota': {}}
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_ex_1(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = [
|
|
(uuids.my_netid1, None, None, None, None, None)]
|
|
mocked_client.list_networks.return_value = {'networks': []}
|
|
|
|
ex = self.assertRaises(exception.NetworkNotFound,
|
|
self.api.validate_networks, self.context,
|
|
requested_networks, 1)
|
|
|
|
self.assertIn(uuids.my_netid1, str(ex))
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(
|
|
id=[uuids.my_netid1])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_ex_2(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = [
|
|
(uuids.my_netid1, None, None, None, None, None),
|
|
(uuids.my_netid2, None, None, None, None, None),
|
|
(uuids.my_netid3, None, None, None, None, None)]
|
|
ids = [uuids.my_netid1, uuids.my_netid2, uuids.my_netid3]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets1}
|
|
|
|
ex = self.assertRaises(exception.NetworkNotFound,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
|
|
self.assertIn(uuids.my_netid2, str(ex))
|
|
self.assertIn(uuids.my_netid3, str(ex))
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_duplicate_enable(self, mock_get_client):
|
|
# Verify that no duplicateNetworks exception is thrown when duplicate
|
|
# network ids are passed to validate_networks.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid1),
|
|
objects.NetworkRequest(network_id=uuids.my_netid1)])
|
|
ids = [uuids.my_netid1, uuids.my_netid1]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets1}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': 50}}
|
|
mocked_client.list_ports.return_value = {'ports': []}
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.my_tenant, fields=['id'])
|
|
|
|
def test_allocate_for_instance_with_requested_networks_duplicates(self):
|
|
# specify a duplicate network to allocate to instance
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=net['id'])
|
|
for net in (self.nets6[0], self.nets6[1])])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=6, requested_networks=requested_networks)
|
|
|
|
def test_allocate_for_instance_requested_networks_duplicates_port(self):
|
|
# specify first port and last port that are in same network
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port['id'])
|
|
for port in (self.port_data1[0], self.port_data3[0])])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=6, requested_networks=requested_networks)
|
|
|
|
def test_allocate_for_instance_requested_networks_duplicates_combo(self):
|
|
# specify a combo net_idx=7 : net2, port in net1, net2, port in net1
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid2),
|
|
objects.NetworkRequest(port_id=self.port_data1[0]['id']),
|
|
objects.NetworkRequest(network_id=uuids.my_netid2),
|
|
objects.NetworkRequest(port_id=self.port_data3[0]['id'])])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=7, requested_networks=requested_networks)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_not_specified(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(objects=[])
|
|
mocked_client.list_networks.side_effect = [
|
|
{'networks': self.nets1}, {'networks': self.nets2}]
|
|
self.assertRaises(exception.NetworkAmbiguous,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=self.context.project_id, shared=False),
|
|
mock.call(shared=True)])
|
|
self.assertEqual(2, mocked_client.list_networks.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_port_not_found(self, mock_get_client):
|
|
# Verify that the correct exception is thrown when a non existent
|
|
# port is passed to validate_networks.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(
|
|
network_id=uuids.my_netid1,
|
|
port_id=uuids.portid_1)])
|
|
|
|
mocked_client.show_port.side_effect = exceptions.PortNotFoundClient
|
|
self.assertRaises(exception.PortNotFound,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(
|
|
requested_networks[0].port_id)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_port_show_raises_non404(self, mock_get_client):
|
|
# Verify that the correct exception is thrown when a non existent
|
|
# port is passed to validate_networks.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port_id = uuids.portid_1
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(
|
|
network_id=uuids.my_netid1,
|
|
port_id=fake_port_id)])
|
|
mocked_client.show_port.side_effect = (
|
|
exceptions.NeutronClientException(status_code=0))
|
|
|
|
exc = self.assertRaises(exception.NovaException,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
expected_exception_message = ('Failed to access port %(port_id)s: '
|
|
'An unknown exception occurred.' %
|
|
{'port_id': fake_port_id})
|
|
self.assertEqual(expected_exception_message, str(exc))
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(
|
|
requested_networks[0].port_id)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_port_in_use(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=self.port_data3[0]['id'])])
|
|
mocked_client.show_port.return_value = {'port': self.port_data3[0]}
|
|
|
|
self.assertRaises(exception.PortInUse,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(
|
|
self.port_data3[0]['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_port_no_subnet_id(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_a = self.port_data3[0]
|
|
port_a['device_id'] = None
|
|
port_a['device_owner'] = None
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port_a['id'])])
|
|
mocked_client.show_port.return_value = {'port': port_a}
|
|
|
|
self.assertRaises(exception.PortRequiresFixedIP,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(port_a['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_no_subnet_id(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id='his_netid4')])
|
|
ids = ['his_netid4']
|
|
mocked_client.list_networks.return_value = {'networks': self.nets4}
|
|
self.assertRaises(exception.NetworkRequiresSubnet,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_ports_in_same_network_enable(self,
|
|
mock_get_client):
|
|
# Verify that duplicateNetworks exception is not thrown when ports
|
|
# on same duplicate network are passed to validate_networks.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_a = self.port_data3[0]
|
|
port_a['fixed_ips'] = {'ip_address': '10.0.0.2',
|
|
'subnet_id': 'subnet_id'}
|
|
port_b = self.port_data1[0]
|
|
self.assertEqual(port_a['network_id'], port_b['network_id'])
|
|
for port in [port_a, port_b]:
|
|
port['device_id'] = None
|
|
port['device_owner'] = None
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port_a['id']),
|
|
objects.NetworkRequest(port_id=port_b['id'])])
|
|
mocked_client.show_port.side_effect = [{'port': port_a},
|
|
{'port': port_b}]
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_has_calls([mock.call(port_a['id']),
|
|
mock.call(port_b['id'])])
|
|
self.assertEqual(2, mocked_client.show_port.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_ports_not_in_same_network(self,
|
|
mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_a = self.port_data3[0]
|
|
port_a['fixed_ips'] = {'ip_address': '10.0.0.2',
|
|
'subnet_id': 'subnet_id'}
|
|
port_b = self.port_data2[1]
|
|
self.assertNotEqual(port_a['network_id'], port_b['network_id'])
|
|
for port in [port_a, port_b]:
|
|
port['device_id'] = None
|
|
port['device_owner'] = None
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port_a['id']),
|
|
objects.NetworkRequest(port_id=port_b['id'])])
|
|
mocked_client.show_port.side_effect = [{'port': port_a},
|
|
{'port': port_b}]
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_has_calls([mock.call(port_a['id']),
|
|
mock.call(port_b['id'])])
|
|
self.assertEqual(2, mocked_client.show_port.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_no_quota(self, mock_get_client):
|
|
# Test validation for a request for one instance needing
|
|
# two ports, where the quota is 2 and 2 ports are in use
|
|
# => instances which can be created = 0
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid1),
|
|
objects.NetworkRequest(network_id=uuids.my_netid2)])
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': 2}}
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data2}
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 1)
|
|
self.assertEqual(0, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.my_tenant, fields=['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_with_ports_and_networks(self, mock_get_client):
|
|
# Test validation for a request for one instance needing
|
|
# one port allocated via nova with another port being passed in.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_b = self.port_data2[1]
|
|
port_b['device_id'] = None
|
|
port_b['device_owner'] = None
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid1),
|
|
objects.NetworkRequest(port_id=port_b['id'])])
|
|
mocked_client.show_port.return_value = {'port': port_b}
|
|
ids = [uuids.my_netid1]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets1}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': 5}}
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data2}
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 1)
|
|
|
|
self.assertEqual(1, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(port_b['id'])
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.my_tenant, fields=['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_one_port_and_no_networks(self, mock_get_client):
|
|
# Test that show quota is not called if no networks are
|
|
# passed in and only ports.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_b = self.port_data2[1]
|
|
port_b['device_id'] = None
|
|
port_b['device_owner'] = None
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port_b['id'])])
|
|
mocked_client.show_port.return_value = {'port': port_b}
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 1)
|
|
|
|
self.assertEqual(1, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_called_once_with(port_b['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_some_quota(self, mock_get_client):
|
|
# Test validation for a request for two instance needing
|
|
# two ports each, where the quota is 5 and 2 ports are in use
|
|
# => instances which can be created = 1
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid1),
|
|
objects.NetworkRequest(network_id=uuids.my_netid2)])
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': 5}}
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data2}
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 2)
|
|
|
|
self.assertEqual(1, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.my_tenant, fields=['id'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_unlimited_quota(self, mock_get_client):
|
|
# Test validation for a request for two instance needing
|
|
# two ports each, where the quota is -1 (unlimited)
|
|
# => instances which can be created = 1
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(network_id=uuids.my_netid1),
|
|
objects.NetworkRequest(network_id=uuids.my_netid2)])
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
mocked_client.list_networks.return_value = {'networks': self.nets2}
|
|
mocked_client.show_quota.return_value = {'quota': {'port': -1}}
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 2)
|
|
|
|
self.assertEqual(2, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(id=ids)
|
|
mocked_client.show_quota.assert_called_once_with(uuids.my_tenant)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_validate_networks_no_quota_but_ports_supplied(self,
|
|
mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_a = self.port_data3[0]
|
|
port_a['fixed_ips'] = {'ip_address': '10.0.0.2',
|
|
'subnet_id': 'subnet_id'}
|
|
port_b = self.port_data2[1]
|
|
self.assertNotEqual(port_a['network_id'], port_b['network_id'])
|
|
for port in [port_a, port_b]:
|
|
port['device_id'] = None
|
|
port['device_owner'] = None
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port_a['id']),
|
|
objects.NetworkRequest(port_id=port_b['id'])])
|
|
mocked_client.show_port.side_effect = [{'port': port_a},
|
|
{'port': port_b}]
|
|
|
|
max_count = self.api.validate_networks(self.context,
|
|
requested_networks, 1)
|
|
|
|
self.assertEqual(1, max_count)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_port.assert_has_calls([mock.call(port_a['id']),
|
|
mock.call(port_b['id'])])
|
|
self.assertEqual(2, mocked_client.show_port.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_fixed_ip_by_address_with_exception(self, mock_get_client,
|
|
port_data=None,
|
|
exception=None):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
if port_data is None:
|
|
port_data = self.port_data2
|
|
mocked_client.list_ports.return_value = {'ports': port_data}
|
|
|
|
result = None
|
|
if exception:
|
|
self.assertRaises(exception, self.api.get_fixed_ip_by_address,
|
|
self.context, self.port_address)
|
|
else:
|
|
result = self.api.get_fixed_ip_by_address(self.context,
|
|
self.port_address)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
fixed_ips='ip_address=%s' % self.port_address)
|
|
|
|
return result
|
|
|
|
def test_get_fixed_ip_by_address_fails_for_no_ports(self):
|
|
self._test_get_fixed_ip_by_address_with_exception(
|
|
port_data=[], exception=exception.FixedIpNotFoundForAddress)
|
|
|
|
def test_get_fixed_ip_by_address_succeeds_for_1_port(self):
|
|
result = self._test_get_fixed_ip_by_address_with_exception(
|
|
port_data=self.port_data1)
|
|
self.assertEqual(self.instance2['uuid'], result['instance_uuid'])
|
|
|
|
def test_get_fixed_ip_by_address_fails_for_more_than_1_port(self):
|
|
self._test_get_fixed_ip_by_address_with_exception(
|
|
exception=exception.FixedIpAssociatedWithMultipleInstances)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_available_networks(self, prv_nets, pub_nets, mock_get_client,
|
|
req_ids=None, context=None):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
nets = prv_nets + pub_nets
|
|
if req_ids:
|
|
mocked_client.list_networks.return_value = {'networks': nets}
|
|
else:
|
|
mocked_client.list_networks.side_effect = [{'networks': prv_nets},
|
|
{'networks': pub_nets}]
|
|
|
|
rets = self.api._get_available_networks(
|
|
context if context else self.context,
|
|
self.instance['project_id'],
|
|
req_ids)
|
|
|
|
self.assertEqual(nets, rets)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
if req_ids:
|
|
mocked_client.list_networks.assert_called_once_with(id=req_ids)
|
|
else:
|
|
mocked_client.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=self.instance['project_id'], shared=False),
|
|
mock.call(shared=True)])
|
|
self.assertEqual(2, mocked_client.list_networks.call_count)
|
|
|
|
def test_get_available_networks_all_private(self):
|
|
self._test_get_available_networks(self.nets2, [])
|
|
|
|
def test_get_available_networks_all_public(self):
|
|
self._test_get_available_networks([], self.nets2)
|
|
|
|
def test_get_available_networks_private_and_public(self):
|
|
self._test_get_available_networks(self.nets1, self.nets4)
|
|
|
|
def test_get_available_networks_with_network_ids(self):
|
|
prv_nets = [self.nets3[0]]
|
|
pub_nets = [self.nets3[-1]]
|
|
# specify only first and last network
|
|
req_ids = [net['id'] for net in (self.nets3[0], self.nets3[-1])]
|
|
self._test_get_available_networks(prv_nets, pub_nets, req_ids=req_ids)
|
|
|
|
def test_get_available_networks_with_custom_policy(self):
|
|
rules = {'network:attach_external_network': ''}
|
|
policy.set_rules(oslo_policy.Rules.from_dict(rules))
|
|
req_ids = [net['id'] for net in self.nets5]
|
|
self._test_get_available_networks(self.nets5, [],
|
|
req_ids=req_ids)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_floating_ip_pools(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
search_opts = {'router:external': True}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool, self.fip_pool_nova]}
|
|
|
|
pools = self.api.get_floating_ip_pools(self.context)
|
|
|
|
expected = [self.fip_pool, self.fip_pool_nova]
|
|
self.assertEqual(expected, pools)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_floating_ip_by_address_not_found(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_unassociated['floating_ip_address']
|
|
mocked_client.list_floatingips.return_value = {'floatingips': []}
|
|
self.assertRaises(exception.FloatingIpNotFoundForAddress,
|
|
self.api.get_floating_ip_by_address,
|
|
self.context, address)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_floating_ip_by_id_not_found(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
floating_ip_id = self.fip_unassociated['id']
|
|
mocked_client.show_floatingip.side_effect = (
|
|
exceptions.NeutronClientException(status_code=404))
|
|
self.assertRaises(exception.FloatingIpNotFound,
|
|
self.api.get_floating_ip,
|
|
self.context, floating_ip_id)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_floatingip.assert_called_once_with(floating_ip_id)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_floating_ip_raises_non404(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
floating_ip_id = self.fip_unassociated['id']
|
|
mocked_client.show_floatingip.side_effect = (
|
|
exceptions.NeutronClientException(status_code=0))
|
|
self.assertRaises(exceptions.NeutronClientException,
|
|
self.api.get_floating_ip,
|
|
self.context, floating_ip_id)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.show_floatingip.assert_called_once_with(floating_ip_id)
|
|
|
|
@mock.patch.object(neutronapi.API, '_refresh_neutron_extensions_cache')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_floating_ip(
|
|
self, fip_ext_enabled, has_port, mock_ntrn, mock_refresh):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
# NOTE(stephenfin): These are clearly not full responses
|
|
mock_nc.show_floatingip.return_value = {
|
|
'floatingip': {
|
|
'id': uuids.fip_id,
|
|
'floating_network_id': uuids.fip_net_id,
|
|
'port_id': uuids.fip_port_id,
|
|
}
|
|
}
|
|
mock_nc.show_network.return_value = {
|
|
'network': {
|
|
'id': uuids.fip_net_id,
|
|
},
|
|
}
|
|
if has_port:
|
|
mock_nc.show_port.return_value = {
|
|
'port': {
|
|
'id': uuids.fip_port_id,
|
|
},
|
|
}
|
|
else:
|
|
mock_nc.show_port.side_effect = exceptions.PortNotFoundClient
|
|
|
|
if fip_ext_enabled:
|
|
self.api.extensions = {
|
|
constants.FIP_PORT_DETAILS: {
|
|
'alias': constants.FIP_PORT_DETAILS,
|
|
},
|
|
}
|
|
else:
|
|
self.api.extensions = {}
|
|
|
|
fip = self.api.get_floating_ip(self.context, uuids.fip_id)
|
|
|
|
if fip_ext_enabled:
|
|
mock_nc.show_port.assert_not_called()
|
|
self.assertNotIn('port_details', fip)
|
|
else:
|
|
mock_nc.show_port.assert_called_once_with(uuids.fip_port_id)
|
|
self.assertIn('port_details', fip)
|
|
|
|
if has_port:
|
|
self.assertIsNotNone(fip['port_details'])
|
|
else:
|
|
self.assertIsNone(fip['port_details'])
|
|
|
|
def test_get_floating_ip_with_fip_port_details_ext(self):
|
|
"""Make sure we used embedded port details if available."""
|
|
self._test_get_floating_ip(True, True)
|
|
|
|
def test_get_floating_ip_without_fip_port_details_ext(self):
|
|
"""Make sure we make a second request for port details if necessary."""
|
|
self._test_get_floating_ip(False, True)
|
|
|
|
def test_get_floating_ip_without_port(self):
|
|
"""Make sure we don't fail for floating IPs without attached ports."""
|
|
self._test_get_floating_ip(False, False)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_floating_ip_by_address_multiple_found(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_unassociated['floating_ip_address']
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_unassociated] * 2}
|
|
self.assertRaises(exception.FloatingIpMultipleFoundForAddress,
|
|
self.api.get_floating_ip_by_address,
|
|
self.context, address)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
|
|
@mock.patch.object(neutronapi.API, '_refresh_neutron_extensions_cache')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_floating_ip_by_address(
|
|
self, fip_ext_enabled, has_port, mock_ntrn, mock_refresh):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
# NOTE(stephenfin): These are clearly not full responses
|
|
mock_nc.list_floatingips.return_value = {
|
|
'floatingips': [
|
|
{
|
|
'id': uuids.fip_id,
|
|
'floating_network_id': uuids.fip_net_id,
|
|
'port_id': uuids.fip_port_id,
|
|
},
|
|
]
|
|
}
|
|
mock_nc.show_network.return_value = {
|
|
'network': {
|
|
'id': uuids.fip_net_id,
|
|
},
|
|
}
|
|
if has_port:
|
|
mock_nc.show_port.return_value = {
|
|
'port': {
|
|
'id': uuids.fip_port_id,
|
|
},
|
|
}
|
|
else:
|
|
mock_nc.show_port.side_effect = exceptions.PortNotFoundClient
|
|
|
|
if fip_ext_enabled:
|
|
self.api.extensions = {
|
|
constants.FIP_PORT_DETAILS: {
|
|
'alias': constants.FIP_PORT_DETAILS,
|
|
},
|
|
}
|
|
else:
|
|
self.api.extensions = {}
|
|
|
|
fip = self.api.get_floating_ip_by_address(self.context, '172.1.2.3')
|
|
|
|
if fip_ext_enabled:
|
|
mock_nc.show_port.assert_not_called()
|
|
self.assertNotIn('port_details', fip)
|
|
else:
|
|
mock_nc.show_port.assert_called_once_with(uuids.fip_port_id)
|
|
self.assertIn('port_details', fip)
|
|
|
|
if has_port:
|
|
self.assertIsNotNone(fip['port_details'])
|
|
else:
|
|
self.assertIsNone(fip['port_details'])
|
|
|
|
def test_get_floating_ip_by_address_with_fip_port_details_ext(self):
|
|
"""Make sure we used embedded port details if available."""
|
|
self._test_get_floating_ip_by_address(True, True)
|
|
|
|
def test_get_floating_ip_by_address_without_fip_port_details_ext(self):
|
|
"""Make sure we make a second request for port details if necessary."""
|
|
self._test_get_floating_ip_by_address(False, True)
|
|
|
|
def test_get_floating_ip_by_address_without_port(self):
|
|
"""Make sure we don't fail for floating IPs without attached ports."""
|
|
self._test_get_floating_ip_by_address(False, False)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_instance_id_by_floating_address(self, fip_data,
|
|
mock_get_client,
|
|
associated=False):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = fip_data['floating_ip_address']
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [fip_data]}
|
|
if associated:
|
|
mocked_client.show_port.return_value = {'port': self.port_data2[1]}
|
|
expected = self.port_data2[1]['device_id']
|
|
else:
|
|
expected = None
|
|
|
|
fip = self.api.get_instance_id_by_floating_address(self.context,
|
|
address)
|
|
|
|
self.assertEqual(expected, fip)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
if associated:
|
|
mocked_client.show_port.assert_called_once_with(
|
|
fip_data['port_id'])
|
|
|
|
def test_get_instance_id_by_floating_address(self):
|
|
self._test_get_instance_id_by_floating_address(self.fip_unassociated)
|
|
|
|
def test_get_instance_id_by_floating_address_associated(self):
|
|
self._test_get_instance_id_by_floating_address(self.fip_associated,
|
|
associated=True)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_floating_ip(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
pool_name = self.fip_pool['name']
|
|
pool_id = self.fip_pool['id']
|
|
search_opts = {'router:external': True,
|
|
'fields': 'id',
|
|
'name': pool_name}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool]}
|
|
mocked_client.create_floatingip.return_value = {
|
|
'floatingip': self.fip_unassociated}
|
|
|
|
fip = self.api.allocate_floating_ip(self.context, 'ext_net')
|
|
|
|
self.assertEqual(self.fip_unassociated['floating_ip_address'], fip)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
mocked_client.create_floatingip.assert_called_once_with(
|
|
{'floatingip': {'floating_network_id': pool_id}})
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_floating_ip_addr_gen_fail(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
pool_name = self.fip_pool['name']
|
|
pool_id = self.fip_pool['id']
|
|
search_opts = {'router:external': True,
|
|
'fields': 'id',
|
|
'name': pool_name}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool]}
|
|
mocked_client.create_floatingip.side_effect = (
|
|
exceptions.IpAddressGenerationFailureClient)
|
|
|
|
self.assertRaises(exception.NoMoreFloatingIps,
|
|
self.api.allocate_floating_ip, self.context,
|
|
'ext_net')
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
mocked_client.create_floatingip.assert_called_once_with(
|
|
{'floatingip': {'floating_network_id': pool_id}})
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_floating_ip_exhausted_fail(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
pool_name = self.fip_pool['name']
|
|
pool_id = self.fip_pool['id']
|
|
search_opts = {'router:external': True,
|
|
'fields': 'id',
|
|
'name': pool_name}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool]}
|
|
mocked_client.create_floatingip.side_effect = (
|
|
exceptions.ExternalIpAddressExhaustedClient)
|
|
|
|
self.assertRaises(exception.NoMoreFloatingIps,
|
|
self.api.allocate_floating_ip, self.context,
|
|
'ext_net')
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
mocked_client.create_floatingip.assert_called_once_with(
|
|
{'floatingip': {'floating_network_id': pool_id}})
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_floating_ip_with_pool_id(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
pool_id = self.fip_pool['id']
|
|
search_opts = {'router:external': True,
|
|
'fields': 'id',
|
|
'id': pool_id}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool]}
|
|
mocked_client.create_floatingip.return_value = {
|
|
'floatingip': self.fip_unassociated}
|
|
|
|
fip = self.api.allocate_floating_ip(self.context, pool_id)
|
|
|
|
self.assertEqual(self.fip_unassociated['floating_ip_address'], fip)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
mocked_client.create_floatingip.assert_called_once_with(
|
|
{'floatingip': {'floating_network_id': pool_id}})
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_floating_ip_with_default_pool(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
pool_name = self.fip_pool_nova['name']
|
|
pool_id = self.fip_pool_nova['id']
|
|
search_opts = {'router:external': True,
|
|
'fields': 'id',
|
|
'name': pool_name}
|
|
mocked_client.list_networks.return_value = {
|
|
'networks': [self.fip_pool_nova]}
|
|
mocked_client.create_floatingip.return_value = {
|
|
'floatingip': self.fip_unassociated}
|
|
|
|
fip = self.api.allocate_floating_ip(self.context)
|
|
|
|
self.assertEqual(self.fip_unassociated['floating_ip_address'], fip)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_networks.assert_called_once_with(**search_opts)
|
|
mocked_client.create_floatingip.assert_called_once_with(
|
|
{'floatingip': {'floating_network_id': pool_id}})
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_disassociate_and_release_floating_ip(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_unassociated['floating_ip_address']
|
|
fip_id = self.fip_unassociated['id']
|
|
floating_ip = {'floating_ip_address': address}
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_unassociated]}
|
|
|
|
self.api.disassociate_and_release_floating_ip(self.context, None,
|
|
floating_ip)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
mocked_client.delete_floatingip.assert_called_once_with(fip_id)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_disassociate_and_release_floating_ip_with_instance(
|
|
self, mock_get_client, mock_cache_update, mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_unassociated['floating_ip_address']
|
|
fip_id = self.fip_unassociated['id']
|
|
floating_ip = {'floating_ip_address': address}
|
|
instance = self._fake_instance_object(self.instance)
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_unassociated]}
|
|
|
|
self.api.disassociate_and_release_floating_ip(self.context, instance,
|
|
floating_ip)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
mocked_client.delete_floatingip.assert_called_once_with(fip_id)
|
|
mock_cache_update.assert_called_once_with(mock.ANY, instance['uuid'],
|
|
mock.ANY)
|
|
mock_get_nw.assert_called_once_with(mock.ANY, instance)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_associate_floating_ip(self, mock_get_client, mock_cache_update,
|
|
mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_unassociated['floating_ip_address']
|
|
fixed_address = self.port_address2
|
|
fip_id = self.fip_unassociated['id']
|
|
instance = self._fake_instance_object(self.instance)
|
|
|
|
mocked_client.list_ports.return_value = {'ports': [self.port_data2[1]]}
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_unassociated]}
|
|
|
|
self.api.associate_floating_ip(self.context, instance,
|
|
address, fixed_address)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
**{'device_owner': 'compute:nova', 'device_id': instance.uuid})
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
mocked_client.update_floatingip.assert_called_once_with(
|
|
fip_id, {'floatingip': {'port_id': self.fip_associated['port_id'],
|
|
'fixed_ip_address': fixed_address}})
|
|
mock_cache_update.assert_called_once_with(mock.ANY, instance['uuid'],
|
|
mock.ANY)
|
|
mock_get_nw.assert_called_once_with(mock.ANY, instance)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.objects.Instance.get_by_uuid')
|
|
def test_reassociate_floating_ip(self, mock_get, mock_get_client,
|
|
mock_cache_update, mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
address = self.fip_associated['floating_ip_address']
|
|
new_fixed_address = self.port_address
|
|
fip_id = self.fip_associated['id']
|
|
|
|
mocked_client.list_ports.return_value = {'ports': [self.port_data2[0]]}
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_associated]}
|
|
mocked_client.show_port.return_value = {'port': self.port_data2[1]}
|
|
mock_get.return_value = fake_instance.fake_instance_obj(
|
|
self.context, **self.instance)
|
|
instance2 = self._fake_instance_object(self.instance2)
|
|
|
|
self.api.associate_floating_ip(self.context, instance2,
|
|
address, new_fixed_address)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
**{'device_owner': 'compute:nova',
|
|
'device_id': self.instance2['uuid']})
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
mocked_client.update_floatingip.assert_called_once_with(
|
|
fip_id, {'floatingip': {'port_id': uuids.portid_1,
|
|
'fixed_ip_address': new_fixed_address}})
|
|
mocked_client.show_port.assert_called_once_with(
|
|
self.fip_associated['port_id'])
|
|
mock_cache_update.assert_has_calls([
|
|
mock.call(mock.ANY, mock_get.return_value['uuid'], mock.ANY),
|
|
mock.call(mock.ANY, instance2['uuid'], mock.ANY)])
|
|
self.assertEqual(2, mock_cache_update.call_count)
|
|
mock_get_nw.assert_has_calls([
|
|
mock.call(mock.ANY, mock_get.return_value),
|
|
mock.call(mock.ANY, instance2)])
|
|
self.assertEqual(2, mock_get_nw.call_count)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_associate_floating_ip_not_found_fixed_ip(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
instance = self._fake_instance_object(self.instance)
|
|
address = self.fip_associated['floating_ip_address']
|
|
fixed_address = self.fip_associated['fixed_ip_address']
|
|
mocked_client.list_ports.return_value = {'ports': [self.port_data2[0]]}
|
|
|
|
self.assertRaises(exception.FixedIpNotFoundForAddress,
|
|
self.api.associate_floating_ip, self.context,
|
|
instance, address, fixed_address)
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
**{'device_owner': 'compute:nova',
|
|
'device_id': self.instance['uuid']})
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_disassociate_floating_ip(self, mock_get_client,
|
|
mock_cache_update, mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
instance = self._fake_instance_object(self.instance)
|
|
address = self.fip_associated['floating_ip_address']
|
|
fip_id = self.fip_associated['id']
|
|
mocked_client.list_floatingips.return_value = {
|
|
'floatingips': [self.fip_associated]}
|
|
|
|
self.api.disassociate_floating_ip(self.context, instance, address)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
floating_ip_address=address)
|
|
mocked_client.update_floatingip.assert_called_once_with(
|
|
fip_id, {'floatingip': {'port_id': None}})
|
|
mock_cache_update.assert_called_once_with(mock.ANY, instance['uuid'],
|
|
mock.ANY)
|
|
mock_get_nw.assert_called_once_with(mock.ANY, instance)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_add_fixed_ip_to_instance(self, mock_get_client,
|
|
mock_cache_update, mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
instance = self._fake_instance_object(self.instance)
|
|
network_id = uuids.my_netid1
|
|
mocked_client.list_subnets.return_value = {
|
|
'subnets': self.subnet_data_n}
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data1}
|
|
port_req_body = {
|
|
'port': {
|
|
'fixed_ips': [{'subnet_id': 'my_subid1'},
|
|
{'subnet_id': 'my_subid1'}],
|
|
},
|
|
}
|
|
port = self.port_data1[0]
|
|
port['fixed_ips'] = [{'subnet_id': 'my_subid1'}]
|
|
mocked_client.update_port.return_value = {'port': port}
|
|
|
|
self.api.add_fixed_ip_to_instance(self.context,
|
|
instance,
|
|
network_id)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_subnets.assert_called_once_with(
|
|
network_id=network_id)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
device_id=instance.uuid, device_owner='compute:nova',
|
|
network_id=network_id)
|
|
mocked_client.update_port.assert_called_once_with(uuids.portid_1,
|
|
port_req_body)
|
|
mock_cache_update.assert_called_once_with(mock.ANY, instance['uuid'],
|
|
mock.ANY)
|
|
mock_get_nw.assert_called_once_with(mock.ANY, instance)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info',
|
|
return_value=model.NetworkInfo())
|
|
@mock.patch.object(db_api, 'instance_info_cache_update',
|
|
return_value=fake_info_cache)
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_remove_fixed_ip_from_instance(self, mock_get_client,
|
|
mock_cache_update, mock_get_nw):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
instance = self._fake_instance_object(self.instance)
|
|
address = '10.0.0.3'
|
|
zone = 'compute:%s' % self.instance['availability_zone']
|
|
mocked_client.list_ports.return_value = {'ports': self.port_data1}
|
|
port_req_body = {
|
|
'port': {
|
|
'fixed_ips': [],
|
|
},
|
|
}
|
|
port = self.port_data1[0]
|
|
port['fixed_ips'] = []
|
|
mocked_client.update_port.return_value = {'port': port}
|
|
|
|
self.api.remove_fixed_ip_from_instance(self.context, instance,
|
|
address)
|
|
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
device_id=self.instance['uuid'], device_owner=zone,
|
|
fixed_ips='ip_address=%s' % address)
|
|
mocked_client.update_port.assert_called_once_with(uuids.portid_1,
|
|
port_req_body)
|
|
mock_cache_update.assert_called_once_with(mock.ANY, instance['uuid'],
|
|
mock.ANY)
|
|
mock_get_nw.assert_called_once_with(mock.ANY, instance)
|
|
|
|
def test_list_floating_ips_without_l3_support(self):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mocked_client.list_floatingips.side_effect = exceptions.NotFound
|
|
|
|
floatingips = self.api._get_floating_ips_by_fixed_and_port(
|
|
mocked_client, '1.1.1.1', 1)
|
|
|
|
self.assertEqual([], floatingips)
|
|
mocked_client.list_floatingips.assert_called_once_with(
|
|
fixed_ip_address='1.1.1.1', port_id=1)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_floating_ips_by_fixed_and_port')
|
|
def test_nw_info_get_ips(self, mock_get_floating):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
fake_port = {
|
|
'fixed_ips': [
|
|
{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
}
|
|
mock_get_floating.return_value = [{'floating_ip_address': '10.0.0.1'}]
|
|
|
|
result = self.api._nw_info_get_ips(mocked_client, fake_port)
|
|
|
|
self.assertEqual(1, len(result))
|
|
self.assertEqual('1.1.1.1', result[0]['address'])
|
|
self.assertEqual('10.0.0.1', result[0]['floating_ips'][0]['address'])
|
|
mock_get_floating.assert_called_once_with(mocked_client, '1.1.1.1',
|
|
'port-id')
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_subnets_from_port')
|
|
def test_nw_info_get_subnets(self, mock_get_subnets):
|
|
fake_port = {
|
|
'fixed_ips': [
|
|
{'ip_address': '1.1.1.1'},
|
|
{'ip_address': '2.2.2.2'}],
|
|
'id': 'port-id',
|
|
}
|
|
fake_subnet = model.Subnet(cidr='1.0.0.0/8')
|
|
fake_ips = [model.IP(x['ip_address']) for x in fake_port['fixed_ips']]
|
|
mock_get_subnets.return_value = [fake_subnet]
|
|
|
|
subnets = self.api._nw_info_get_subnets(self.context, fake_port,
|
|
fake_ips)
|
|
|
|
self.assertEqual(1, len(subnets))
|
|
self.assertEqual(1, len(subnets[0]['ips']))
|
|
self.assertEqual('1.1.1.1', subnets[0]['ips'][0]['address'])
|
|
mock_get_subnets.assert_called_once_with(self.context, fake_port, None)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_nw_info_build_network(self, vif_type, mock_get_client,
|
|
mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port = {
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
'network_id': 'net-id',
|
|
'binding:vif_type': vif_type,
|
|
}
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
fake_nets = [{'id': 'net-id', 'name': 'foo', 'tenant_id': 'tenant',
|
|
'mtu': 9000}]
|
|
|
|
net, iid = self.api._nw_info_build_network(self.context, fake_port,
|
|
fake_nets, fake_subnets)
|
|
|
|
self.assertEqual(fake_subnets, net['subnets'])
|
|
self.assertEqual('net-id', net['id'])
|
|
self.assertEqual('foo', net['label'])
|
|
self.assertEqual('tenant', net.get_meta('tenant_id'))
|
|
self.assertEqual(9000, net.get_meta('mtu'))
|
|
self.assertEqual(CONF.flat_injected, net.get_meta('injected'))
|
|
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mock_get_physnet.assert_called_once_with(self.context, mocked_client,
|
|
'net-id')
|
|
|
|
return net, iid
|
|
|
|
def test_nw_info_build_network_ovs(self):
|
|
net, iid = self._test_nw_info_build_network(model.VIF_TYPE_OVS)
|
|
self.assertEqual(CONF.neutron.ovs_bridge, net['bridge'])
|
|
self.assertNotIn('should_create_bridge', net)
|
|
self.assertEqual('port-id', iid)
|
|
|
|
def test_nw_info_build_network_dvs(self):
|
|
net, iid = self._test_nw_info_build_network(model.VIF_TYPE_DVS)
|
|
self.assertEqual('net-id', net['bridge'])
|
|
self.assertNotIn('should_create_bridge', net)
|
|
self.assertNotIn('ovs_interfaceid', net)
|
|
self.assertIsNone(iid)
|
|
|
|
def test_nw_info_build_network_bridge(self):
|
|
net, iid = self._test_nw_info_build_network(model.VIF_TYPE_BRIDGE)
|
|
self.assertEqual('brqnet-id', net['bridge'])
|
|
self.assertTrue(net['should_create_bridge'])
|
|
self.assertIsNone(iid)
|
|
|
|
def test_nw_info_build_network_tap(self):
|
|
net, iid = self._test_nw_info_build_network(model.VIF_TYPE_TAP)
|
|
self.assertIsNone(net['bridge'])
|
|
self.assertNotIn('should_create_bridge', net)
|
|
self.assertIsNone(iid)
|
|
|
|
def test_nw_info_build_network_other(self):
|
|
net, iid = self._test_nw_info_build_network(None)
|
|
self.assertIsNone(net['bridge'])
|
|
self.assertNotIn('should_create_bridge', net)
|
|
self.assertIsNone(iid)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_nw_info_build_no_match(self, mock_get_client, mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port = {
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
'network_id': 'net-id1',
|
|
'tenant_id': 'tenant',
|
|
'binding:vif_type': model.VIF_TYPE_OVS,
|
|
}
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
fake_nets = [{'id': 'net-id2', 'name': 'foo', 'tenant_id': 'tenant'}]
|
|
net, iid = self.api._nw_info_build_network(self.context, fake_port,
|
|
fake_nets, fake_subnets)
|
|
self.assertEqual(fake_subnets, net['subnets'])
|
|
self.assertEqual('net-id1', net['id'])
|
|
self.assertEqual('tenant', net['meta']['tenant_id'])
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mock_get_physnet.assert_called_once_with(self.context, mocked_client,
|
|
'net-id1')
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_nw_info_build_network_vhostuser(self, mock_get_client,
|
|
mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port = {
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
'network_id': 'net-id',
|
|
'binding:vif_type': model.VIF_TYPE_VHOSTUSER,
|
|
'binding:vif_details': {
|
|
model.VIF_DETAILS_VHOSTUSER_OVS_PLUG: True
|
|
}
|
|
}
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
fake_nets = [{'id': 'net-id', 'name': 'foo', 'tenant_id': 'tenant'}]
|
|
net, iid = self.api._nw_info_build_network(self.context, fake_port,
|
|
fake_nets, fake_subnets)
|
|
self.assertEqual(fake_subnets, net['subnets'])
|
|
self.assertEqual('net-id', net['id'])
|
|
self.assertEqual('foo', net['label'])
|
|
self.assertEqual('tenant', net.get_meta('tenant_id'))
|
|
self.assertEqual(CONF.flat_injected, net.get_meta('injected'))
|
|
self.assertEqual(CONF.neutron.ovs_bridge, net['bridge'])
|
|
self.assertNotIn('should_create_bridge', net)
|
|
self.assertEqual('port-id', iid)
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mock_get_physnet.assert_called_once_with(self.context, mocked_client,
|
|
'net-id')
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_nw_info_build_network_vhostuser_fp(self, mock_get_client,
|
|
mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port = {
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
'network_id': 'net-id',
|
|
'binding:vif_type': model.VIF_TYPE_VHOSTUSER,
|
|
'binding:vif_details': {
|
|
model.VIF_DETAILS_VHOSTUSER_FP_PLUG: True,
|
|
model.VIF_DETAILS_VHOSTUSER_OVS_PLUG: False,
|
|
}
|
|
}
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
fake_nets = [{'id': 'net-id', 'name': 'foo', 'tenant_id': 'tenant'}]
|
|
net, ovs_interfaceid = self.api._nw_info_build_network(
|
|
self.context, fake_port, fake_nets, fake_subnets)
|
|
self.assertEqual(fake_subnets, net['subnets'])
|
|
self.assertEqual('net-id', net['id'])
|
|
self.assertEqual('foo', net['label'])
|
|
self.assertEqual('tenant', net.get_meta('tenant_id'))
|
|
self.assertEqual('brqnet-id', net['bridge'])
|
|
self.assertIsNone(ovs_interfaceid)
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mock_get_physnet.assert_called_once_with(self.context, mocked_client,
|
|
'net-id')
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_nw_info_build_custom_bridge(self, vif_type, mock_get_client,
|
|
mock_get_physnet,
|
|
extra_details=None):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_port = {
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'id': 'port-id',
|
|
'network_id': 'net-id',
|
|
'binding:vif_type': vif_type,
|
|
'binding:vif_details': {
|
|
model.VIF_DETAILS_BRIDGE_NAME: 'custom-bridge',
|
|
}
|
|
}
|
|
if extra_details:
|
|
fake_port['binding:vif_details'].update(extra_details)
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
fake_nets = [{'id': 'net-id', 'name': 'foo', 'tenant_id': 'tenant'}]
|
|
net, iid = self.api._nw_info_build_network(self.context, fake_port,
|
|
fake_nets, fake_subnets)
|
|
self.assertNotEqual(CONF.neutron.ovs_bridge, net['bridge'])
|
|
self.assertEqual('custom-bridge', net['bridge'])
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mock_get_physnet.assert_called_once_with(self.context, mocked_client,
|
|
'net-id')
|
|
|
|
def test_nw_info_build_custom_ovs_bridge(self):
|
|
self._test_nw_info_build_custom_bridge(model.VIF_TYPE_OVS)
|
|
|
|
def test_nw_info_build_custom_ovs_bridge_vhostuser(self):
|
|
self._test_nw_info_build_custom_bridge(model.VIF_TYPE_VHOSTUSER,
|
|
extra_details={model.VIF_DETAILS_VHOSTUSER_OVS_PLUG: True})
|
|
|
|
def test_nw_info_build_custom_lb_bridge(self):
|
|
self._test_nw_info_build_custom_bridge(model.VIF_TYPE_BRIDGE)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info',
|
|
return_value=(None, False))
|
|
@mock.patch.object(neutronapi.API, '_get_preexisting_port_ids',
|
|
return_value=['port5'])
|
|
@mock.patch.object(neutronapi.API, '_get_subnets_from_port',
|
|
return_value=[model.Subnet(cidr='1.0.0.0/8')])
|
|
@mock.patch.object(neutronapi.API, '_get_floating_ips_by_fixed_and_port',
|
|
return_value=[{'floating_ip_address': '10.0.0.1'}])
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_build_network_info_model(self, mock_get_client,
|
|
mock_get_floating, mock_get_subnets,
|
|
mock_get_preexisting, mock_get_physnet):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_inst = objects.Instance()
|
|
fake_inst.project_id = uuids.fake
|
|
fake_inst.uuid = uuids.instance
|
|
fake_inst.info_cache = objects.InstanceInfoCache()
|
|
fake_inst.info_cache.network_info = model.NetworkInfo()
|
|
fake_ports = [
|
|
# admin_state_up=True and status='ACTIVE' thus vif.active=True
|
|
{'id': 'port1',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:01',
|
|
'binding:vif_type': model.VIF_TYPE_BRIDGE,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'binding:vif_details': {},
|
|
},
|
|
# admin_state_up=False and status='DOWN' thus vif.active=True
|
|
{'id': 'port2',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': False,
|
|
'status': 'DOWN',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:02',
|
|
'binding:vif_type': model.VIF_TYPE_BRIDGE,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'binding:vif_details': {},
|
|
},
|
|
# admin_state_up=True and status='DOWN' thus vif.active=False
|
|
{'id': 'port0',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'DOWN',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:03',
|
|
'binding:vif_type': model.VIF_TYPE_BRIDGE,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'binding:vif_details': {},
|
|
},
|
|
# admin_state_up=True and status='ACTIVE' thus vif.active=True
|
|
{'id': 'port3',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:04',
|
|
'binding:vif_type': model.VIF_TYPE_HW_VEB,
|
|
'binding:vnic_type': model.VNIC_TYPE_DIRECT,
|
|
constants.BINDING_PROFILE: {'pci_vendor_info': '1137:0047',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'physnet1'},
|
|
'binding:vif_details': {model.VIF_DETAILS_PROFILEID: 'pfid'},
|
|
},
|
|
# admin_state_up=True and status='ACTIVE' thus vif.active=True
|
|
{'id': 'port4',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:05',
|
|
'binding:vif_type': model.VIF_TYPE_802_QBH,
|
|
'binding:vnic_type': model.VNIC_TYPE_MACVTAP,
|
|
constants.BINDING_PROFILE: {'pci_vendor_info': '1137:0047',
|
|
'pci_slot': '0000:0a:00.2',
|
|
'physical_network': 'physnet1'},
|
|
'binding:vif_details': {model.VIF_DETAILS_PROFILEID: 'pfid'},
|
|
},
|
|
# admin_state_up=True and status='ACTIVE' thus vif.active=True
|
|
# This port has no binding:vnic_type to verify default is assumed
|
|
{'id': 'port5',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:06',
|
|
'binding:vif_type': model.VIF_TYPE_BRIDGE,
|
|
# No binding:vnic_type
|
|
'binding:vif_details': {},
|
|
},
|
|
# This does not match the networks we provide below,
|
|
# so it should be ignored (and is here to verify that)
|
|
{'id': 'port6',
|
|
'network_id': 'other-net-id',
|
|
'admin_state_up': True,
|
|
'status': 'DOWN',
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
},
|
|
]
|
|
fake_nets = [
|
|
{'id': 'net-id',
|
|
'name': 'foo',
|
|
'tenant_id': uuids.fake,
|
|
}
|
|
]
|
|
mocked_client.list_ports.return_value = {'ports': fake_ports}
|
|
|
|
requested_ports = [fake_ports[2], fake_ports[0], fake_ports[1],
|
|
fake_ports[3], fake_ports[4], fake_ports[5]]
|
|
expected_get_floating_calls = []
|
|
for requested_port in requested_ports:
|
|
expected_get_floating_calls.append(mock.call(mocked_client,
|
|
'1.1.1.1',
|
|
requested_port['id']))
|
|
expected_get_subnets_calls = []
|
|
for requested_port in requested_ports:
|
|
expected_get_subnets_calls.append(
|
|
mock.call(self.context, requested_port, mocked_client))
|
|
|
|
fake_inst.info_cache = objects.InstanceInfoCache.new(
|
|
self.context, uuids.instance)
|
|
fake_inst.info_cache.network_info = model.NetworkInfo.hydrate([])
|
|
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context, fake_inst,
|
|
fake_nets,
|
|
[fake_ports[2]['id'],
|
|
fake_ports[0]['id'],
|
|
fake_ports[1]['id'],
|
|
fake_ports[3]['id'],
|
|
fake_ports[4]['id'],
|
|
fake_ports[5]['id']],
|
|
preexisting_port_ids=['port3'])
|
|
|
|
self.assertEqual(6, len(nw_infos))
|
|
index = 0
|
|
for nw_info in nw_infos:
|
|
self.assertEqual(requested_ports[index]['mac_address'],
|
|
nw_info['address'])
|
|
self.assertEqual('tapport' + str(index), nw_info['devname'])
|
|
self.assertIsNone(nw_info['ovs_interfaceid'])
|
|
self.assertEqual(requested_ports[index]['binding:vif_type'],
|
|
nw_info['type'])
|
|
if nw_info['type'] == model.VIF_TYPE_BRIDGE:
|
|
self.assertEqual('brqnet-id', nw_info['network']['bridge'])
|
|
self.assertEqual(requested_ports[index].get('binding:vnic_type',
|
|
model.VNIC_TYPE_NORMAL), nw_info['vnic_type'])
|
|
self.assertEqual(requested_ports[index].get('binding:vif_details'),
|
|
nw_info.get('details'))
|
|
self.assertEqual(
|
|
# If the requested port does not define a binding:profile, or
|
|
# has it set to None, we default to an empty dict to avoid
|
|
# NoneType errors.
|
|
requested_ports[index].get(
|
|
constants.BINDING_PROFILE) or {},
|
|
nw_info.get('profile'))
|
|
self.assertTrue(nw_info.get('delegate_create'))
|
|
index += 1
|
|
|
|
self.assertFalse(nw_infos[0]['active'])
|
|
self.assertTrue(nw_infos[1]['active'])
|
|
self.assertTrue(nw_infos[2]['active'])
|
|
self.assertTrue(nw_infos[3]['active'])
|
|
self.assertTrue(nw_infos[4]['active'])
|
|
self.assertTrue(nw_infos[5]['active'])
|
|
|
|
self.assertEqual('port0', nw_infos[0]['id'])
|
|
self.assertEqual('port1', nw_infos[1]['id'])
|
|
self.assertEqual('port2', nw_infos[2]['id'])
|
|
self.assertEqual('port3', nw_infos[3]['id'])
|
|
self.assertEqual('port4', nw_infos[4]['id'])
|
|
self.assertEqual('port5', nw_infos[5]['id'])
|
|
|
|
self.assertFalse(nw_infos[0]['preserve_on_delete'])
|
|
self.assertFalse(nw_infos[1]['preserve_on_delete'])
|
|
self.assertFalse(nw_infos[2]['preserve_on_delete'])
|
|
self.assertTrue(nw_infos[3]['preserve_on_delete'])
|
|
self.assertFalse(nw_infos[4]['preserve_on_delete'])
|
|
self.assertTrue(nw_infos[5]['preserve_on_delete'])
|
|
|
|
mock_get_client.assert_has_calls([
|
|
mock.call(self.context, admin=True)] * 7, any_order=True)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.fake, device_id=uuids.instance)
|
|
mock_get_floating.assert_has_calls(expected_get_floating_calls)
|
|
self.assertEqual(len(expected_get_floating_calls),
|
|
mock_get_floating.call_count)
|
|
mock_get_subnets.assert_has_calls(expected_get_subnets_calls)
|
|
self.assertEqual(len(expected_get_subnets_calls),
|
|
mock_get_subnets.call_count)
|
|
mock_get_preexisting.assert_called_once_with(fake_inst)
|
|
mock_get_physnet.assert_has_calls([
|
|
mock.call(self.context, mocked_client, 'net-id')] * 6)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.network.neutron.API._nw_info_get_subnets')
|
|
@mock.patch('nova.network.neutron.API._nw_info_get_ips')
|
|
@mock.patch('nova.network.neutron.API._nw_info_build_network')
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
@mock.patch('nova.network.neutron.API._gather_port_ids_and_networks')
|
|
def test_build_network_info_model_empty(
|
|
self, mock_gather_port_ids_and_networks,
|
|
mock_get_preexisting_port_ids,
|
|
mock_nw_info_build_network,
|
|
mock_nw_info_get_ips,
|
|
mock_nw_info_get_subnets,
|
|
mock_get_client):
|
|
# An empty instance info network cache should not be populated from
|
|
# ports found in Neutron.
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_inst = objects.Instance()
|
|
fake_inst.project_id = uuids.fake
|
|
fake_inst.uuid = uuids.instance
|
|
fake_inst.info_cache = objects.InstanceInfoCache()
|
|
fake_inst.info_cache.network_info = model.NetworkInfo()
|
|
fake_ports = [
|
|
# admin_state_up=True and status='ACTIVE' thus vif.active=True
|
|
{'id': 'port1',
|
|
'network_id': 'net-id',
|
|
'admin_state_up': True,
|
|
'status': 'ACTIVE',
|
|
'fixed_ips': [{'ip_address': '1.1.1.1'}],
|
|
'mac_address': 'de:ad:be:ef:00:01',
|
|
'binding:vif_type': model.VIF_TYPE_BRIDGE,
|
|
'binding:vnic_type': model.VNIC_TYPE_NORMAL,
|
|
'binding:vif_details': {},
|
|
},
|
|
]
|
|
fake_subnets = [model.Subnet(cidr='1.0.0.0/8')]
|
|
|
|
mocked_client.list_ports.return_value = {'ports': fake_ports}
|
|
|
|
mock_gather_port_ids_and_networks.return_value = ([], [])
|
|
mock_get_preexisting_port_ids.return_value = []
|
|
mock_nw_info_build_network.return_value = (None, None)
|
|
mock_nw_info_get_ips.return_value = []
|
|
mock_nw_info_get_subnets.return_value = fake_subnets
|
|
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context, fake_inst)
|
|
|
|
self.assertEqual(0, len(nw_infos))
|
|
mock_get_client.assert_called_once_with(self.context, admin=True)
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
tenant_id=uuids.fake, device_id=uuids.instance)
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_physnet_tunneled_info',
|
|
new=mock.Mock(return_value=(None, False)))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_preexisting_port_ids',
|
|
new=mock.Mock(return_value=[]))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_subnets_from_port',
|
|
new=mock.Mock(return_value=[model.Subnet(cidr='1.0.0.0/8')]))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_floating_ips_by_fixed_and_port',
|
|
new=mock.Mock(return_value=[{'floating_ip_address': '10.0.0.1'}]))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_build_network_info_model_full_vnic_type_change(
|
|
self, mock_get_client
|
|
):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_inst = objects.Instance()
|
|
fake_inst.project_id = uuids.fake
|
|
fake_inst.uuid = uuids.instance
|
|
fake_ports = [
|
|
{
|
|
"id": "port1",
|
|
"network_id": "net-id",
|
|
"tenant_id": uuids.fake,
|
|
"admin_state_up": True,
|
|
"status": "ACTIVE",
|
|
"fixed_ips": [{"ip_address": "1.1.1.1"}],
|
|
"mac_address": "de:ad:be:ef:00:01",
|
|
"binding:vif_type": model.VIF_TYPE_BRIDGE,
|
|
"binding:vnic_type": model.VNIC_TYPE_DIRECT,
|
|
"binding:vif_details": {},
|
|
},
|
|
]
|
|
mocked_client.list_ports.return_value = {'ports': fake_ports}
|
|
fake_inst.info_cache = objects.InstanceInfoCache.new(
|
|
self.context, uuids.instance)
|
|
fake_inst.info_cache.network_info = model.NetworkInfo.hydrate([])
|
|
|
|
# build the network info first
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context,
|
|
fake_inst,
|
|
force_refresh=True,
|
|
)
|
|
|
|
self.assertEqual(1, len(nw_infos))
|
|
fake_inst.info_cache.network_info = nw_infos
|
|
|
|
# change the vnic_type of the port and rebuild the network info
|
|
fake_ports[0]["binding:vnic_type"] = model.VNIC_TYPE_MACVTAP
|
|
with mock.patch(
|
|
"nova.network.neutron.API._log_error_if_vnic_type_changed"
|
|
) as mock_log:
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context,
|
|
fake_inst,
|
|
force_refresh=True,
|
|
)
|
|
|
|
mock_log.assert_called_once_with(
|
|
fake_ports[0]["id"], "direct", "macvtap", fake_inst)
|
|
self.assertEqual(1, len(nw_infos))
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_physnet_tunneled_info',
|
|
new=mock.Mock(return_value=(None, False)))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_preexisting_port_ids',
|
|
new=mock.Mock(return_value=[]))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_subnets_from_port',
|
|
new=mock.Mock(return_value=[model.Subnet(cidr='1.0.0.0/8')]))
|
|
@mock.patch.object(
|
|
neutronapi.API,
|
|
'_get_floating_ips_by_fixed_and_port',
|
|
new=mock.Mock(return_value=[{'floating_ip_address': '10.0.0.1'}]))
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_build_network_info_model_single_vnic_type_change(
|
|
self, mock_get_client
|
|
):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
fake_inst = objects.Instance()
|
|
fake_inst.project_id = uuids.fake
|
|
fake_inst.uuid = uuids.instance
|
|
fake_ports = [
|
|
{
|
|
"id": "port1",
|
|
"network_id": "net-id",
|
|
"tenant_id": uuids.fake,
|
|
"admin_state_up": True,
|
|
"status": "ACTIVE",
|
|
"fixed_ips": [{"ip_address": "1.1.1.1"}],
|
|
"mac_address": "de:ad:be:ef:00:01",
|
|
"binding:vif_type": model.VIF_TYPE_BRIDGE,
|
|
"binding:vnic_type": model.VNIC_TYPE_DIRECT,
|
|
"binding:vif_details": {},
|
|
},
|
|
]
|
|
fake_nets = [
|
|
{
|
|
"id": "net-id",
|
|
"name": "foo",
|
|
"tenant_id": uuids.fake,
|
|
}
|
|
]
|
|
mocked_client.list_ports.return_value = {'ports': fake_ports}
|
|
fake_inst.info_cache = objects.InstanceInfoCache.new(
|
|
self.context, uuids.instance)
|
|
fake_inst.info_cache.network_info = model.NetworkInfo.hydrate([])
|
|
|
|
# build the network info first
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context,
|
|
fake_inst,
|
|
fake_nets,
|
|
[fake_ports[0]["id"]],
|
|
refresh_vif_id=fake_ports[0]["id"],
|
|
)
|
|
|
|
self.assertEqual(1, len(nw_infos))
|
|
fake_inst.info_cache.network_info = nw_infos
|
|
|
|
# change the vnic_type of the port and rebuild the network info
|
|
fake_ports[0]["binding:vnic_type"] = model.VNIC_TYPE_MACVTAP
|
|
with mock.patch(
|
|
"nova.network.neutron.API._log_error_if_vnic_type_changed"
|
|
) as mock_log:
|
|
nw_infos = self.api._build_network_info_model(
|
|
self.context,
|
|
fake_inst,
|
|
fake_nets,
|
|
[fake_ports[0]["id"]],
|
|
refresh_vif_id=fake_ports[0]["id"],
|
|
)
|
|
|
|
mock_log.assert_called_once_with(
|
|
fake_ports[0]["id"], "direct", "macvtap", fake_inst)
|
|
self.assertEqual(1, len(nw_infos))
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_subnets_from_port(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
port_data = copy.copy(self.port_data1[0])
|
|
# add another IP on the same subnet and verify the subnet is deduped
|
|
port_data['fixed_ips'].append({'ip_address': '10.0.1.3',
|
|
'subnet_id': 'my_subid1'})
|
|
subnet_data1 = copy.copy(self.subnet_data1)
|
|
subnet_data1[0]['host_routes'] = [
|
|
{'destination': '192.168.0.0/24', 'nexthop': '1.0.0.10'}
|
|
]
|
|
mocked_client.list_subnets.return_value = {'subnets': subnet_data1}
|
|
mocked_client.list_ports.return_value = {'ports': []}
|
|
|
|
subnets = self.api._get_subnets_from_port(self.context, port_data)
|
|
|
|
self.assertEqual(1, len(subnets))
|
|
self.assertEqual(1, len(subnets[0]['routes']))
|
|
self.assertEqual(subnet_data1[0]['host_routes'][0]['destination'],
|
|
subnets[0]['routes'][0]['cidr'])
|
|
self.assertEqual(subnet_data1[0]['host_routes'][0]['nexthop'],
|
|
subnets[0]['routes'][0]['gateway']['address'])
|
|
mock_get_client.assert_called_once_with(self.context)
|
|
mocked_client.list_subnets.assert_called_once_with(
|
|
id=[port_data['fixed_ips'][0]['subnet_id']])
|
|
mocked_client.list_ports.assert_called_once_with(
|
|
network_id=subnet_data1[0]['network_id'],
|
|
device_owner='network:dhcp')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_subnets_from_port_enabled_dhcp(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
|
|
port_data = copy.copy(self.port_data1[0])
|
|
# add another IP on the same subnet and verify the subnet is deduped
|
|
port_data['fixed_ips'].append({'ip_address': '10.0.1.3',
|
|
'subnet_id': 'my_subid1'})
|
|
subnet_data1 = copy.copy(self.subnet_data1)
|
|
subnet_data1[0]['enable_dhcp'] = True
|
|
|
|
mocked_client.list_subnets.return_value = {'subnets': subnet_data1}
|
|
mocked_client.list_ports.return_value = {'ports': self.dhcp_port_data1}
|
|
|
|
subnets = self.api._get_subnets_from_port(self.context, port_data)
|
|
|
|
self.assertEqual(self.dhcp_port_data1[0]['fixed_ips'][0]['ip_address'],
|
|
subnets[0]['meta']['dhcp_server'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_subnets_from_port_enabled_dhcp_no_dhcp_ports(self,
|
|
mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
|
|
port_data = copy.copy(self.port_data1[0])
|
|
# add another IP on the same subnet and verify the subnet is deduped
|
|
port_data['fixed_ips'].append({'ip_address': '10.0.1.3',
|
|
'subnet_id': 'my_subid1'})
|
|
subnet_data1 = copy.copy(self.subnet_data1)
|
|
subnet_data1[0]['enable_dhcp'] = True
|
|
|
|
mocked_client.list_subnets.return_value = {'subnets': subnet_data1}
|
|
mocked_client.list_ports.return_value = {'ports': []}
|
|
|
|
subnets = self.api._get_subnets_from_port(self.context, port_data)
|
|
|
|
self.assertEqual(subnet_data1[0]['gateway_ip'],
|
|
subnets[0]['meta']['dhcp_server'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_physnet_tunneled_info_multi_segment(self, mock_get_client):
|
|
test_net = {'network': {'segments':
|
|
[{'provider:physical_network': 'physnet10',
|
|
'provider:segmentation_id': 1000,
|
|
'provider:network_type': 'vlan'},
|
|
{'provider:physical_network': None,
|
|
'provider:segmentation_id': 153,
|
|
'provider:network_type': 'vxlan'}]}}
|
|
test_ext_list = {'extensions':
|
|
[{'name': 'Multi Provider Network',
|
|
'alias': 'multi-provider'}]}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
mock_client.show_network.return_value = test_net
|
|
physnet_name, tunneled = self.api._get_physnet_tunneled_info(
|
|
self.context, mock_client, 'test-net')
|
|
|
|
mock_client.show_network.assert_called_once_with(
|
|
'test-net', fields='segments')
|
|
self.assertEqual('physnet10', physnet_name)
|
|
self.assertFalse(tunneled)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_physnet_tunneled_info_vlan_with_multi_segment_ext(
|
|
self, mock_get_client):
|
|
test_net = {'network': {'provider:physical_network': 'physnet10',
|
|
'provider:segmentation_id': 1000,
|
|
'provider:network_type': 'vlan'}}
|
|
test_ext_list = {'extensions':
|
|
[{'name': 'Multi Provider Network',
|
|
'alias': 'multi-provider'}]}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
mock_client.show_network.return_value = test_net
|
|
physnet_name, tunneled = self.api._get_physnet_tunneled_info(
|
|
self.context, mock_client, 'test-net')
|
|
|
|
mock_client.show_network.assert_called_with(
|
|
'test-net', fields=['provider:physical_network',
|
|
'provider:network_type'])
|
|
self.assertEqual('physnet10', physnet_name)
|
|
self.assertFalse(tunneled)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_physnet_tunneled_info_multi_segment_no_physnet(
|
|
self, mock_get_client):
|
|
test_net = {'network': {'segments':
|
|
[{'provider:physical_network': None,
|
|
'provider:segmentation_id': 1000,
|
|
'provider:network_type': 'vlan'},
|
|
{'provider:physical_network': None,
|
|
'provider:segmentation_id': 153,
|
|
'provider:network_type': 'vlan'}]}}
|
|
test_ext_list = {'extensions':
|
|
[{'name': 'Multi Provider Network',
|
|
'alias': 'multi-provider'}]}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
mock_client.show_network.return_value = test_net
|
|
self.assertRaises(exception.NovaException,
|
|
self.api._get_physnet_tunneled_info,
|
|
self.context, mock_client, 'test-net')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_physnet_tunneled_info_tunneled(
|
|
self, mock_get_client):
|
|
test_net = {'network': {'provider:network_type': 'vxlan'}}
|
|
test_ext_list = {'extensions': []}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
mock_client.show_network.return_value = test_net
|
|
physnet_name, tunneled = self.api._get_physnet_tunneled_info(
|
|
self.context, mock_client, 'test-net')
|
|
|
|
mock_client.show_network.assert_called_once_with(
|
|
'test-net', fields=['provider:physical_network',
|
|
'provider:network_type'])
|
|
self.assertTrue(tunneled)
|
|
self.assertIsNone(physnet_name)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_phynet_tunneled_info_non_tunneled(
|
|
self, mock_get_client):
|
|
test_net = {'network': {'provider:network_type': 'vlan'}}
|
|
test_ext_list = {'extensions': []}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
mock_client.show_network.return_value = test_net
|
|
physnet_name, tunneled = self.api._get_physnet_tunneled_info(
|
|
self.context, mock_client, 'test-net')
|
|
|
|
mock_client.show_network.assert_called_once_with(
|
|
'test-net', fields=['provider:physical_network',
|
|
'provider:network_type'])
|
|
self.assertFalse(tunneled)
|
|
self.assertIsNone(physnet_name)
|
|
|
|
def test_is_remote_managed(self):
|
|
cases = {
|
|
(model.VNIC_TYPE_NORMAL, False),
|
|
(model.VNIC_TYPE_DIRECT, False),
|
|
(model.VNIC_TYPE_MACVTAP, False),
|
|
(model.VNIC_TYPE_DIRECT_PHYSICAL, False),
|
|
(model.VNIC_TYPE_BAREMETAL, False),
|
|
(model.VNIC_TYPE_VIRTIO_FORWARDER, False),
|
|
(model.VNIC_TYPE_VDPA, False),
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT, False),
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT_PHYSICAL, False),
|
|
(model.VNIC_TYPE_REMOTE_MANAGED, True),
|
|
}
|
|
|
|
for vnic_type, expected in cases:
|
|
self.assertEqual(self.api._is_remote_managed(vnic_type), expected)
|
|
|
|
def _test_get_port_vnic_info(
|
|
self, mock_get_client, binding_vnic_type, expected_vnic_type,
|
|
port_resource_request=None, numa_policy=None
|
|
):
|
|
test_port = {
|
|
'port': {'id': 'my_port_id2',
|
|
'network_id': 'net-id',
|
|
},
|
|
}
|
|
|
|
if binding_vnic_type:
|
|
test_port['port']['binding:vnic_type'] = binding_vnic_type
|
|
if port_resource_request:
|
|
test_port['port'][
|
|
constants.RESOURCE_REQUEST] = port_resource_request
|
|
if numa_policy:
|
|
test_port['port'][constants.NUMA_POLICY] = numa_policy
|
|
|
|
mock_get_client.reset_mock()
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.show_port.return_value = test_port
|
|
|
|
(vnic_type, trusted, network_id, resource_request,
|
|
numa, device_profile) = (
|
|
self.api._get_port_vnic_info(
|
|
self.context, mock_client, test_port['port']['id']))
|
|
|
|
mock_client.show_port.assert_called_once_with(test_port['port']['id'],
|
|
fields=['binding:vnic_type', 'binding:profile', 'network_id',
|
|
constants.RESOURCE_REQUEST, constants.NUMA_POLICY,
|
|
'device_profile'])
|
|
self.assertEqual(expected_vnic_type, vnic_type)
|
|
self.assertEqual('net-id', network_id)
|
|
self.assertIsNone(trusted)
|
|
self.assertEqual(port_resource_request, resource_request)
|
|
self.assertEqual(numa_policy, numa)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.MagicMock())
|
|
def test_get_port_vnic_info_1(self, mock_get_client):
|
|
self._test_get_port_vnic_info(mock_get_client, model.VNIC_TYPE_DIRECT,
|
|
model.VNIC_TYPE_DIRECT)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_2(self, mock_get_client):
|
|
self._test_get_port_vnic_info(mock_get_client, model.VNIC_TYPE_NORMAL,
|
|
model.VNIC_TYPE_NORMAL)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_3(self, mock_get_client):
|
|
self._test_get_port_vnic_info(mock_get_client, None,
|
|
model.VNIC_TYPE_NORMAL)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_4(self, mock_get_client):
|
|
policies = ['required', 'legacy', 'preferred']
|
|
for policy_name in policies:
|
|
self._test_get_port_vnic_info(
|
|
mock_get_client, None, model.VNIC_TYPE_NORMAL,
|
|
numa_policy=policy_name)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_port_vnic_info_requested_resources(self, mock_get_client):
|
|
self._test_get_port_vnic_info(
|
|
mock_get_client, None, model.VNIC_TYPE_NORMAL,
|
|
port_resource_request={
|
|
"resources": {
|
|
"NET_BW_EGR_KILOBIT_PER_SEC": 6000,
|
|
"NET_BW_IGR_KILOBIT_PER_SEC": 6000,
|
|
},
|
|
"required": [
|
|
"CUSTOM_PHYSNET_PHYSNET0",
|
|
"CUSTOM_VNIC_TYPE_NORMAL"
|
|
]
|
|
}
|
|
)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_5(self, mock_get_client):
|
|
self._test_get_port_vnic_info(mock_get_client,
|
|
model.VNIC_TYPE_ACCELERATOR_DIRECT,
|
|
model.VNIC_TYPE_ACCELERATOR_DIRECT)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_6(self, mock_get_client):
|
|
self._test_get_port_vnic_info(mock_get_client,
|
|
model.VNIC_TYPE_ACCELERATOR_DIRECT_PHYSICAL,
|
|
model.VNIC_TYPE_ACCELERATOR_DIRECT_PHYSICAL)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_get_port_vnic_info_trusted(self, mock_get_client):
|
|
test_port = {
|
|
'port': {'id': 'my_port_id1',
|
|
'network_id': 'net-id',
|
|
'binding:vnic_type': model.VNIC_TYPE_DIRECT,
|
|
'binding:profile': {"trusted": "Yes"},
|
|
},
|
|
}
|
|
test_ext_list = {'extensions': []}
|
|
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.show_port.return_value = test_port
|
|
mock_client.list_extensions.return_value = test_ext_list
|
|
result = self.api._get_port_vnic_info(
|
|
self.context, mock_client, test_port['port']['id'])
|
|
vnic_type, trusted, network_id, resource_requests, _, _ = result
|
|
|
|
mock_client.show_port.assert_called_once_with(test_port['port']['id'],
|
|
fields=['binding:vnic_type', 'binding:profile', 'network_id',
|
|
constants.RESOURCE_REQUEST, constants.NUMA_POLICY,
|
|
'device_profile'])
|
|
self.assertEqual(model.VNIC_TYPE_DIRECT, vnic_type)
|
|
self.assertEqual('net-id', network_id)
|
|
self.assertTrue(trusted)
|
|
self.assertIsNone(resource_requests)
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_deferred_ip_port_immediate_allocation(self, mock_show):
|
|
port = {'network_id': 'my_netid1',
|
|
'device_id': None,
|
|
'id': uuids.port,
|
|
'fixed_ips': [], # no fixed ip
|
|
'ip_allocation': 'immediate', }
|
|
|
|
mock_show.return_value = port
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port['id'])])
|
|
self.assertRaises(exception.PortRequiresFixedIP,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_deferred_ip_port_deferred_allocation(self, mock_show):
|
|
port = {'network_id': 'my_netid1',
|
|
'device_id': None,
|
|
'id': uuids.port,
|
|
'fixed_ips': [], # no fixed ip
|
|
'ip_allocation': 'deferred', }
|
|
|
|
mock_show.return_value = port
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port['id'])])
|
|
count = self.api.validate_networks(self.context, requested_networks, 1)
|
|
self.assertEqual(1, count)
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_deferred_ip_port_none_allocation(self, mock_show):
|
|
"""Test behavior when the 'none' IP allocation policy is used."""
|
|
port = {
|
|
'network_id': 'my_netid1',
|
|
'device_id': None,
|
|
'id': uuids.port,
|
|
'fixed_ips': [], # no fixed ip
|
|
'ip_allocation': 'none',
|
|
'binding:vif_details': {
|
|
'connectivity': 'l2',
|
|
},
|
|
}
|
|
|
|
mock_show.return_value = port
|
|
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=port['id'])])
|
|
count = self.api.validate_networks(self.context, requested_networks, 1)
|
|
self.assertEqual(1, count)
|
|
|
|
@mock.patch('oslo_concurrency.lockutils.lock')
|
|
def test_get_instance_nw_info_locks_per_instance(self, mock_lock):
|
|
instance = objects.Instance(uuid=uuids.fake)
|
|
api = neutronapi.API()
|
|
mock_lock.side_effect = test.TestingException
|
|
self.assertRaises(test.TestingException,
|
|
api.get_instance_nw_info, 'context', instance)
|
|
mock_lock.assert_called_once_with('refresh_cache-%s' % instance.uuid)
|
|
|
|
@mock.patch('nova.network.neutron.LOG')
|
|
def test_get_instance_nw_info_verify_duplicates_ignored(self, mock_log):
|
|
"""test that the returned networks & port_ids from
|
|
_gather_port_ids_and_networks doesn't contain any duplicates
|
|
|
|
The test fakes an instance with two ports connected to two networks.
|
|
The _gather_port_ids_and_networks method will be called with the
|
|
instance and a list of port ids of which one port id is configured
|
|
already to the instance (== duplicate #1) and a list of
|
|
networks that already contains a network to which an instance port
|
|
is connected (== duplicate #2).
|
|
|
|
All-in-all, we expect the resulting port ids list to contain 3 items
|
|
(["instance_port_1", "port_1", "port_2"]) and the resulting networks
|
|
list to contain 3 items (["net_1", "net_2", "instance_network_1"])
|
|
while the warning message for duplicate items was executed twice
|
|
(due to "duplicate #1" & "duplicate #2")
|
|
"""
|
|
|
|
networks = [model.Network(id="net_1"),
|
|
model.Network(id="net_2")]
|
|
port_ids = ["port_1", "port_2"]
|
|
|
|
instance_networks = [{"id": "instance_network_1",
|
|
"name": "fake_network",
|
|
"tenant_id": "fake_tenant_id"}]
|
|
instance_port_ids = ["instance_port_1"]
|
|
|
|
network_info = model.NetworkInfo(
|
|
[{'id': port_ids[0],
|
|
'network': networks[0]},
|
|
{'id': instance_port_ids[0],
|
|
'network': model.Network(
|
|
id=instance_networks[0]["id"],
|
|
label=instance_networks[0]["name"],
|
|
meta={"tenant_id": instance_networks[0]["tenant_id"]})}]
|
|
)
|
|
|
|
instance_uuid = uuids.fake
|
|
instance = objects.Instance(uuid=instance_uuid,
|
|
info_cache=objects.InstanceInfoCache(
|
|
context=self.context,
|
|
instance_uuid=instance_uuid,
|
|
network_info=network_info))
|
|
|
|
new_networks, new_port_ids = self.api._gather_port_ids_and_networks(
|
|
self.context, instance, networks, port_ids)
|
|
|
|
self.assertEqual(new_networks, networks + instance_networks)
|
|
self.assertEqual(new_port_ids, instance_port_ids + port_ids)
|
|
self.assertEqual(2, mock_log.warning.call_count)
|
|
|
|
@mock.patch('oslo_concurrency.lockutils.lock')
|
|
@mock.patch.object(neutronapi.API, '_get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.update_instance_cache_with_nw_info')
|
|
def test_get_instance_nw_info(self, mock_update, mock_get, mock_lock):
|
|
fake_result = mock.sentinel.get_nw_info_result
|
|
mock_get.return_value = fake_result
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
result = self.api.get_instance_nw_info(self.context, instance)
|
|
mock_get.assert_called_once_with(self.context, instance)
|
|
mock_update.assert_called_once_with(self.api, self.context, instance,
|
|
nw_info=fake_result)
|
|
self.assertEqual(fake_result, result)
|
|
|
|
def _test_validate_networks_fixed_ip_no_dup(self, nets, requested_networks,
|
|
ids, list_port_values):
|
|
|
|
def _fake_list_ports(**search_opts):
|
|
for args, return_value in list_port_values:
|
|
if args == search_opts:
|
|
return return_value
|
|
self.fail('Unexpected call to list_ports %s' % search_opts)
|
|
|
|
with test.nested(
|
|
mock.patch.object(client.Client, 'list_ports',
|
|
side_effect=_fake_list_ports),
|
|
mock.patch.object(client.Client, 'list_networks',
|
|
return_value={'networks': nets}),
|
|
mock.patch.object(client.Client, 'show_quota',
|
|
return_value={'quota': {'port': 50}})) as (
|
|
list_ports_mock, list_networks_mock, show_quota_mock):
|
|
|
|
self.api.validate_networks(self.context, requested_networks, 1)
|
|
|
|
self.assertEqual(len(list_port_values),
|
|
len(list_ports_mock.call_args_list))
|
|
list_networks_mock.assert_called_once_with(id=ids)
|
|
show_quota_mock.assert_called_once_with(uuids.my_tenant)
|
|
|
|
def test_validate_networks_over_limit_quota(self):
|
|
"""Test validates that a relevant exception is being raised when
|
|
there are more ports defined, than there is a quota for it.
|
|
"""
|
|
requested_networks = [
|
|
(uuids.my_netid1, '10.0.1.2', None, None, None, None),
|
|
(uuids.my_netid2, '10.0.1.3', None, None, None, None)]
|
|
|
|
list_port_values = [({'network_id': uuids.my_netid1,
|
|
'fixed_ips': 'ip_address=10.0.1.2',
|
|
'fields': 'device_id'},
|
|
{'ports': []}),
|
|
({'network_id': uuids.my_netid2,
|
|
'fixed_ips': 'ip_address=10.0.1.3',
|
|
'fields': 'device_id'},
|
|
{'ports': []}),
|
|
|
|
({'tenant_id': uuids.my_tenant, 'fields': ['id']},
|
|
{'ports': [1, 2, 3, 4, 5]})]
|
|
|
|
nets = [{'subnets': '1'}, {'subnets': '2'}]
|
|
|
|
def _fake_list_ports(**search_opts):
|
|
for args, return_value in list_port_values:
|
|
if args == search_opts:
|
|
return return_value
|
|
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=nets),
|
|
mock.patch.object(client.Client, 'list_ports',
|
|
side_effect=_fake_list_ports),
|
|
mock.patch.object(client.Client, 'show_quota',
|
|
return_value={'quota': {'port': 1}})):
|
|
exc = self.assertRaises(exception.PortLimitExceeded,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
expected_exception_msg = ('The number of defined ports: '
|
|
'%(ports)d is over the limit: '
|
|
'%(quota)d' %
|
|
{'ports': 5,
|
|
'quota': 1})
|
|
self.assertEqual(expected_exception_msg, str(exc))
|
|
|
|
def test_validate_networks_fixed_ip_no_dup1(self):
|
|
# Test validation for a request for a network with a
|
|
# fixed ip that is not already in use because no fixed ips in use
|
|
|
|
nets1 = [{'id': uuids.my_netid1,
|
|
'name': 'my_netname1',
|
|
'subnets': ['mysubnid1'],
|
|
'tenant_id': uuids.my_tenant}]
|
|
|
|
requested_networks = [
|
|
(uuids.my_netid1, '10.0.1.2', None, None, None, None)]
|
|
ids = [uuids.my_netid1]
|
|
list_port_values = [({'network_id': uuids.my_netid1,
|
|
'fixed_ips': 'ip_address=10.0.1.2',
|
|
'fields': 'device_id'},
|
|
{'ports': []}),
|
|
({'tenant_id': uuids.my_tenant, 'fields': ['id']},
|
|
{'ports': []})]
|
|
self._test_validate_networks_fixed_ip_no_dup(nets1, requested_networks,
|
|
ids, list_port_values)
|
|
|
|
def test_validate_networks_fixed_ip_no_dup2(self):
|
|
# Test validation for a request for a network with a
|
|
# fixed ip that is not already in use because not used on this net id
|
|
|
|
nets2 = [{'id': uuids.my_netid1,
|
|
'name': 'my_netname1',
|
|
'subnets': ['mysubnid1'],
|
|
'tenant_id': uuids.my_tenant},
|
|
{'id': uuids.my_netid2,
|
|
'name': 'my_netname2',
|
|
'subnets': ['mysubnid2'],
|
|
'tenant_id': uuids.my_tenant}]
|
|
|
|
requested_networks = [
|
|
(uuids.my_netid1, '10.0.1.2', None, None, None, None),
|
|
(uuids.my_netid2, '10.0.1.3', None, None, None, None)]
|
|
ids = [uuids.my_netid1, uuids.my_netid2]
|
|
list_port_values = [({'network_id': uuids.my_netid1,
|
|
'fixed_ips': 'ip_address=10.0.1.2',
|
|
'fields': 'device_id'},
|
|
{'ports': []}),
|
|
({'network_id': uuids.my_netid2,
|
|
'fixed_ips': 'ip_address=10.0.1.3',
|
|
'fields': 'device_id'},
|
|
{'ports': []}),
|
|
|
|
({'tenant_id': uuids.my_tenant, 'fields': ['id']},
|
|
{'ports': []})]
|
|
|
|
self._test_validate_networks_fixed_ip_no_dup(nets2, requested_networks,
|
|
ids, list_port_values)
|
|
|
|
def test_validate_networks_fixed_ip_dup(self):
|
|
# Test validation for a request for a network with a
|
|
# fixed ip that is already in use
|
|
|
|
requested_networks = [
|
|
(uuids.my_netid1, '10.0.1.2', None, None, None, None)]
|
|
list_port_mock_params = {'network_id': uuids.my_netid1,
|
|
'fixed_ips': 'ip_address=10.0.1.2',
|
|
'fields': 'device_id'}
|
|
list_port_mock_return = {'ports': [({'device_id': 'my_deviceid'})]}
|
|
|
|
with mock.patch.object(client.Client, 'list_ports',
|
|
return_value=list_port_mock_return) as (
|
|
list_ports_mock):
|
|
|
|
self.assertRaises(exception.FixedIpAlreadyInUse,
|
|
self.api.validate_networks,
|
|
self.context, requested_networks, 1)
|
|
|
|
list_ports_mock.assert_called_once_with(**list_port_mock_params)
|
|
|
|
def test_allocate_floating_ip_exceed_limit(self):
|
|
# Verify that the correct exception is thrown when quota exceed
|
|
pool_name = 'dummy'
|
|
api = neutronapi.API()
|
|
with test.nested(
|
|
mock.patch.object(client.Client, 'create_floatingip'),
|
|
mock.patch.object(api,
|
|
'_get_floating_ip_pool_id_by_name_or_id')) as (
|
|
create_mock, get_mock):
|
|
create_mock.side_effect = exceptions.OverQuotaClient()
|
|
|
|
self.assertRaises(exception.FloatingIpLimitExceeded,
|
|
api.allocate_floating_ip,
|
|
self.context, pool_name)
|
|
|
|
def test_allocate_floating_ip_no_ipv4_subnet(self):
|
|
api = neutronapi.API()
|
|
net_id = uuids.fake
|
|
error_msg = ('Bad floatingip request: Network %s does not contain '
|
|
'any IPv4 subnet' % net_id)
|
|
with test.nested(
|
|
mock.patch.object(client.Client, 'create_floatingip'),
|
|
mock.patch.object(api,
|
|
'_get_floating_ip_pool_id_by_name_or_id')) as (
|
|
create_mock, get_mock):
|
|
create_mock.side_effect = exceptions.BadRequest(error_msg)
|
|
|
|
self.assertRaises(exception.FloatingIpBadRequest,
|
|
api.allocate_floating_ip, self.context,
|
|
'ext_net')
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch('nova.network.neutron.API._get_floating_ip_by_address',
|
|
return_value={'port_id': None, 'id': 'abc'})
|
|
def test_release_floating_ip(self, mock_get_ip, mock_ntrn):
|
|
"""Validate default behavior."""
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
address = '172.24.4.227'
|
|
|
|
self.api.release_floating_ip(self.context, address)
|
|
|
|
mock_ntrn.assert_called_once_with(self.context)
|
|
mock_get_ip.assert_called_once_with(mock_nc, address)
|
|
mock_nc.delete_floatingip.assert_called_once_with('abc')
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch('nova.network.neutron.API._get_floating_ip_by_address',
|
|
return_value={'port_id': 'abc', 'id': 'abc'})
|
|
def test_release_floating_ip_associated(self, mock_get_ip, mock_ntrn):
|
|
"""Ensure release fails if a port is still associated with it.
|
|
|
|
If the floating IP has a port associated with it, as indicated by a
|
|
configured port_id, then any attempt to release should fail.
|
|
"""
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
address = '172.24.4.227'
|
|
|
|
self.assertRaises(exception.FloatingIpAssociated,
|
|
self.api.release_floating_ip,
|
|
self.context, address)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch('nova.network.neutron.API._get_floating_ip_by_address',
|
|
return_value={'port_id': None, 'id': 'abc'})
|
|
def test_release_floating_ip_not_found(self, mock_get_ip, mock_ntrn):
|
|
"""Ensure neutron's NotFound exception is correctly handled.
|
|
|
|
Sometimes, trying to delete a floating IP multiple times in a short
|
|
delay can trigger an exception because the operation is not atomic. If
|
|
neutronclient's call to delete fails with a NotFound error, then we
|
|
should correctly handle this.
|
|
"""
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.delete_floatingip.side_effect = exceptions.NotFound()
|
|
address = '172.24.4.227'
|
|
|
|
self.assertRaises(exception.FloatingIpNotFoundForAddress,
|
|
self.api.release_floating_ip,
|
|
self.context, address)
|
|
|
|
@mock.patch.object(client.Client, 'create_port')
|
|
def test_create_port_minimal_raise_no_more_ip(self, create_port_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
create_port_mock.side_effect = \
|
|
exceptions.IpAddressGenerationFailureClient()
|
|
self.assertRaises(
|
|
exception.NoMoreFixedIps,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
neutronapi.get_client(self.context),
|
|
instance, uuids.my_netid1
|
|
)
|
|
self.assertTrue(create_port_mock.called)
|
|
|
|
@mock.patch.object(client.Client, 'update_port',
|
|
side_effect=exceptions.MacAddressInUseClient())
|
|
def test_update_port_for_instance_mac_address_in_use(self,
|
|
update_port_mock):
|
|
port_uuid = uuids.port
|
|
instance = objects.Instance(uuid=uuids.instance)
|
|
port_req_body = {'port': {
|
|
'id': port_uuid,
|
|
'mac_address': 'XX:XX:XX:XX:XX:XX',
|
|
'network_id': uuids.network_id}}
|
|
|
|
self.assertRaises(exception.PortInUse,
|
|
self.api._update_port,
|
|
neutronapi.get_client(self.context),
|
|
instance, port_uuid, port_req_body)
|
|
update_port_mock.assert_called_once_with(port_uuid, port_req_body)
|
|
|
|
@mock.patch.object(client.Client, 'update_port',
|
|
side_effect=exceptions.HostNotCompatibleWithFixedIpsClient())
|
|
def test_update_port_for_instance_fixed_ips_invalid(self,
|
|
update_port_mock):
|
|
port_uuid = uuids.port
|
|
instance = objects.Instance(uuid=uuids.instance)
|
|
port_req_body = {'port': {
|
|
'id': port_uuid,
|
|
'mac_address': 'XX:XX:XX:XX:XX:XX',
|
|
'network_id': uuids.network_id}}
|
|
|
|
self.assertRaises(exception.FixedIpInvalidOnHost,
|
|
self.api._update_port,
|
|
neutronapi.get_client(self.context),
|
|
instance, port_uuid, port_req_body)
|
|
update_port_mock.assert_called_once_with(port_uuid, port_req_body)
|
|
|
|
@mock.patch.object(client.Client, 'update_port')
|
|
def test_update_port_for_instance_binding_failure(self,
|
|
update_port_mock):
|
|
port_uuid = uuids.port
|
|
instance = objects.Instance(uuid=uuids.instance)
|
|
port_req_body = {'port': {
|
|
'id': port_uuid,
|
|
'mac_address': 'XX:XX:XX:XX:XX:XX',
|
|
'network_id': uuids.network_id}}
|
|
update_port_mock.return_value = {'port': {
|
|
'id': port_uuid,
|
|
'binding:vif_type': model.VIF_TYPE_BINDING_FAILED
|
|
}}
|
|
|
|
self.assertRaises(exception.PortBindingFailed,
|
|
self.api._update_port,
|
|
neutronapi.get_client(self.context),
|
|
instance, port_uuid, port_req_body)
|
|
|
|
@mock.patch.object(client.Client, 'create_port',
|
|
side_effect=exceptions.IpAddressInUseClient())
|
|
def test_create_port_minimal_raise_ip_in_use(self, create_port_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ip = '1.1.1.1'
|
|
|
|
self.assertRaises(
|
|
exception.FixedIpAlreadyInUse,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
neutronapi.get_client(self.context),
|
|
instance, uuids.my_netid1, fixed_ip=fake_ip
|
|
)
|
|
self.assertTrue(create_port_mock.called)
|
|
|
|
@mock.patch.object(client.Client, 'create_port',
|
|
side_effect=exceptions.IpAddressAlreadyAllocatedClient())
|
|
def test_create_port_minimal_raise_ip_already_allocated(self,
|
|
create_port_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ip = '1.1.1.1'
|
|
|
|
self.assertRaises(
|
|
exception.FixedIpAlreadyInUse,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
neutronapi.get_client(self.context),
|
|
instance, uuids.my_netid1, fixed_ip=fake_ip
|
|
)
|
|
self.assertTrue(create_port_mock.called)
|
|
|
|
@mock.patch.object(client.Client, 'create_port',
|
|
side_effect=exceptions.InvalidIpForNetworkClient())
|
|
def test_create_port_minimal_raise_invalid_ip(self, create_port_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ip = '1.1.1.1'
|
|
|
|
exc = self.assertRaises(
|
|
exception.InvalidInput,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
neutronapi.get_client(self.context),
|
|
instance, uuids.my_netid1, fixed_ip=fake_ip
|
|
)
|
|
|
|
expected_exception_msg = ('Invalid input received: Fixed IP %(ip)s is '
|
|
'not a valid ip address for network '
|
|
'%(net_id)s.' %
|
|
{'ip': fake_ip, 'net_id': uuids.my_netid1})
|
|
self.assertEqual(expected_exception_msg, str(exc))
|
|
self.assertTrue(create_port_mock.called)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
def test_create_port_minimal_raise_qos_not_supported(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
mock_client = mock.MagicMock()
|
|
mock_client.create_port.return_value = {'port': {
|
|
'id': uuids.port_id,
|
|
constants.RESOURCE_REQUEST: {
|
|
'resources': {'CUSTOM_RESOURCE_CLASS': 42}}
|
|
}}
|
|
|
|
exc = self.assertRaises(
|
|
exception.NetworksWithQoSPolicyNotSupported,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
mock_client, instance, uuids.my_netid1
|
|
)
|
|
expected_exception_msg = ('Using networks with QoS policy is not '
|
|
'supported for instance %(instance)s. '
|
|
'(Network ID is %(net_id)s)' %
|
|
{'instance': instance.uuid,
|
|
'net_id': uuids.my_netid1})
|
|
self.assertEqual(expected_exception_msg, str(exc))
|
|
mock_client.delete_port.assert_called_once_with(uuids.port_id)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
def test_create_port_minimal_raise_extended_qos_not_supported(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
mock_client = mock.MagicMock()
|
|
mock_client.create_port.return_value = {
|
|
'port': {
|
|
'id': uuids.port_id,
|
|
constants.RESOURCE_REQUEST: {
|
|
'request_groups': [
|
|
{
|
|
"id": uuids.group1,
|
|
'resources': {'CUSTOM_RESOURCE_CLASS': 42},
|
|
}
|
|
],
|
|
}
|
|
}
|
|
}
|
|
|
|
exc = self.assertRaises(
|
|
exception.NetworksWithQoSPolicyNotSupported,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
mock_client, instance, uuids.my_netid1
|
|
)
|
|
expected_exception_msg = ('Using networks with QoS policy is not '
|
|
'supported for instance %(instance)s. '
|
|
'(Network ID is %(net_id)s)' %
|
|
{'instance': instance.uuid,
|
|
'net_id': uuids.my_netid1})
|
|
self.assertEqual(expected_exception_msg, str(exc))
|
|
mock_client.delete_port.assert_called_once_with(uuids.port_id)
|
|
|
|
@mock.patch('nova.network.neutron.LOG')
|
|
def test_create_port_minimal_raise_qos_not_supported_cleanup_fails(
|
|
self, mock_log):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
mock_client = mock.MagicMock()
|
|
mock_client.create_port.return_value = {'port': {
|
|
'id': uuids.port_id,
|
|
constants.RESOURCE_REQUEST: {
|
|
'resources': {'CUSTOM_RESOURCE_CLASS': 42}}
|
|
}}
|
|
mock_client.delete_port.side_effect = \
|
|
exceptions.NeutronClientException()
|
|
|
|
exc = self.assertRaises(
|
|
exception.NetworksWithQoSPolicyNotSupported,
|
|
self.api._create_port_minimal,
|
|
self.context,
|
|
mock_client, instance, uuids.my_netid1
|
|
)
|
|
expected_exception_msg = ('Using networks with QoS policy is not '
|
|
'supported for instance %(instance)s. '
|
|
'(Network ID is %(net_id)s)' %
|
|
{'instance': instance.uuid,
|
|
'net_id': uuids.my_netid1})
|
|
self.assertEqual(expected_exception_msg, str(exc))
|
|
mock_client.delete_port.assert_called_once_with(uuids.port_id)
|
|
self.assertTrue(mock_log.exception.called)
|
|
|
|
def test_get_network_detail_not_found(self):
|
|
api = neutronapi.API()
|
|
expected_exc = exceptions.NetworkNotFoundClient()
|
|
network_uuid = '02cacbca-7d48-4a2c-8011-43eecf8a9786'
|
|
with mock.patch.object(client.Client, 'show_network',
|
|
side_effect=expected_exc) as (
|
|
fake_show_network):
|
|
self.assertRaises(exception.NetworkNotFound,
|
|
api.get,
|
|
self.context,
|
|
network_uuid)
|
|
fake_show_network.assert_called_once_with(network_uuid)
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
@mock.patch('nova.network.neutron.API.'
|
|
'_refresh_neutron_extensions_cache')
|
|
def test_deallocate_for_instance_uses_delete_helper(self,
|
|
mock_refresh,
|
|
mock_preexisting):
|
|
# setup fake data
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
mock_preexisting.return_value = []
|
|
port_data = {'ports': [{'id': uuids.fake}]}
|
|
ports = set([port['id'] for port in port_data.get('ports')])
|
|
api = neutronapi.API()
|
|
# setup mocks
|
|
mock_client = mock.Mock()
|
|
mock_client.list_ports.return_value = port_data
|
|
with test.nested(
|
|
mock.patch.object(neutronapi, 'get_client',
|
|
return_value=mock_client),
|
|
mock.patch.object(api, '_delete_ports')
|
|
) as (
|
|
mock_get_client, mock_delete
|
|
):
|
|
# run the code
|
|
api.deallocate_for_instance(self.context, instance)
|
|
# assert the calls
|
|
mock_client.list_ports.assert_called_once_with(
|
|
device_id=instance.uuid)
|
|
mock_delete.assert_called_once_with(
|
|
mock_client, instance, ports, raise_if_fail=True)
|
|
|
|
def _test_delete_ports(self, expect_raise):
|
|
results = [exceptions.NeutronClientException, None]
|
|
mock_client = mock.Mock()
|
|
with mock.patch.object(mock_client, 'delete_port',
|
|
side_effect=results):
|
|
api = neutronapi.API()
|
|
api._delete_ports(mock_client, {'uuid': 'foo'}, ['port1', 'port2'],
|
|
raise_if_fail=expect_raise)
|
|
|
|
def test_delete_ports_raise(self):
|
|
self.assertRaises(exceptions.NeutronClientException,
|
|
self._test_delete_ports, True)
|
|
|
|
def test_delete_ports_no_raise(self):
|
|
self._test_delete_ports(False)
|
|
|
|
def test_delete_ports_never_raise_404(self):
|
|
mock_client = mock.Mock()
|
|
mock_client.delete_port.side_effect = exceptions.PortNotFoundClient
|
|
api = neutronapi.API()
|
|
api._delete_ports(mock_client, {'uuid': 'foo'}, ['port1'],
|
|
raise_if_fail=True)
|
|
mock_client.delete_port.assert_called_once_with('port1')
|
|
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
def test_deallocate_port_for_instance_fails(self, mock_preexisting):
|
|
mock_preexisting.return_value = []
|
|
mock_client = mock.Mock()
|
|
mock_client.show_port.side_effect = exceptions.Unauthorized()
|
|
api = neutronapi.API()
|
|
with test.nested(
|
|
mock.patch.object(neutronapi, 'get_client',
|
|
return_value=mock_client),
|
|
mock.patch.object(api, 'get_instance_nw_info')
|
|
) as (
|
|
get_client, get_nw_info
|
|
):
|
|
self.assertRaises(exceptions.Unauthorized,
|
|
api.deallocate_port_for_instance,
|
|
self.context, instance={'uuid': uuids.fake},
|
|
port_id=uuids.fake)
|
|
# make sure that we didn't try to reload nw info
|
|
self.assertFalse(get_nw_info.called)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def _test_show_port_exceptions(self, client_exc, expected_nova_exc,
|
|
get_client_mock):
|
|
show_port_mock = mock.Mock(side_effect=client_exc)
|
|
get_client_mock.return_value.show_port = show_port_mock
|
|
self.assertRaises(expected_nova_exc, self.api.show_port,
|
|
self.context, 'fake_port_id')
|
|
|
|
def test_show_port_not_found(self):
|
|
self._test_show_port_exceptions(exceptions.PortNotFoundClient,
|
|
exception.PortNotFound)
|
|
|
|
def test_show_port_forbidden(self):
|
|
self._test_show_port_exceptions(exceptions.Unauthorized,
|
|
exception.Forbidden)
|
|
|
|
def test_show_port_unknown_exception(self):
|
|
self._test_show_port_exceptions(exceptions.NeutronClientException,
|
|
exception.NovaException)
|
|
|
|
def test_get_network(self):
|
|
api = neutronapi.API()
|
|
fake_network = {
|
|
'network': {'id': uuids.instance, 'name': 'fake-network'}
|
|
}
|
|
|
|
with mock.patch.object(client.Client, 'show_network') as mock_show:
|
|
mock_show.return_value = fake_network
|
|
rsp = api.get(self.context, uuids.instance)
|
|
self.assertEqual(fake_network['network'], rsp)
|
|
|
|
def test_get_all_networks(self):
|
|
api = neutronapi.API()
|
|
fake_networks = {
|
|
'networks': [
|
|
{'id': uuids.network_1, 'name': 'fake-network1'},
|
|
{'id': uuids.network_2, 'name': 'fake-network2'},
|
|
],
|
|
}
|
|
with mock.patch.object(client.Client, 'list_networks') as mock_list:
|
|
mock_list.return_value = fake_networks
|
|
rsp = api.get_all(self.context)
|
|
self.assertEqual(fake_networks['networks'], rsp)
|
|
|
|
@mock.patch.object(neutronapi.API, "_refresh_neutron_extensions_cache")
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_instance_vnic_index(self, mock_get_client,
|
|
mock_refresh_extensions):
|
|
api = neutronapi.API()
|
|
api.extensions = set([constants.VNIC_INDEX])
|
|
mock_client = mock_get_client.return_value
|
|
mock_client.update_port.return_value = 'port'
|
|
|
|
instance = {'project_id': '9d049e4b60b64716978ab415e6fbd5c0',
|
|
'uuid': uuids.fake,
|
|
'display_name': 'test_instance',
|
|
'availability_zone': 'nova',
|
|
'host': 'some_host'}
|
|
instance = objects.Instance(**instance)
|
|
vif = {'id': 'fake-port-id'}
|
|
api.update_instance_vnic_index(self.context, instance, vif, 7)
|
|
port_req_body = {'port': {'vnic_index': 7}}
|
|
mock_client.update_port.assert_called_once_with('fake-port-id',
|
|
port_req_body)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_migration_profile(
|
|
self, get_client_mock
|
|
):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
|
|
# We pass in a port profile which has a migration attribute and also
|
|
# a second port profile attribute 'fake_profile' this can be
|
|
# an sriov port profile attribute or a pci_slot attribute, but for
|
|
# now we are just using a fake one to show that the code does not
|
|
# remove the portbinding_profile if there is one.
|
|
binding_profile = {'fake_profile': 'fake_data',
|
|
constants.MIGRATING_ATTR: 'my-dest-host'}
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
constants.BINDING_PROFILE: binding_profile,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
'my-host')
|
|
# Assert that update_port was called on the port with a
|
|
# different host and also the migration profile from the port is
|
|
# removed since it does not match with the current host.
|
|
update_port_mock.assert_called_once_with(
|
|
'fake-port-1', {'port': {
|
|
constants.BINDING_HOST_ID: 'my-host',
|
|
'device_owner':
|
|
'compute:%s' %
|
|
instance.availability_zone,
|
|
constants.BINDING_PROFILE: {
|
|
'fake_profile': 'fake_data'}}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_binding_profile_none(
|
|
self, get_client_mock
|
|
):
|
|
"""Tests _update_port_binding_for_instance when the binding:profile
|
|
value is None.
|
|
"""
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
|
|
fake_ports = {'ports': [
|
|
{'id': uuids.portid,
|
|
constants.BINDING_PROFILE: None,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
'my-host')
|
|
# Assert that update_port was called on the port with a
|
|
# different host but with no binding profile.
|
|
update_port_mock.assert_called_once_with(
|
|
uuids.portid, {'port': {
|
|
constants.BINDING_HOST_ID: 'my-host',
|
|
'device_owner':
|
|
'compute:%s' %
|
|
instance.availability_zone}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_same_host(
|
|
self, get_client_mock
|
|
):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
|
|
# We test two ports, one with the same host as the host passed in and
|
|
# one where binding:host_id isn't set, so we update that port.
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
constants.BINDING_HOST_ID: instance.host},
|
|
{'id': 'fake-port-2'}]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
instance.host)
|
|
# Assert that update_port was only called on the port without a host.
|
|
update_port_mock.assert_called_once_with(
|
|
'fake-port-2',
|
|
{'port': {constants.BINDING_HOST_ID: instance.host,
|
|
'device_owner': 'compute:%s' %
|
|
instance.availability_zone}})
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_vf_pci_device_profile',
|
|
new=mock.Mock(return_value={}))
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_pci(
|
|
self, get_client_mock, get_pci_device_devspec_mock):
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_pci_device_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
instance.migration_context.old_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(
|
|
vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0a:00.1',
|
|
compute_node_id=1,
|
|
request_id='1234567890',
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_VF)])
|
|
instance.migration_context.new_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(
|
|
vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0b:00.1',
|
|
compute_node_id=2,
|
|
request_id='1234567890',
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_VF)])
|
|
instance.pci_devices = instance.migration_context.old_pci_devices
|
|
|
|
# Validate that non-direct port aren't updated (fake-port-2).
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'direct',
|
|
constants.BINDING_HOST_ID: 'fake-host-old',
|
|
constants.BINDING_PROFILE:
|
|
{'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'old_phys_net',
|
|
'pci_vendor_info': 'old_pci_vendor_info'}},
|
|
{'id': 'fake-port-2',
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
instance.host, migration)
|
|
# Assert that update_port is called with the binding:profile
|
|
# corresponding to the PCI device specified.
|
|
update_port_mock.assert_called_once_with(
|
|
'fake-port-1',
|
|
{'port':
|
|
{
|
|
constants.BINDING_HOST_ID: 'fake-host',
|
|
'device_owner': 'compute:%s' % instance.availability_zone,
|
|
constants.BINDING_PROFILE:
|
|
{'pci_slot': '0000:0b:00.1',
|
|
'physical_network': 'physnet1',
|
|
'pci_vendor_info': '1377:0047'}}})
|
|
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_pci_fail(self,
|
|
get_client_mock,
|
|
get_pci_device_devspec_mock):
|
|
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_pci_device_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
instance.migration_context.old_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0c:00.1',
|
|
compute_node_id=1,
|
|
request_id='1234567890')])
|
|
instance.migration_context.new_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0d:00.1',
|
|
compute_node_id=2,
|
|
request_id='1234567890')])
|
|
instance.pci_devices = instance.migration_context.old_pci_devices
|
|
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'direct',
|
|
constants.BINDING_HOST_ID: 'fake-host-old',
|
|
constants.BINDING_PROFILE:
|
|
{'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'old_phys_net',
|
|
'pci_vendor_info': 'old_pci_vendor_info'}}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
# Assert exception is raised if the mapping is wrong.
|
|
self.assertRaises(exception.PortUpdateFailed,
|
|
self.api._update_port_binding_for_instance,
|
|
self.context,
|
|
instance,
|
|
instance.host,
|
|
migration)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_pci_no_migration(self,
|
|
get_client_mock,
|
|
get_pci_device_devspec_mock):
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_pci_device_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
instance.migration_context.old_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0a:00.1',
|
|
compute_node_id=1,
|
|
request_id='1234567890')])
|
|
instance.migration_context.new_pci_devices = objects.PciDeviceList(
|
|
objects=[objects.PciDevice(vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0b:00.1',
|
|
compute_node_id=2,
|
|
request_id='1234567890')])
|
|
instance.pci_devices = instance.migration_context.old_pci_devices
|
|
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'direct',
|
|
constants.BINDING_HOST_ID: instance.host,
|
|
constants.BINDING_PROFILE:
|
|
{'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'phys_net',
|
|
'pci_vendor_info': 'pci_vendor_info'}}]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
# Try to update the port binding with no migration object.
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
instance.host)
|
|
# No ports should be updated if the port's pci binding did not change.
|
|
update_port_mock.assert_not_called()
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_same_host_failed_vif_type(
|
|
self, get_client_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
list_ports_mock = mock.Mock()
|
|
update_port_mock = mock.Mock()
|
|
|
|
FAILED_VIF_TYPES = (model.VIF_TYPE_UNBOUND,
|
|
model.VIF_TYPE_BINDING_FAILED)
|
|
for vif_type in FAILED_VIF_TYPES:
|
|
binding_profile = {'fake_profile': 'fake_data',
|
|
constants.MIGRATING_ATTR: 'my-dest-host'}
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vif_type': 'fake-vif-type',
|
|
constants.BINDING_PROFILE: binding_profile,
|
|
constants.BINDING_HOST_ID: instance.host},
|
|
{'id': 'fake-port-2',
|
|
'binding:vif_type': vif_type,
|
|
constants.BINDING_PROFILE: binding_profile,
|
|
constants.BINDING_HOST_ID: instance.host}
|
|
]}
|
|
|
|
list_ports_mock.return_value = fake_ports
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
update_port_mock.reset_mock()
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
instance.host)
|
|
# Assert that update_port was called on the port with a
|
|
# failed vif_type and MIGRATING_ATTR is removed
|
|
update_port_mock.assert_called_once_with(
|
|
'fake-port-2',
|
|
{'port': {constants.BINDING_HOST_ID: instance.host,
|
|
constants.BINDING_PROFILE: {
|
|
'fake_profile': 'fake_data'},
|
|
'device_owner': 'compute:%s' %
|
|
instance.availability_zone
|
|
}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_diff_host_unbound_vif_type(
|
|
self, get_client_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
|
|
binding_profile = {'fake_profile': 'fake_data',
|
|
constants.MIGRATING_ATTR: 'my-dest-host'}
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vif_type': model.VIF_TYPE_UNBOUND,
|
|
constants.BINDING_PROFILE: binding_profile,
|
|
constants.BINDING_HOST_ID: instance.host},
|
|
]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
'my-host')
|
|
|
|
# Assert that update_port was called on the port with a
|
|
# 'unbound' vif_type, host updated and MIGRATING_ATTR is removed
|
|
update_port_mock.assert_called_once_with(
|
|
'fake-port-1', {'port': {
|
|
constants.BINDING_HOST_ID: 'my-host',
|
|
constants.BINDING_PROFILE: {
|
|
'fake_profile': 'fake_data'},
|
|
'device_owner': 'compute:%s' %
|
|
instance.availability_zone
|
|
}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi.API, '_get_pci_mapping_for_migration')
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_live_migration(
|
|
self,
|
|
get_client_mock,
|
|
get_devspec_mock,
|
|
get_pci_mapping_mock):
|
|
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'direct',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE:
|
|
{'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'phys_net',
|
|
'pci_vendor_info': 'vendor_info'}}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='live-migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(self.context, instance,
|
|
'new-host', migration)
|
|
# Assert _get_pci_mapping_for_migration was not called
|
|
self.assertFalse(get_pci_mapping_mock.called)
|
|
|
|
# Assert that update_port() does not update binding:profile
|
|
# and that it updates host ID
|
|
called_port_id = update_port_mock.call_args[0][0]
|
|
called_port_attributes = update_port_mock.call_args[0][1]
|
|
self.assertEqual(called_port_id, fake_ports['ports'][0]['id'])
|
|
self.assertNotIn(
|
|
constants.BINDING_PROFILE, called_port_attributes['port'])
|
|
self.assertEqual(
|
|
called_port_attributes['port'][
|
|
constants.BINDING_HOST_ID],
|
|
'new-host')
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_sriov_pf(
|
|
self, get_client_mock, get_pci_device_devspec_mock
|
|
):
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_pci_device_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
instance.migration_context.old_pci_devices = objects.PciDeviceList(
|
|
objects=[
|
|
objects.PciDevice(
|
|
vendor_id='8086',
|
|
product_id='154d',
|
|
address='0000:0a:01',
|
|
compute_node_id=1,
|
|
request_id=uuids.pci_req,
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_PF,
|
|
extra_info={'mac_address': 'b4:96:91:34:f4:36'},
|
|
)
|
|
]
|
|
)
|
|
instance.pci_devices = instance.migration_context.old_pci_devices
|
|
instance.migration_context.new_pci_devices = objects.PciDeviceList(
|
|
objects=[
|
|
objects.PciDevice(
|
|
vendor_id='8086',
|
|
product_id='154d',
|
|
address='0000:0a:02',
|
|
compute_node_id=2,
|
|
request_id=uuids.pci_req,
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_PF,
|
|
extra_info={'mac_address': 'b4:96:91:34:f4:dd'},
|
|
)
|
|
]
|
|
)
|
|
instance.pci_devices = instance.migration_context.new_pci_devices
|
|
|
|
fake_ports = {
|
|
'ports': [
|
|
{
|
|
'id': uuids.port,
|
|
'binding:vnic_type': 'direct-physical',
|
|
constants.BINDING_HOST_ID: 'fake-host-old',
|
|
constants.BINDING_PROFILE: {
|
|
'pci_slot': '0000:0a:01',
|
|
'physical_network': 'old_phys_net',
|
|
'pci_vendor_info': 'old_pci_vendor_info',
|
|
},
|
|
},
|
|
]
|
|
}
|
|
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, instance.host, migration)
|
|
|
|
# Assert that update_port is called with the binding:profile
|
|
# corresponding to the PCI device specified including MAC address.
|
|
update_port_mock.assert_called_once_with(
|
|
uuids.port,
|
|
{
|
|
'port': {
|
|
constants.BINDING_HOST_ID: 'fake-host',
|
|
'device_owner': 'compute:%s' % instance.availability_zone,
|
|
constants.BINDING_PROFILE: {
|
|
'pci_slot': '0000:0a:02',
|
|
'physical_network': 'physnet1',
|
|
'pci_vendor_info': '8086:154d',
|
|
'device_mac_address': 'b4:96:91:34:f4:dd',
|
|
},
|
|
}
|
|
},
|
|
)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_sriov_pf_no_migration(
|
|
self, get_client_mock, get_pci_device_devspec_mock
|
|
):
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
get_pci_device_devspec_mock.return_value = devspec
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.pci_requests = objects.InstancePCIRequests(
|
|
instance_uuid=instance.uuid,
|
|
requests=[
|
|
objects.InstancePCIRequest(
|
|
requester_id=uuids.port,
|
|
request_id=uuids.pci_req,
|
|
)
|
|
],
|
|
)
|
|
instance.pci_devices = objects.PciDeviceList(
|
|
objects=[
|
|
objects.PciDevice(
|
|
vendor_id='8086',
|
|
product_id='154d',
|
|
address='0000:0a:02',
|
|
compute_node_id=2,
|
|
request_id=uuids.pci_req,
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_PF,
|
|
extra_info={'mac_address': 'b4:96:91:34:f4:36'},
|
|
)
|
|
]
|
|
)
|
|
|
|
fake_ports = {
|
|
'ports': [
|
|
{
|
|
'id': uuids.port,
|
|
'binding:vnic_type': 'direct-physical',
|
|
constants.BINDING_HOST_ID: 'fake-host-old',
|
|
constants.BINDING_PROFILE: {
|
|
'pci_slot': '0000:0a:01',
|
|
'physical_network': 'old_phys_net',
|
|
'pci_vendor_info': 'old_pci_vendor_info',
|
|
'device_mac_address': 'b4:96:91:34:f4:dd'
|
|
},
|
|
},
|
|
]
|
|
}
|
|
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, instance.host)
|
|
|
|
# Assert that update_port is called with the binding:profile
|
|
# corresponding to the PCI device specified including MAC address.
|
|
update_port_mock.assert_called_once_with(
|
|
uuids.port,
|
|
{
|
|
'port': {
|
|
constants.BINDING_HOST_ID: 'fake-host',
|
|
'device_owner': 'compute:%s' % instance.availability_zone,
|
|
constants.BINDING_PROFILE: {
|
|
'pci_slot': '0000:0a:02',
|
|
'physical_network': 'physnet1',
|
|
'pci_vendor_info': '8086:154d',
|
|
'device_mac_address': 'b4:96:91:34:f4:36',
|
|
},
|
|
}
|
|
},
|
|
)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_resource_req(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'normal',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE:
|
|
{'allocation': uuids.source_compute_rp},
|
|
'resource_request': mock.sentinel.resource_request}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, 'new-host', migration,
|
|
{'fake-port-1': [uuids.dest_compute_rp]})
|
|
get_client_mock.return_value.update_port.assert_called_once_with(
|
|
'fake-port-1',
|
|
{'port': {'device_owner': 'compute:None',
|
|
'binding:profile': {'allocation': uuids.dest_compute_rp},
|
|
'binding:host_id': 'new-host'}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_extended_resource_req(
|
|
self, get_client_mock
|
|
):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {
|
|
'ports': [
|
|
{
|
|
'id': 'fake-port-1',
|
|
'binding:vnic_type': 'normal',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE: {
|
|
'allocation': {
|
|
uuids.group1: uuids.source_compute_bw_rp,
|
|
uuids.group2: uuids.source_compute_pps_rp,
|
|
},
|
|
},
|
|
'resource_request': {
|
|
'request_groups': [
|
|
{
|
|
"id": uuids.group1
|
|
},
|
|
{
|
|
"id": uuids.group2
|
|
},
|
|
],
|
|
},
|
|
}
|
|
]
|
|
}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, 'new-host', migration,
|
|
{
|
|
uuids.group1: [uuids.dest_compute_bw_rp],
|
|
uuids.group2: [uuids.dest_compute_pps_rp],
|
|
}
|
|
)
|
|
get_client_mock.return_value.update_port.assert_called_once_with(
|
|
'fake-port-1',
|
|
{
|
|
'port': {
|
|
'device_owner': 'compute:None',
|
|
'binding:profile': {
|
|
'allocation': {
|
|
uuids.group1: uuids.dest_compute_bw_rp,
|
|
uuids.group2: uuids.dest_compute_pps_rp,
|
|
}
|
|
},
|
|
'binding:host_id': 'new-host'
|
|
}
|
|
}
|
|
)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_update_port_bindings_for_instance_with_resource_req_unshelve(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'normal',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE: {
|
|
'allocation': uuids.source_compute_rp},
|
|
'resource_request': mock.sentinel.resource_request}]}
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
# NOTE(gibi): during unshelve migration object is not created
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, 'new-host', None,
|
|
{'fake-port-1': [uuids.dest_compute_rp]})
|
|
get_client_mock.return_value.update_port.assert_called_once_with(
|
|
'fake-port-1',
|
|
{'port': {'device_owner': 'compute:None',
|
|
'binding:profile': {'allocation': uuids.dest_compute_rp},
|
|
'binding:host_id': 'new-host'}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_resource_req_no_mapping(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'normal',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE:
|
|
{'allocation': uuids.source_compute_rp},
|
|
'resource_request': mock.sentinel.resource_request}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
ex = self.assertRaises(
|
|
exception.PortUpdateFailed,
|
|
self.api._update_port_binding_for_instance, self.context,
|
|
instance, 'new-host', migration, provider_mappings=None)
|
|
self.assertIn(
|
|
"Provider mappings are not available to the compute service but "
|
|
"are required for ports with a resource request.",
|
|
str(ex))
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_bindings_for_instance_with_resource_req_live_mig(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
fake_ports = {'ports': [
|
|
{'id': 'fake-port-1',
|
|
'binding:vnic_type': 'normal',
|
|
constants.BINDING_HOST_ID: 'old-host',
|
|
constants.BINDING_PROFILE:
|
|
{'allocation': uuids.dest_compute_rp},
|
|
'resource_request': mock.sentinel.resource_request}]}
|
|
migration = objects.Migration(
|
|
status='confirmed', migration_type='live-migration')
|
|
list_ports_mock = mock.Mock(return_value=fake_ports)
|
|
get_client_mock.return_value.list_ports = list_ports_mock
|
|
|
|
# No mapping is passed in as during live migration the conductor
|
|
# already created the binding and added the allocation key
|
|
self.api._update_port_binding_for_instance(
|
|
self.context, instance, 'new-host', migration, {})
|
|
|
|
# Note that binding:profile is not updated
|
|
get_client_mock.return_value.update_port.assert_called_once_with(
|
|
'fake-port-1',
|
|
{'port': {'device_owner': 'compute:None',
|
|
'binding:host_id': 'new-host'}})
|
|
|
|
def test_get_pci_mapping_for_migration(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
migration = objects.Migration(status='confirmed')
|
|
|
|
with mock.patch.object(instance.migration_context,
|
|
'get_pci_mapping_for_migration') as map_func:
|
|
self.api._get_pci_mapping_for_migration(instance, migration)
|
|
map_func.assert_called_with(False)
|
|
|
|
def test_get_pci_mapping_for_migration_reverted(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = objects.MigrationContext()
|
|
migration = objects.Migration(status='reverted')
|
|
|
|
with mock.patch.object(instance.migration_context,
|
|
'get_pci_mapping_for_migration') as map_func:
|
|
self.api._get_pci_mapping_for_migration(instance, migration)
|
|
map_func.assert_called_with(True)
|
|
|
|
def test_get_pci_mapping_for_migration_no_migration_context(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.migration_context = None
|
|
pci_mapping = self.api._get_pci_mapping_for_migration(
|
|
instance, None)
|
|
self.assertDictEqual({}, pci_mapping)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_profile_for_migration_teardown_false(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
# We test with an instance host and destination_host where the
|
|
# port will be moving.
|
|
get_ports = {'ports': [
|
|
{'id': uuids.port_id,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
migrate_profile = {
|
|
constants.MIGRATING_ATTR: 'my-new-host'}
|
|
port_data = {'port':
|
|
{
|
|
constants.BINDING_PROFILE: migrate_profile}}
|
|
|
|
self.api.setup_networks_on_host(self.context,
|
|
instance,
|
|
host='my-new-host',
|
|
teardown=False)
|
|
update_port_mock.assert_called_once_with(
|
|
uuids.port_id,
|
|
port_data)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_profile_for_migration_teardown_false_none_profile(
|
|
self, get_client_mock):
|
|
"""Tests setup_networks_on_host when migrating the port to the
|
|
destination host and the binding:profile is None in the port.
|
|
"""
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
# We test with an instance host and destination_host where the
|
|
# port will be moving but with binding:profile set to None.
|
|
get_ports = {
|
|
'ports': [
|
|
{'id': uuids.port_id,
|
|
constants.BINDING_HOST_ID: instance.host,
|
|
constants.BINDING_PROFILE: None}
|
|
]
|
|
}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
migrate_profile = {
|
|
constants.MIGRATING_ATTR: 'my-new-host'}
|
|
port_data = {
|
|
'port': {
|
|
constants.BINDING_PROFILE: migrate_profile
|
|
}
|
|
}
|
|
self.api.setup_networks_on_host(
|
|
self.context, instance, host='my-new-host', teardown=False)
|
|
update_port_mock.assert_called_once_with(
|
|
uuids.port_id,
|
|
port_data)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test__setup_migration_port_profile_called_on_teardown_false(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
port_id = uuids.port_id
|
|
get_ports = {'ports': [
|
|
{'id': port_id,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
self.api._setup_migration_port_profile = mock.Mock()
|
|
self.api.setup_networks_on_host(self.context,
|
|
instance,
|
|
host='my-new-host',
|
|
teardown=False)
|
|
self.api._setup_migration_port_profile.assert_called_once_with(
|
|
self.context, instance, 'my-new-host',
|
|
mock.ANY, get_ports['ports'])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test__setup_migration_port_profile_not_called_with_host_match(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
get_ports = {'ports': [
|
|
{'id': uuids.port_id,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
self.api._setup_migration_port_profile = mock.Mock()
|
|
self.api._clear_migration_port_profile = mock.Mock()
|
|
self.api.setup_networks_on_host(self.context,
|
|
instance,
|
|
host=instance.host,
|
|
teardown=False)
|
|
self.api._setup_migration_port_profile.assert_not_called()
|
|
self.api._clear_migration_port_profile.assert_not_called()
|
|
|
|
def test__setup_migration_port_profile_no_update(self):
|
|
"""Tests the case that the port binding profile already has the
|
|
"migrating_to" attribute set to the provided host so the port update
|
|
call is skipped.
|
|
"""
|
|
ports = [{
|
|
constants.BINDING_HOST_ID: 'source-host',
|
|
constants.BINDING_PROFILE: {
|
|
constants.MIGRATING_ATTR: 'dest-host'
|
|
}
|
|
}] * 2
|
|
with mock.patch.object(self.api, '_update_port_with_migration_profile',
|
|
new_callable=mock.NonCallableMock):
|
|
self.api._setup_migration_port_profile(
|
|
self.context, mock.sentinel.instance, 'dest-host',
|
|
mock.sentinel.admin_client, ports)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_profile_for_migration_teardown_true_with_profile(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
migrate_profile = {
|
|
constants.MIGRATING_ATTR: 'new-host'}
|
|
# Pass a port with an migration profile attribute.
|
|
port_id = uuids.port_id
|
|
get_ports = {'ports': [
|
|
{'id': port_id,
|
|
constants.BINDING_PROFILE: migrate_profile,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
mocked_client = get_client_mock.return_value
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
self.api.setup_networks_on_host(self.context,
|
|
instance,
|
|
host='new-host',
|
|
teardown=True)
|
|
|
|
mocked_client.update_port.assert_called_once_with(
|
|
port_id, {'port': {constants.BINDING_PROFILE: migrate_profile}})
|
|
mocked_client.delete_port_binding.assert_called_once_with(
|
|
port_id, 'new-host')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_update_port_profile_for_migration_teardown_true_with_profile_exc(
|
|
self, get_client_mock):
|
|
"""Tests that delete_port_binding raises PortBindingDeletionFailed
|
|
which is raised through to the caller.
|
|
"""
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
migrate_profile = {
|
|
constants.MIGRATING_ATTR: 'new-host'}
|
|
# Pass a port with an migration profile attribute.
|
|
get_ports = {
|
|
'ports': [
|
|
{'id': uuids.port1,
|
|
constants.BINDING_PROFILE: migrate_profile,
|
|
constants.BINDING_HOST_ID: instance.host},
|
|
{'id': uuids.port2,
|
|
constants.BINDING_PROFILE: migrate_profile,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
self.api._clear_migration_port_profile = mock.Mock()
|
|
NeutronError = exceptions.NeutronClientException(status_code=500)
|
|
mocked_client = get_client_mock.return_value
|
|
mocked_client.delete_port_binding.side_effect = NeutronError
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
ex = self.assertRaises(
|
|
exception.PortBindingDeletionFailed,
|
|
self.api.setup_networks_on_host,
|
|
self.context, instance, host='new-host', teardown=True)
|
|
# Make sure both ports show up in the exception message.
|
|
self.assertIn(uuids.port1, str(ex))
|
|
self.assertIn(uuids.port2, str(ex))
|
|
|
|
self.api._clear_migration_port_profile.assert_called_once_with(
|
|
self.context, instance, get_client_mock.return_value,
|
|
get_ports['ports'])
|
|
mocked_client.delete_port_binding.assert_has_calls([
|
|
mock.call(uuids.port1, 'new-host'),
|
|
mock.call(uuids.port2, 'new-host'),
|
|
])
|
|
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_update_port_profile_for_migration_teardown_true_no_profile(
|
|
self, get_client_mock):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
self.api.has_port_binding_extension = mock.Mock(return_value=True)
|
|
# Pass a port without any migration profile attribute.
|
|
get_ports = {'ports': [
|
|
{'id': uuids.port_id,
|
|
constants.BINDING_HOST_ID: instance.host}]}
|
|
self.api.list_ports = mock.Mock(return_value=get_ports)
|
|
update_port_mock = mock.Mock()
|
|
get_client_mock.return_value.update_port = update_port_mock
|
|
with mock.patch.object(self.api, 'has_port_binding_extension',
|
|
return_value=False):
|
|
self.api.setup_networks_on_host(self.context,
|
|
instance,
|
|
host=instance.host,
|
|
teardown=True)
|
|
update_port_mock.assert_not_called()
|
|
|
|
def test__update_port_with_migration_profile_raise_exception(self):
|
|
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
port_id = uuids.port_id
|
|
migrate_profile = {'fake-attribute': 'my-new-host'}
|
|
port_profile = {'port': {
|
|
constants.BINDING_PROFILE: migrate_profile}}
|
|
update_port_mock = mock.Mock(side_effect=test.TestingException())
|
|
admin_client = mock.Mock(update_port=update_port_mock)
|
|
self.assertRaises(test.TestingException,
|
|
self.api._update_port_with_migration_profile,
|
|
instance, port_id, migrate_profile, admin_client)
|
|
update_port_mock.assert_called_once_with(port_id, port_profile)
|
|
|
|
@mock.patch('nova.objects.Instance.get_network_info')
|
|
def test_get_preexisting_port_ids(self, mock_get_nw_info):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
mock_get_nw_info.return_value = [model.VIF(
|
|
id='1', preserve_on_delete=False), model.VIF(
|
|
id='2', preserve_on_delete=True), model.VIF(
|
|
id='3', preserve_on_delete=True)]
|
|
result = self.api._get_preexisting_port_ids(instance)
|
|
self.assertEqual(['2', '3'], result, "Invalid preexisting ports")
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_unbind_ports_get_client(self, mock_neutron, mock_show):
|
|
mock_ctx = mock.Mock(is_admin=False)
|
|
ports = ["1", "2", "3"]
|
|
|
|
# mock_show return a fake port detail, which did not contain arq
|
|
# binding info. mock_neutron would give you a mock world.
|
|
mock_show.side_effect = [{"id": "1"}, {"id": "2"}, {"id": "3"}]
|
|
|
|
self.api._unbind_ports(mock_ctx, ports, mock_neutron)
|
|
|
|
get_client_calls = []
|
|
get_client_calls.append(mock.call(mock_ctx, admin=True))
|
|
|
|
self.assertEqual(1, mock_neutron.call_count)
|
|
mock_neutron.assert_has_calls(get_client_calls, True)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_unbind_ports(self, mock_neutron, mock_show):
|
|
mock_client = mock.Mock()
|
|
mock_update_port = mock.Mock()
|
|
mock_client.update_port = mock_update_port
|
|
mock_ctx = mock.Mock(is_admin=False)
|
|
ports = ["1", "2", "3"]
|
|
mock_show.side_effect = [{"id": "1"}, {"id": "2"}, {"id": "3"}]
|
|
|
|
api = neutronapi.API()
|
|
api._unbind_ports(mock_ctx, ports, mock_neutron, mock_client)
|
|
|
|
body = {'port': {'device_id': '', 'device_owner': ''}}
|
|
body['port'][constants.BINDING_HOST_ID] = None
|
|
body['port'][constants.BINDING_PROFILE] = {}
|
|
update_port_calls = []
|
|
for p in ports:
|
|
update_port_calls.append(mock.call(p, body))
|
|
|
|
self.assertEqual(3, mock_update_port.call_count)
|
|
mock_update_port.assert_has_calls(update_port_calls)
|
|
|
|
def test_unbind_ports_no_port_ids(self):
|
|
# Tests that None entries in the ports list are filtered out.
|
|
mock_client = mock.Mock()
|
|
mock_update_port = mock.Mock()
|
|
mock_client.update_port = mock_update_port
|
|
mock_ctx = mock.Mock(is_admin=False)
|
|
|
|
api = neutronapi.API()
|
|
api._unbind_ports(mock_ctx, [None], mock_client, mock_client)
|
|
self.assertFalse(mock_update_port.called)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch('nova.network.neutron.API.get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.excutils')
|
|
@mock.patch('nova.network.neutron.API._delete_ports')
|
|
@mock.patch('nova.network.neutron.API._check_external_network_attach')
|
|
@mock.patch('nova.network.neutron.LOG')
|
|
@mock.patch('nova.network.neutron.API._unbind_ports')
|
|
@mock.patch('nova.network.neutron.API._populate_neutron_extension_values')
|
|
@mock.patch('nova.network.neutron.API._get_available_networks')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def test_allocate_for_instance_unbind(self, mock_vif,
|
|
mock_ntrn,
|
|
mock_avail_nets,
|
|
mock_ext_vals,
|
|
mock_unbind,
|
|
mock_log,
|
|
mock_cena,
|
|
mock_del_ports,
|
|
mock_exeu,
|
|
mock_giwn):
|
|
mock_nc = mock.Mock()
|
|
|
|
def show_port(port_id):
|
|
return {'port': {'network_id': 'net-1', 'id': port_id,
|
|
'mac_address': 'fakemac',
|
|
'tenant_id': 'proj-1'}}
|
|
mock_nc.show_port = show_port
|
|
|
|
mock_ntrn.return_value = mock_nc
|
|
|
|
def update_port(port_id, body):
|
|
if port_id == uuids.fail_port_id:
|
|
raise Exception
|
|
return {"port": {'mac_address': 'fakemac',
|
|
'id': port_id}}
|
|
mock_nc.update_port.side_effect = update_port
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
|
|
nw_req = objects.NetworkRequestList(
|
|
objects = [objects.NetworkRequest(port_id=uuids.portid_1),
|
|
objects.NetworkRequest(port_id=uuids.portid_2),
|
|
objects.NetworkRequest(port_id=uuids.fail_port_id)])
|
|
mock_avail_nets.return_value = [{'id': 'net-1',
|
|
'subnets': ['subnet1']}]
|
|
|
|
self.api.allocate_for_instance(mock.sentinel.ctx, mock_inst,
|
|
requested_networks=nw_req)
|
|
|
|
mock_unbind.assert_called_once_with(mock.sentinel.ctx,
|
|
[uuids.portid_1, uuids.portid_2],
|
|
mock.ANY,
|
|
mock.ANY)
|
|
|
|
@mock.patch('nova.network.neutron.LOG')
|
|
@mock.patch('nova.network.neutron.API._delete_ports')
|
|
@mock.patch('nova.network.neutron.API._unbind_ports')
|
|
@mock.patch('nova.network.neutron.API._get_preexisting_port_ids')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch.object(objects.VirtualInterface, 'delete_by_instance_uuid')
|
|
def test_preexisting_deallocate_for_instance(self, mock_delete_vifs,
|
|
mock_ntrn,
|
|
mock_gppids,
|
|
mock_unbind,
|
|
mock_deletep,
|
|
mock_log):
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.list_ports.return_value = {'ports': [
|
|
{'id': uuids.portid_1}, {'id': uuids.portid_2},
|
|
{'id': uuids.portid_3}
|
|
]}
|
|
nw_req = objects.NetworkRequestList(
|
|
objects = [objects.NetworkRequest(network_id='net-1',
|
|
address='192.168.0.3',
|
|
port_id=uuids.portid_1,
|
|
pci_request_id=uuids.pci_1,
|
|
arq_uuid=uuids.arq,
|
|
device_profile=None)])
|
|
mock_gppids.return_value = [uuids.portid_3]
|
|
|
|
self.api.deallocate_for_instance(mock.sentinel.ctx, mock_inst,
|
|
requested_networks=nw_req)
|
|
|
|
mock_unbind.assert_called_once_with(mock.sentinel.ctx,
|
|
set([uuids.portid_1,
|
|
uuids.portid_3]),
|
|
mock.ANY)
|
|
mock_deletep.assert_called_once_with(mock_nc,
|
|
mock_inst,
|
|
set([uuids.portid_2]),
|
|
raise_if_fail=True)
|
|
mock_delete_vifs.assert_called_once_with(mock.sentinel.ctx, 'inst-1')
|
|
|
|
@mock.patch('nova.network.neutron.API._delete_nic_metadata')
|
|
@mock.patch('nova.network.neutron.API.get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.API._unbind_ports')
|
|
@mock.patch('nova.objects.Instance.get_network_info')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch.object(objects.VirtualInterface, 'get_by_uuid')
|
|
def test_preexisting_deallocate_port_for_instance(self,
|
|
mock_get_vif_by_uuid,
|
|
mock_ntrn,
|
|
mock_inst_get_nwinfo,
|
|
mock_unbind,
|
|
mock_netinfo,
|
|
mock_del_nic_meta):
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
mock_inst.get_network_info.return_value = [model.VIF(
|
|
id='1', preserve_on_delete=False), model.VIF(
|
|
id='2', preserve_on_delete=True), model.VIF(
|
|
id='3', preserve_on_delete=True)]
|
|
mock_client = mock.Mock()
|
|
mock_client.show_port.return_value = {'port': {}}
|
|
mock_ntrn.return_value = mock_client
|
|
vif = objects.VirtualInterface()
|
|
vif.destroy = mock.MagicMock()
|
|
mock_get_vif_by_uuid.return_value = vif
|
|
_, port_allocation = self.api.deallocate_port_for_instance(
|
|
mock.sentinel.ctx, mock_inst, '2')
|
|
mock_unbind.assert_called_once_with(mock.sentinel.ctx, ['2'],
|
|
mock_client)
|
|
mock_get_vif_by_uuid.assert_called_once_with(mock.sentinel.ctx, '2')
|
|
mock_del_nic_meta.assert_called_once_with(mock_inst, vif)
|
|
vif.destroy.assert_called_once_with()
|
|
self.assertEqual({}, port_allocation)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch('nova.network.neutron.API.get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.API._delete_nic_metadata')
|
|
@mock.patch.object(objects.VirtualInterface, 'get_by_uuid')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_deallocate_port_for_instance_port_with_allocation(
|
|
self, mock_get_client, mock_get_vif_by_uuid, mock_del_nic_meta,
|
|
mock_netinfo):
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
mock_inst.get_network_info.return_value = [
|
|
model.VIF(id=uuids.port_uid, preserve_on_delete=True)
|
|
]
|
|
vif = objects.VirtualInterface()
|
|
vif.tag = 'foo'
|
|
vif.destroy = mock.MagicMock()
|
|
mock_get_vif_by_uuid.return_value = vif
|
|
|
|
mock_client = mock.Mock()
|
|
mock_client.show_port.return_value = {
|
|
'port': {
|
|
constants.RESOURCE_REQUEST: {
|
|
'resources': {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 1000
|
|
}
|
|
},
|
|
'binding:profile': {
|
|
'allocation': uuids.rp1
|
|
}
|
|
}
|
|
}
|
|
mock_get_client.return_value = mock_client
|
|
|
|
_, port_allocation = self.api.deallocate_port_for_instance(
|
|
mock.sentinel.ctx, mock_inst, uuids.port_id)
|
|
|
|
self.assertEqual(
|
|
{
|
|
uuids.rp1: {
|
|
"resources": {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 1000
|
|
}
|
|
}
|
|
},
|
|
port_allocation)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch('nova.network.neutron.API.get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.API._delete_nic_metadata')
|
|
@mock.patch.object(objects.VirtualInterface, 'get_by_uuid')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_deallocate_port_for_instance_port_with_extended_allocation(
|
|
self, mock_get_client, mock_get_vif_by_uuid, mock_del_nic_meta,
|
|
mock_netinfo):
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
mock_inst.get_network_info.return_value = [
|
|
model.VIF(id=uuids.port_uid, preserve_on_delete=True)
|
|
]
|
|
vif = objects.VirtualInterface()
|
|
vif.tag = 'foo'
|
|
vif.destroy = mock.MagicMock()
|
|
mock_get_vif_by_uuid.return_value = vif
|
|
|
|
mock_client = mock.Mock()
|
|
mock_client.show_port.return_value = {
|
|
'port': {
|
|
constants.RESOURCE_REQUEST: {
|
|
'request_groups': [
|
|
{
|
|
'id': uuids.group1,
|
|
'resources': {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 1000,
|
|
}
|
|
}
|
|
],
|
|
},
|
|
'binding:profile': {
|
|
'allocation': {uuids.group1: uuids.rp1}
|
|
}
|
|
}
|
|
}
|
|
mock_get_client.return_value = mock_client
|
|
|
|
_, port_allocation = self.api.deallocate_port_for_instance(
|
|
mock.sentinel.ctx, mock_inst, uuids.port_id)
|
|
|
|
self.assertEqual(
|
|
{
|
|
uuids.rp1: {
|
|
"resources": {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 1000
|
|
}
|
|
}
|
|
},
|
|
port_allocation
|
|
)
|
|
|
|
@mock.patch('nova.network.neutron.API.get_instance_nw_info')
|
|
@mock.patch('nova.network.neutron.API._delete_nic_metadata')
|
|
@mock.patch.object(objects.VirtualInterface, 'get_by_uuid')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_deallocate_port_for_instance_port_already_deleted(
|
|
self, mock_get_client, mock_get_vif_by_uuid, mock_del_nic_meta,
|
|
mock_netinfo):
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
network_info = [
|
|
model.VIF(
|
|
id=uuids.port_id, preserve_on_delete=True,
|
|
profile={'allocation': uuids.rp1})
|
|
]
|
|
mock_inst.get_network_info.return_value = network_info
|
|
vif = objects.VirtualInterface()
|
|
vif.destroy = mock.MagicMock()
|
|
mock_get_vif_by_uuid.return_value = vif
|
|
|
|
mock_client = mock.Mock()
|
|
mock_client.show_port.side_effect = exception.PortNotFound(
|
|
port_id=uuids.port_id)
|
|
mock_get_client.return_value = mock_client
|
|
|
|
_, port_allocation = self.api.deallocate_port_for_instance(
|
|
mock.sentinel.ctx, mock_inst, uuids.port_id)
|
|
|
|
self.assertEqual({}, port_allocation)
|
|
self.assertIn(
|
|
'Resource allocation for this port may be leaked',
|
|
self.stdlog.logger.output)
|
|
|
|
def test_delete_nic_metadata(self):
|
|
vif = objects.VirtualInterface(address='aa:bb:cc:dd:ee:ff', tag='foo')
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.device_metadata = objects.InstanceDeviceMetadata(
|
|
devices=[objects.NetworkInterfaceMetadata(
|
|
mac='aa:bb:cc:dd:ee:ff', tag='foo')])
|
|
instance.save = mock.Mock()
|
|
self.api._delete_nic_metadata(instance, vif)
|
|
self.assertEqual(0, len(instance.device_metadata.devices))
|
|
instance.save.assert_called_once_with()
|
|
|
|
def test_delete_nic_metadata_no_metadata(self):
|
|
vif = objects.VirtualInterface(address='aa:bb:cc:dd:ee:ff', tag='foo')
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
instance.device_metadata = None
|
|
instance.save = mock.Mock()
|
|
self.api._delete_nic_metadata(instance, vif)
|
|
instance.save.assert_not_called()
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch('nova.network.neutron.API._check_external_network_attach')
|
|
@mock.patch('nova.network.neutron.API._populate_neutron_extension_values')
|
|
@mock.patch('nova.network.neutron.API._get_available_networks')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_port_binding_failed_created_port(self, mock_ntrn,
|
|
mock_avail_nets,
|
|
mock_ext_vals,
|
|
mock_cena):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid=uuids.inst_1)
|
|
mock_avail_nets.return_value = [{'id': 'net-1',
|
|
'subnets': ['subnet1']}]
|
|
mock_nc.create_port.return_value = {'port': {'id': uuids.portid_1}}
|
|
port_response = {'port': {'id': uuids.portid_1,
|
|
'tenant_id': mock_inst.project_id,
|
|
'binding:vif_type': 'binding_failed'}}
|
|
mock_nc.update_port.return_value = port_response
|
|
|
|
self.assertRaises(exception.PortBindingFailed,
|
|
self.api.allocate_for_instance,
|
|
mock.sentinel.ctx,
|
|
mock_inst, None)
|
|
mock_nc.delete_port.assert_called_once_with(uuids.portid_1)
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_port_binding_failed_with_request(self, mock_ntrn,
|
|
mock_show_port):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_inst = mock.Mock(project_id="proj-1",
|
|
availability_zone='zone-1',
|
|
uuid='inst-1')
|
|
mock_show_port.return_value = {
|
|
'id': uuids.portid_1,
|
|
'tenant_id': mock_inst.project_id,
|
|
'binding:vif_type': 'binding_failed'}
|
|
nw_req = objects.NetworkRequestList(
|
|
objects = [objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
|
|
self.assertRaises(exception.PortBindingFailed,
|
|
self.api.allocate_for_instance,
|
|
mock.sentinel.ctx, mock_inst,
|
|
requested_networks=nw_req)
|
|
|
|
@mock.patch('nova.objects.virtual_interface.VirtualInterface.create')
|
|
@mock.patch('nova.network.neutron.API._check_external_network_attach')
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.API._update_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_port_with_resource_request_has_allocation_in_binding(
|
|
self, mock_get_client, mock_update_port, mock_show_port,
|
|
mock_check_external, mock_vif_create):
|
|
|
|
nw_req = objects.NetworkRequestList(
|
|
objects = [objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
mock_inst = mock.Mock(
|
|
uuid=uuids.instance_uuid,
|
|
project_id=uuids.project_id,
|
|
availability_zone='nova',
|
|
)
|
|
port = {
|
|
'id': uuids.portid_1,
|
|
'tenant_id': uuids.project_id,
|
|
'network_id': uuids.networkid_1,
|
|
'mac_address': 'fake-mac',
|
|
constants.RESOURCE_REQUEST: 'fake-request'
|
|
}
|
|
mock_show_port.return_value = port
|
|
mock_get_client.return_value.list_networks.return_value = {
|
|
"networks": [{'id': uuids.networkid_1,
|
|
'port_security_enabled': False}]}
|
|
mock_update_port.return_value = port
|
|
|
|
with mock.patch.object(self.api, 'get_instance_nw_info'):
|
|
self.api.allocate_for_instance(
|
|
mock.sentinel.ctx, mock_inst,
|
|
requested_networks=nw_req,
|
|
resource_provider_mapping={uuids.portid_1: [uuids.rp1]})
|
|
|
|
mock_update_port.assert_called_once_with(
|
|
mock_get_client.return_value, mock_inst, uuids.portid_1,
|
|
{
|
|
'port': {
|
|
'binding:host_id': None,
|
|
'device_id': uuids.instance_uuid,
|
|
'binding:profile': {
|
|
'allocation': uuids.rp1},
|
|
'device_owner': 'compute:nova'}})
|
|
mock_show_port.assert_called_once_with(
|
|
mock.sentinel.ctx, uuids.portid_1,
|
|
neutron_client=mock_get_client.return_value)
|
|
|
|
@mock.patch(
|
|
"nova.network.neutron.API.has_extended_resource_request_extension")
|
|
@mock.patch('nova.objects.virtual_interface.VirtualInterface.create')
|
|
@mock.patch('nova.network.neutron.API._check_external_network_attach')
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.API._update_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_port_with_extended_resource_request_has_allocation_in_binding(
|
|
self, mock_get_client, mock_update_port, mock_show_port,
|
|
mock_check_external, mock_vif_create, mock_has_extended_res_req):
|
|
|
|
nw_req = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
mock_inst = mock.Mock(
|
|
uuid=uuids.instance_uuid,
|
|
project_id=uuids.project_id,
|
|
availability_zone='nova',
|
|
)
|
|
port = {
|
|
'id': uuids.portid_1,
|
|
'tenant_id': uuids.project_id,
|
|
'network_id': uuids.networkid_1,
|
|
'mac_address': 'fake-mac',
|
|
constants.RESOURCE_REQUEST: {
|
|
"request_groups": [
|
|
{"id": uuids.group1},
|
|
{"id": uuids.group2},
|
|
]
|
|
}
|
|
}
|
|
mock_show_port.return_value = port
|
|
mock_get_client.return_value.list_networks.return_value = {
|
|
"networks": [{'id': uuids.networkid_1,
|
|
'port_security_enabled': False}]}
|
|
mock_update_port.return_value = port
|
|
mock_has_extended_res_req.return_value = True
|
|
|
|
with mock.patch.object(self.api, 'get_instance_nw_info'):
|
|
self.api.allocate_for_instance(
|
|
mock.sentinel.ctx, mock_inst,
|
|
requested_networks=nw_req,
|
|
resource_provider_mapping={
|
|
uuids.group1: [uuids.rp1],
|
|
uuids.group2: [uuids.rp2],
|
|
})
|
|
|
|
mock_update_port.assert_called_once_with(
|
|
mock_get_client.return_value, mock_inst, uuids.portid_1,
|
|
{
|
|
'port': {
|
|
'binding:host_id': None,
|
|
'device_id': uuids.instance_uuid,
|
|
'binding:profile': {
|
|
'allocation': {
|
|
uuids.group1: uuids.rp1,
|
|
uuids.group2: uuids.rp2,
|
|
}
|
|
},
|
|
'device_owner': 'compute:nova'}})
|
|
mock_show_port.assert_called_once_with(
|
|
mock.sentinel.ctx, uuids.portid_1,
|
|
neutron_client=mock_get_client.return_value)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_floating_ip_by_address_not_found_neutron_not_found(self,
|
|
mock_ntrn):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.list_floatingips.side_effect = exceptions.NotFound()
|
|
address = '172.24.4.227'
|
|
self.assertRaises(exception.FloatingIpNotFoundForAddress,
|
|
self.api.get_floating_ip_by_address,
|
|
self.context, address)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_floating_ip_by_address_not_found_neutron_raises_non404(self,
|
|
mock_ntrn):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.list_floatingips.side_effect = exceptions.InternalServerError()
|
|
address = '172.24.4.227'
|
|
self.assertRaises(exceptions.InternalServerError,
|
|
self.api.get_floating_ip_by_address,
|
|
self.context, address)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_floating_ips_by_project_not_found(self, mock_ntrn):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.list_floatingips.side_effect = exceptions.NotFound()
|
|
fips = self.api.get_floating_ips_by_project(self.context)
|
|
self.assertEqual([], fips)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_floating_ips_by_project_not_found_legacy(self, mock_ntrn):
|
|
# FIXME(danms): Remove this test along with the code path it tests
|
|
# when bug 1513879 is fixed.
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
# neutronclient doesn't raise NotFound in this scenario, it raises a
|
|
# NeutronClientException with status_code=404
|
|
notfound = exceptions.NeutronClientException(status_code=404)
|
|
mock_nc.list_floatingips.side_effect = notfound
|
|
fips = self.api.get_floating_ips_by_project(self.context)
|
|
self.assertEqual([], fips)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_floating_ips_by_project_raises_non404(self, mock_ntrn):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
mock_nc.list_floatingips.side_effect = exceptions.InternalServerError()
|
|
self.assertRaises(exceptions.InternalServerError,
|
|
self.api.get_floating_ips_by_project,
|
|
self.context)
|
|
|
|
@mock.patch.object(neutronapi.API, '_refresh_neutron_extensions_cache')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def _test_get_floating_ips_by_project(
|
|
self, fip_ext_enabled, has_ports, mock_ntrn, mock_refresh):
|
|
mock_nc = mock.Mock()
|
|
mock_ntrn.return_value = mock_nc
|
|
# NOTE(stephenfin): These are clearly not full responses
|
|
mock_nc.list_floatingips.return_value = {
|
|
'floatingips': [
|
|
{
|
|
'id': uuids.fip_id,
|
|
'floating_network_id': uuids.fip_net_id,
|
|
'port_id': uuids.fip_port_id,
|
|
}
|
|
]
|
|
}
|
|
mock_nc.list_networks.return_value = {
|
|
'networks': [
|
|
{
|
|
'id': uuids.fip_net_id,
|
|
},
|
|
],
|
|
}
|
|
if has_ports:
|
|
mock_nc.list_ports.return_value = {
|
|
'ports': [
|
|
{
|
|
'id': uuids.fip_port_id,
|
|
},
|
|
],
|
|
}
|
|
else:
|
|
mock_nc.list_ports.return_value = {'ports': []}
|
|
|
|
if fip_ext_enabled:
|
|
self.api.extensions = {
|
|
constants.FIP_PORT_DETAILS: {
|
|
'alias': constants.FIP_PORT_DETAILS,
|
|
},
|
|
}
|
|
else:
|
|
self.api.extensions = {}
|
|
|
|
fips = self.api.get_floating_ips_by_project(self.context)
|
|
|
|
mock_nc.list_networks.assert_called_once_with(
|
|
id=[uuids.fip_net_id])
|
|
self.assertEqual(1, len(fips))
|
|
|
|
if fip_ext_enabled:
|
|
mock_nc.list_ports.assert_not_called()
|
|
self.assertNotIn('port_details', fips[0])
|
|
else:
|
|
mock_nc.list_ports.assert_called_once_with(
|
|
tenant_id=self.context.project_id)
|
|
self.assertIn('port_details', fips[0])
|
|
|
|
if has_ports:
|
|
self.assertIsNotNone(fips[0]['port_details'])
|
|
else:
|
|
self.assertIsNone(fips[0]['port_details'])
|
|
|
|
def test_get_floating_ips_by_project_with_fip_port_details_ext(self):
|
|
"""Make sure we used embedded port details if available."""
|
|
self._test_get_floating_ips_by_project(True, True)
|
|
|
|
def test_get_floating_ips_by_project_without_fip_port_details_ext(self):
|
|
"""Make sure we make a second request for port details if necessary."""
|
|
self._test_get_floating_ips_by_project(False, True)
|
|
|
|
def test_get_floating_ips_by_project_without_ports(self):
|
|
"""Make sure we don't fail for floating IPs without attached ports."""
|
|
self._test_get_floating_ips_by_project(False, False)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=True))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_unbind_ports_reset_dns_name_by_admin(self, mock_show):
|
|
neutron = mock.Mock()
|
|
neutron.show_network.return_value = {
|
|
'network': {
|
|
'id': 'net1',
|
|
'dns_domain': None
|
|
}
|
|
}
|
|
port_client = mock.Mock()
|
|
ports = [uuids.port_id]
|
|
mock_show.return_value = {'id': uuids.port}
|
|
self.api._unbind_ports(self.context, ports, neutron, port_client)
|
|
port_req_body = {'port': {'binding:host_id': None,
|
|
'binding:profile': {},
|
|
'device_id': '',
|
|
'device_owner': '',
|
|
'dns_name': ''}}
|
|
port_client.update_port.assert_called_once_with(
|
|
uuids.port_id, port_req_body)
|
|
neutron.update_port.assert_not_called()
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=True))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_unbind_ports_reset_dns_name_by_non_admin(self, mock_show):
|
|
neutron = mock.Mock()
|
|
neutron.show_network.return_value = {
|
|
'network': {
|
|
'id': 'net1',
|
|
'dns_domain': 'test.domain'
|
|
}
|
|
}
|
|
port_client = mock.Mock()
|
|
ports = [uuids.port_id]
|
|
mock_show.return_value = {'id': uuids.port}
|
|
self.api._unbind_ports(self.context, ports, neutron, port_client)
|
|
admin_port_req_body = {'port': {'binding:host_id': None,
|
|
'binding:profile': {},
|
|
'device_id': '',
|
|
'device_owner': ''}}
|
|
non_admin_port_req_body = {'port': {'dns_name': ''}}
|
|
port_client.update_port.assert_called_once_with(
|
|
uuids.port_id, admin_port_req_body)
|
|
neutron.update_port.assert_called_once_with(
|
|
uuids.port_id, non_admin_port_req_body)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_unbind_ports_reset_allocation_in_port_binding(self, mock_show):
|
|
neutron = mock.Mock()
|
|
port_client = mock.Mock()
|
|
ports = [uuids.port_id]
|
|
mock_show.return_value = {'id': uuids.port,
|
|
'binding:profile': {'allocation': uuids.rp1}}
|
|
self.api._unbind_ports(self.context, ports, neutron, port_client)
|
|
port_req_body = {'port': {'binding:host_id': None,
|
|
'binding:profile': {},
|
|
'device_id': '',
|
|
'device_owner': ''}}
|
|
port_client.update_port.assert_called_once_with(
|
|
uuids.port_id, port_req_body)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
def test_unbind_ports_reset_binding_profile(self, mock_show):
|
|
neutron = mock.Mock()
|
|
port_client = mock.Mock()
|
|
ports = [uuids.port_id]
|
|
mock_show.return_value = {
|
|
'id': uuids.port,
|
|
'binding:profile': {'pci_vendor_info': '1377:0047',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'card_serial_number': 'MT2113X00000',
|
|
'physical_network': 'physnet1',
|
|
'capabilities': ['switchdev']}
|
|
}
|
|
self.api._unbind_ports(self.context, ports, neutron, port_client)
|
|
port_req_body = {'port': {'binding:host_id': None,
|
|
'binding:profile':
|
|
{'capabilities': ['switchdev']},
|
|
'device_id': '',
|
|
'device_owner': ''}
|
|
}
|
|
port_client.update_port.assert_called_once_with(
|
|
uuids.port_id, port_req_body)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._populate_neutron_extension_values')
|
|
@mock.patch('nova.network.neutron.API._update_port',
|
|
# called twice, fails on the 2nd call and triggers the cleanup
|
|
side_effect=(mock.MagicMock(),
|
|
exception.PortInUse(
|
|
port_id=uuids.created_port_id)))
|
|
@mock.patch.object(objects.VirtualInterface, 'create')
|
|
@mock.patch.object(objects.VirtualInterface, 'destroy')
|
|
@mock.patch('nova.network.neutron.API._unbind_ports')
|
|
@mock.patch('nova.network.neutron.API._delete_ports')
|
|
def test_update_ports_for_instance_fails_rollback_ports_and_vifs(self,
|
|
mock_delete_ports,
|
|
mock_unbind_ports,
|
|
mock_vif_destroy,
|
|
mock_vif_create,
|
|
mock_update_port,
|
|
mock_populate_ext_values):
|
|
"""Makes sure we rollback ports and VIFs if we fail updating ports"""
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
ntrn = mock.Mock(spec=client.Client)
|
|
# we have two requests, one with a preexisting port and one where nova
|
|
# created the port (on the same network)
|
|
requests_and_created_ports = [
|
|
(objects.NetworkRequest(network_id=uuids.network_id,
|
|
port_id=uuids.preexisting_port_id),
|
|
None), # None means Nova didn't create this port
|
|
(objects.NetworkRequest(network_id=uuids.network_id,
|
|
port_id=uuids.created_port_id),
|
|
uuids.created_port_id),
|
|
]
|
|
network = {'id': uuids.network_id}
|
|
nets = {uuids.network_id: network}
|
|
self.assertRaises(exception.PortInUse,
|
|
self.api._update_ports_for_instance,
|
|
self.context, instance, ntrn, ntrn,
|
|
requests_and_created_ports, nets, bind_host_id=None,
|
|
requested_ports_dict=None,
|
|
network_arqs = None)
|
|
# assert the calls
|
|
mock_update_port.assert_has_calls([
|
|
mock.call(ntrn, instance, uuids.preexisting_port_id, mock.ANY),
|
|
mock.call(ntrn, instance, uuids.created_port_id, mock.ANY)
|
|
])
|
|
# we only got to create one vif since the 2nd _update_port call fails
|
|
mock_vif_create.assert_called_once_with()
|
|
# we only destroy one vif since we only created one
|
|
mock_vif_destroy.assert_called_once_with()
|
|
# we unbind the pre-existing port
|
|
mock_unbind_ports.assert_called_once_with(
|
|
self.context, [uuids.preexisting_port_id], ntrn, ntrn)
|
|
# we delete the created port
|
|
mock_delete_ports.assert_called_once_with(
|
|
ntrn, instance, [uuids.created_port_id])
|
|
|
|
@mock.patch('nova.network.neutron.API._get_floating_ip_by_address',
|
|
return_value={"port_id": "1"})
|
|
@mock.patch('nova.network.neutron.API._show_port',
|
|
side_effect=exception.PortNotFound(port_id='1'))
|
|
def test_get_instance_id_by_floating_address_port_not_found(self,
|
|
mock_show,
|
|
mock_get):
|
|
api = neutronapi.API()
|
|
fip = api.get_instance_id_by_floating_address(self.context,
|
|
'172.24.4.227')
|
|
self.assertIsNone(fip)
|
|
|
|
@mock.patch('nova.network.neutron.API._show_port',
|
|
side_effect=exception.PortNotFound(port_id=uuids.port))
|
|
@mock.patch.object(neutronapi.LOG, 'exception')
|
|
def test_unbind_ports_port_show_portnotfound(self, mock_log, mock_show):
|
|
api = neutronapi.API()
|
|
neutron_client = mock.Mock()
|
|
api._unbind_ports(self.context, [uuids.port_id],
|
|
neutron_client, neutron_client)
|
|
mock_show.assert_called_once_with(
|
|
mock.ANY, uuids.port_id,
|
|
fields=['binding:profile', 'network_id'],
|
|
neutron_client=mock.ANY)
|
|
mock_log.assert_not_called()
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch.object(neutronapi, 'LOG')
|
|
def test_unbind_ports_port_show_portnotfound_multiple_ports(
|
|
self, mock_log, mock_show,
|
|
):
|
|
"""Ensure we continue unbinding ports even when one isn't found."""
|
|
mock_show.side_effect = [
|
|
exception.PortNotFound(port_id=uuids.port_a),
|
|
{'id': uuids.port_b},
|
|
]
|
|
api = neutronapi.API()
|
|
neutron_client = mock.Mock()
|
|
|
|
api._unbind_ports(
|
|
self.context,
|
|
[uuids.port_a, uuids.port_b],
|
|
neutron_client,
|
|
neutron_client,
|
|
)
|
|
|
|
mock_show.assert_has_calls(
|
|
[
|
|
mock.call(
|
|
self.context,
|
|
uuids.port_a,
|
|
fields=['binding:profile', 'network_id'],
|
|
neutron_client=neutron_client,
|
|
),
|
|
mock.call(
|
|
self.context,
|
|
uuids.port_b,
|
|
fields=['binding:profile', 'network_id'],
|
|
neutron_client=neutron_client,
|
|
),
|
|
]
|
|
)
|
|
# Only the port that exists should be updated
|
|
neutron_client.update_port.assert_called_once_with(
|
|
uuids.port_b,
|
|
{
|
|
'port': {
|
|
'device_id': '',
|
|
'device_owner': '',
|
|
'binding:profile': {},
|
|
'binding:host_id': None,
|
|
}
|
|
}
|
|
)
|
|
mock_log.exception.assert_not_called()
|
|
mock_log.debug.assert_called_with(
|
|
'Unable to show port %s as it no longer exists.', uuids.port_a,
|
|
)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port',
|
|
side_effect=Exception)
|
|
@mock.patch.object(neutronapi.LOG, 'exception')
|
|
def test_unbind_ports_port_show_unexpected_error(self,
|
|
mock_log,
|
|
mock_show):
|
|
api = neutronapi.API()
|
|
neutron_client = mock.Mock()
|
|
mock_show.return_value = {'id': uuids.port}
|
|
api._unbind_ports(self.context, [uuids.port_id],
|
|
neutron_client, neutron_client)
|
|
neutron_client.update_port.assert_called_once_with(
|
|
uuids.port_id, {'port': {
|
|
'device_id': '', 'device_owner': '',
|
|
'binding:profile': {}, 'binding:host_id': None}})
|
|
self.assertTrue(mock_log.called)
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch.object(neutronapi.LOG, 'exception')
|
|
def test_unbind_ports_port_update_portnotfound(self, mock_log, mock_show):
|
|
api = neutronapi.API()
|
|
neutron_client = mock.Mock()
|
|
neutron_client.update_port = mock.Mock(
|
|
side_effect=exceptions.PortNotFoundClient)
|
|
mock_show.return_value = {'id': uuids.port}
|
|
api._unbind_ports(self.context, [uuids.port_id],
|
|
neutron_client, neutron_client)
|
|
neutron_client.update_port.assert_called_once_with(
|
|
uuids.port_id, {'port': {
|
|
'device_id': '', 'device_owner': '',
|
|
'binding:profile': {}, 'binding:host_id': None}})
|
|
mock_log.assert_not_called()
|
|
|
|
@mock.patch('nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch.object(neutronapi.LOG, 'exception')
|
|
def test_unbind_ports_port_update_unexpected_error(
|
|
self, mock_log, mock_show,
|
|
):
|
|
api = neutronapi.API()
|
|
neutron_client = mock.Mock()
|
|
neutron_client.update_port = mock.Mock(
|
|
side_effect=test.TestingException)
|
|
mock_show.return_value = {'id': uuids.port}
|
|
api._unbind_ports(self.context, [uuids.port_id],
|
|
neutron_client, neutron_client)
|
|
neutron_client.update_port.assert_called_once_with(
|
|
uuids.port_id, {'port': {
|
|
'device_id': '', 'device_owner': '',
|
|
'binding:profile': {}, 'binding:host_id': None}})
|
|
self.assertTrue(mock_log.called)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_requests_no_allocate(self, mock_get_client):
|
|
"""Ensure physnet info is not retrieved when networks are not to be
|
|
allocated.
|
|
"""
|
|
requested_networks = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id=net_req_obj.NETWORK_ID_NONE)
|
|
])
|
|
pci_requests = objects.InstancePCIRequests()
|
|
api = neutronapi.API()
|
|
|
|
result = api.create_resource_requests(
|
|
self.context, requested_networks, pci_requests)
|
|
network_metadata, port_resource_requests, _ = result
|
|
|
|
self.assertFalse(mock_get_client.called)
|
|
self.assertIsNone(network_metadata)
|
|
self.assertEqual([], port_resource_requests)
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, 'has_extended_resource_request_extension',
|
|
return_value=False)
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch.object(neutronapi, 'get_client', return_value=mock.Mock())
|
|
def test_create_resource_requests_auto_allocated(
|
|
self, mock_get_client, mock_get_physnet_tunneled_info,
|
|
mock_has_extended_res_req
|
|
):
|
|
"""Ensure physnet info is not retrieved for auto-allocated networks.
|
|
|
|
This isn't possible so we shouldn't attempt to do it.
|
|
"""
|
|
requested_networks = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id=net_req_obj.NETWORK_ID_AUTO)
|
|
])
|
|
pci_requests = objects.InstancePCIRequests()
|
|
api = neutronapi.API()
|
|
|
|
result = api.create_resource_requests(
|
|
self.context, requested_networks, pci_requests)
|
|
network_metadata, port_resource_requests, _ = result
|
|
|
|
mock_get_physnet_tunneled_info.assert_not_called()
|
|
self.assertEqual(set(), network_metadata.physnets)
|
|
self.assertFalse(network_metadata.tunneled)
|
|
self.assertEqual([], port_resource_requests)
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, 'has_extended_resource_request_extension',
|
|
return_value=False)
|
|
@mock.patch('nova.objects.request_spec.RequestGroup.from_port_request')
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch.object(neutronapi.API, "_get_port_vnic_info")
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_requests(
|
|
self, getclient, mock_get_port_vnic_info,
|
|
mock_get_physnet_tunneled_info, mock_from_port_request,
|
|
mock_has_extended_res_req
|
|
):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects = [
|
|
objects.NetworkRequest(port_id=uuids.portid_1),
|
|
objects.NetworkRequest(network_id='net1'),
|
|
objects.NetworkRequest(port_id=uuids.portid_2),
|
|
objects.NetworkRequest(port_id=uuids.portid_3),
|
|
objects.NetworkRequest(port_id=uuids.portid_4),
|
|
objects.NetworkRequest(port_id=uuids.portid_5),
|
|
objects.NetworkRequest(port_id=uuids.trusted_port),
|
|
objects.NetworkRequest(port_id=uuids.portid_vdpa),
|
|
objects.NetworkRequest(port_id=uuids.portid_remote_managed)])
|
|
pci_requests = objects.InstancePCIRequests(requests=[])
|
|
# _get_port_vnic_info should be called for every NetworkRequest with a
|
|
# port_id attribute (so six times)
|
|
mock_get_port_vnic_info.side_effect = [
|
|
(model.VNIC_TYPE_DIRECT, None, 'netN', None, None, None),
|
|
(model.VNIC_TYPE_NORMAL, None, 'netN',
|
|
mock.sentinel.resource_request1, None, None),
|
|
(model.VNIC_TYPE_MACVTAP, None, 'netN', None, None, None),
|
|
(model.VNIC_TYPE_MACVTAP, None, 'netN', None, None, None),
|
|
(model.VNIC_TYPE_DIRECT_PHYSICAL, None, 'netN', None, None, None),
|
|
(model.VNIC_TYPE_DIRECT, True, 'netN',
|
|
mock.sentinel.resource_request2, None, None),
|
|
(model.VNIC_TYPE_VDPA, None, 'netN', None, None, None),
|
|
(model.VNIC_TYPE_REMOTE_MANAGED, None, 'netN', None, None, None),
|
|
]
|
|
# _get_physnet_tunneled_info should be called for every NetworkRequest
|
|
# (so seven times)
|
|
mock_get_physnet_tunneled_info.side_effect = [
|
|
('physnet1', False), ('physnet1', False), ('', True),
|
|
('physnet1', False), ('physnet2', False), ('physnet3', False),
|
|
('physnet4', False), ('physnet1', False), ('physnet1', False),
|
|
]
|
|
api = neutronapi.API()
|
|
|
|
mock_from_port_request.side_effect = [
|
|
mock.sentinel.request_group1,
|
|
mock.sentinel.request_group2,
|
|
]
|
|
|
|
result = api.create_resource_requests(
|
|
self.context, requested_networks, pci_requests)
|
|
network_metadata, port_resource_requests, _ = result
|
|
|
|
self.assertEqual([
|
|
mock.sentinel.request_group1,
|
|
mock.sentinel.request_group2],
|
|
port_resource_requests)
|
|
self.assertEqual(7, len(pci_requests.requests))
|
|
has_pci_request_id = [net.pci_request_id is not None for net in
|
|
requested_networks.objects]
|
|
self.assertEqual(pci_requests.requests[3].spec[0]["dev_type"],
|
|
"type-PF")
|
|
self.assertEqual(pci_requests.requests[5].spec[0]["dev_type"], "vdpa")
|
|
self.assertEqual(pci_requests.requests[6].spec[0]["remote_managed"],
|
|
'True')
|
|
expected_results = [True, False, False, True, True, True, True, True,
|
|
True]
|
|
self.assertEqual(expected_results, has_pci_request_id)
|
|
# Make sure only the trusted VF has the 'trusted' tag set in the spec.
|
|
for pci_req in pci_requests.requests:
|
|
spec = pci_req.spec[0]
|
|
if spec[pci_request.PCI_NET_TAG] == 'physnet4':
|
|
# trusted should be true in the spec for this request
|
|
self.assertIn(pci_request.PCI_TRUSTED_TAG, spec)
|
|
self.assertEqual('True', spec[pci_request.PCI_TRUSTED_TAG])
|
|
else:
|
|
self.assertNotIn(pci_request.PCI_TRUSTED_TAG, spec)
|
|
|
|
# Only remote-managed ports must have the remote_managed tag set
|
|
# to True.
|
|
for pci_req in pci_requests.requests:
|
|
spec = pci_req.spec[0]
|
|
if pci_req.requester_id == uuids.portid_remote_managed:
|
|
self.assertEqual('True',
|
|
spec[pci_request.PCI_REMOTE_MANAGED_TAG])
|
|
else:
|
|
self.assertEqual('False',
|
|
spec[pci_request.PCI_REMOTE_MANAGED_TAG])
|
|
|
|
# Only SRIOV ports and those with a resource_request will have
|
|
# pci_req.requester_id.
|
|
self.assertEqual(
|
|
[uuids.portid_1, uuids.portid_3, uuids.portid_4, uuids.portid_5,
|
|
uuids.trusted_port, uuids.portid_vdpa,
|
|
uuids.portid_remote_managed],
|
|
[pci_req.requester_id for pci_req in pci_requests.requests])
|
|
|
|
self.assertCountEqual(
|
|
['physnet1', 'physnet2', 'physnet3', 'physnet4'],
|
|
network_metadata.physnets)
|
|
self.assertTrue(network_metadata.tunneled)
|
|
mock_from_port_request.assert_has_calls([
|
|
mock.call(
|
|
context=None,
|
|
port_uuid=uuids.portid_2,
|
|
port_resource_request=mock.sentinel.resource_request1),
|
|
mock.call(
|
|
context=None,
|
|
port_uuid=uuids.trusted_port,
|
|
port_resource_request=mock.sentinel.resource_request2),
|
|
])
|
|
mock_has_extended_res_req.assert_called_once_with(
|
|
self.context, getclient.return_value)
|
|
|
|
@mock.patch(
|
|
'nova.accelerator.cyborg._CyborgClient.get_device_request_groups')
|
|
@mock.patch(
|
|
'nova.accelerator.cyborg._CyborgClient.get_device_profile_groups')
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch('nova.accelerator.cyborg.get_device_amount_of_dp_groups')
|
|
@mock.patch('nova.objects.request_spec.RequestGroup.from_port_request')
|
|
@mock.patch.object(neutronapi.API, "_get_port_vnic_info")
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_requests_with_arq(self, getclient,
|
|
mock_get_port_vnic_info, mock_from_port_request,
|
|
mock_get_device_num,
|
|
mock_get_physnet_tunneled_info,
|
|
mock_get_dp_group,
|
|
mock_get_rg):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects = [
|
|
objects.NetworkRequest(port_id=uuids.portid_1)
|
|
])
|
|
|
|
mock_get_port_vnic_info.side_effect = [
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT, None, 'netN',
|
|
None, None, 'smat_nic'),
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT_PHYSICAL, None,
|
|
'netN', None, None, 'smat_nic')
|
|
]
|
|
mock_get_physnet_tunneled_info.side_effect = [
|
|
('physnet1', False), ('physnet2', False)
|
|
]
|
|
rg = objects.RequestGroup(requester_id='request_group_1')
|
|
rg.add_resource(rclass='CUSTOM_NIC_TRAIT', amount=1)
|
|
mock_get_rg.return_value = [rg]
|
|
mock_get_device_num.return_value = 1
|
|
result = self.api.create_resource_requests(
|
|
self.context, requested_networks, pci_requests=None)
|
|
|
|
network_metadata, port_resource_requests, _ = result
|
|
mock_get_dp_group.assert_called_once_with('smat_nic')
|
|
mock_get_physnet_tunneled_info.assert_called_once_with(
|
|
self.context, mock.ANY, 'netN')
|
|
self.assertEqual({'physnet1'}, network_metadata.physnets)
|
|
self.assertEqual([rg], port_resource_requests)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch.object(neutronapi.API, "_get_port_vnic_info")
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_requests_with_arq_excption(self, getclient,
|
|
mock_get_port_vnic_info, mock_get_physnet_tunneled_info):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects = [
|
|
objects.NetworkRequest(port_id=uuids.portid_1)
|
|
])
|
|
|
|
mock_get_port_vnic_info.side_effect = [
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT, None, 'netN',
|
|
None, None, None)
|
|
]
|
|
mock_get_physnet_tunneled_info.side_effect = [
|
|
('physnet1', False), ('physnet2', False)
|
|
]
|
|
|
|
self.assertRaises(exception.DeviceProfileError,
|
|
self.api.create_resource_requests,
|
|
self.context, requested_networks, pci_requests=None)
|
|
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch('nova.accelerator.cyborg.get_device_amount_of_dp_groups')
|
|
@mock.patch.object(neutronapi.API, "_get_port_vnic_info")
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_requests_arq_reject_multi_devices(self,
|
|
getclient,
|
|
mock_get_port_vnic_info,
|
|
mock_get_device_num,
|
|
mock_get_physnet_tunneled_info):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects = [
|
|
objects.NetworkRequest(port_id=uuids.portid_1)
|
|
])
|
|
|
|
mock_get_port_vnic_info.side_effect = [
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT, None, 'netN',
|
|
None, None, 'smat_nic'),
|
|
(model.VNIC_TYPE_ACCELERATOR_DIRECT_PHYSICAL, None,
|
|
'netN', None, None, 'smat_nic')
|
|
]
|
|
mock_get_physnet_tunneled_info.side_effect = [
|
|
('physnet1', False), ('physnet2', False)
|
|
]
|
|
|
|
mock_get_device_num.return_value = 2
|
|
self.assertRaises(exception.DeviceProfileError,
|
|
self.api.create_resource_requests,
|
|
self.context, requested_networks, pci_requests=None)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.support_create_with_resource_request',
|
|
new=mock.Mock(return_value=True))
|
|
@mock.patch.object(
|
|
neutronapi.API, 'has_extended_resource_request_extension',
|
|
return_value=True)
|
|
@mock.patch(
|
|
'nova.objects.request_spec.RequestLevelParams.extend_with'
|
|
)
|
|
@mock.patch(
|
|
'nova.objects.request_spec.RequestLevelParams.from_port_request'
|
|
)
|
|
@mock.patch(
|
|
'nova.objects.request_spec.RequestGroup.from_extended_port_request')
|
|
@mock.patch.object(neutronapi.API, '_get_physnet_tunneled_info')
|
|
@mock.patch.object(neutronapi.API, "_get_port_vnic_info")
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_create_resource_request_extended(
|
|
self, getclient, mock_get_port_vnic_info,
|
|
mock_get_physnet_tunneled_info, mock_from_port_request,
|
|
mock_req_lvl_param, mock_extened_req_lvl_param,
|
|
mock_has_extended_res_req
|
|
):
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[
|
|
objects.NetworkRequest(port_id=uuids.portid_1),
|
|
objects.NetworkRequest(port_id=uuids.portid_2),
|
|
objects.NetworkRequest(port_id=uuids.portid_3),
|
|
])
|
|
pci_requests = objects.InstancePCIRequests(requests=[])
|
|
mock_get_port_vnic_info.side_effect = [
|
|
(model.VNIC_TYPE_NORMAL, None, 'netN',
|
|
mock.sentinel.resource_request1, None, None),
|
|
(model.VNIC_TYPE_NORMAL, None, 'netN',
|
|
mock.sentinel.resource_request2, None, None),
|
|
(model.VNIC_TYPE_NORMAL, None, 'netN', None, None, None),
|
|
]
|
|
# _get_physnet_tunneled_info should be called for every NetworkRequest
|
|
mock_get_physnet_tunneled_info.side_effect = [
|
|
('physnet1', False), ('physnet2', False), ('physnet3', False)]
|
|
api = neutronapi.API()
|
|
|
|
# Simulate that both port1 and port2 have such an extended resource
|
|
# request that is resolved to more than one request groups, but port3
|
|
# has no request
|
|
mock_from_port_request.side_effect = [
|
|
[
|
|
mock.sentinel.port1_request_group1,
|
|
mock.sentinel.port1_request_group2,
|
|
],
|
|
[
|
|
mock.sentinel.port2_request_group1,
|
|
mock.sentinel.port2_request_group2,
|
|
],
|
|
]
|
|
# also both port1 and port2 has same subtree params
|
|
mock_req_lvl_param.side_effect = [
|
|
mock.sentinel.port1_req_lvl_param,
|
|
mock.sentinel.port2_req_lvl_param,
|
|
]
|
|
|
|
result = api.create_resource_requests(
|
|
self.context, requested_networks, pci_requests)
|
|
network_metadata, port_resource_requests, req_lvl_param = result
|
|
|
|
# assert that all the request groups are collected from both ports
|
|
self.assertEqual(
|
|
[
|
|
mock.sentinel.port1_request_group1,
|
|
mock.sentinel.port1_request_group2,
|
|
mock.sentinel.port2_request_group1,
|
|
mock.sentinel.port2_request_group2,
|
|
],
|
|
port_resource_requests)
|
|
# the same subtree requests are combined from the two ports
|
|
mock_req_lvl_param.assert_has_calls(
|
|
[
|
|
mock.call(
|
|
port_resource_request=mock.sentinel.resource_request1),
|
|
mock.call(
|
|
port_resource_request=mock.sentinel.resource_request2),
|
|
]
|
|
|
|
)
|
|
mock_extened_req_lvl_param.assert_has_calls(
|
|
[
|
|
mock.call(mock.sentinel.port1_req_lvl_param),
|
|
mock.call(mock.sentinel.port2_req_lvl_param),
|
|
]
|
|
)
|
|
self.assertIsInstance(req_lvl_param, objects.RequestLevelParams)
|
|
|
|
mock_from_port_request.assert_has_calls([
|
|
mock.call(
|
|
context=None,
|
|
port_resource_request=mock.sentinel.resource_request1),
|
|
mock.call(
|
|
context=None,
|
|
port_resource_request=mock.sentinel.resource_request2),
|
|
])
|
|
mock_has_extended_res_req.assert_called_once_with(
|
|
self.context, getclient.return_value)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_associate_floating_ip_conflict(self, mock_get_client):
|
|
"""Tests that if Neutron raises a Conflict we handle it and re-raise
|
|
as a nova-specific exception.
|
|
"""
|
|
mock_get_client.return_value.update_floatingip.side_effect = (
|
|
exceptions.Conflict(
|
|
"Cannot associate floating IP 172.24.5.15 "
|
|
"(60a8f00b-4404-4518-ad66-00448a155904) with port "
|
|
"95ee1ffb-6d41-447d-a90e-b6ce5d9c92fa using fixed IP "
|
|
"10.1.0.9, as that fixed IP already has a floating IP on "
|
|
"external network bdcda645-f612-40ab-a956-0d95af42cf7c.")
|
|
)
|
|
with test.nested(
|
|
mock.patch.object(
|
|
self.api, '_get_port_id_by_fixed_address',
|
|
return_value='95ee1ffb-6d41-447d-a90e-b6ce5d9c92fa'),
|
|
mock.patch.object(
|
|
self.api, '_get_floating_ip_by_address',
|
|
return_value={'id': uuids.floating_ip_id})
|
|
) as (
|
|
_get_floating_ip_by_address, _get_port_id_by_fixed_address
|
|
):
|
|
instance = fake_instance.fake_instance_obj(
|
|
self.context, uuid='2a2200ec-02fe-484e-885b-9bae7b21ecba')
|
|
self.assertRaises(exception.FloatingIpAssociateFailed,
|
|
self.api.associate_floating_ip,
|
|
self.context, instance,
|
|
'172.24.5.15', '10.1.0.9')
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
@mock.patch('nova.network.neutron.LOG.warning')
|
|
@mock.patch('nova.network.neutron.update_instance_cache_with_nw_info')
|
|
def test_associate_floating_ip_refresh_error_trap(self, mock_update_cache,
|
|
mock_log_warning,
|
|
mock_get_client):
|
|
"""Tests that when _update_inst_info_cache_for_disassociated_fip
|
|
raises an exception, associate_floating_ip traps and logs it but
|
|
does not re-raise.
|
|
"""
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt)
|
|
floating_addr = '172.24.5.15'
|
|
fixed_addr = '10.1.0.9'
|
|
fip = {'id': uuids.floating_ip_id, 'port_id': uuids.old_port_id}
|
|
# Setup the mocks.
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_get_port_id_by_fixed_address',
|
|
return_value=uuids.new_port_id),
|
|
mock.patch.object(self.api, '_get_floating_ip_by_address',
|
|
return_value=fip),
|
|
mock.patch.object(self.api,
|
|
'_update_inst_info_cache_for_disassociated_fip',
|
|
side_effect=exception.PortNotFound(
|
|
port_id=uuids.old_port_id))
|
|
) as (
|
|
_get_port_id_by_fixed_address,
|
|
_get_floating_ip_by_address,
|
|
_update_inst_info_cache_for_disassociated_fip
|
|
):
|
|
# Run the code.
|
|
self.api.associate_floating_ip(
|
|
ctxt, instance, floating_addr, fixed_addr)
|
|
# Assert the calls.
|
|
mock_get_client.assert_called_once_with(ctxt)
|
|
mock_client = mock_get_client.return_value
|
|
_get_port_id_by_fixed_address.assert_called_once_with(
|
|
mock_client, instance, fixed_addr)
|
|
_get_floating_ip_by_address.assert_called_once_with(
|
|
mock_client, floating_addr)
|
|
mock_client.update_floatingip.assert_called_once_with(
|
|
uuids.floating_ip_id, test.MatchType(dict))
|
|
_update_inst_info_cache_for_disassociated_fip.assert_called_once_with(
|
|
ctxt, instance, mock_client, fip)
|
|
mock_log_warning.assert_called_once()
|
|
self.assertIn('An error occurred while trying to refresh the '
|
|
'network info cache for an instance associated '
|
|
'with port', mock_log_warning.call_args[0][0])
|
|
mock_update_cache.assert_called_once_with( # from @refresh_cache
|
|
self.api, ctxt, instance, nw_info=None)
|
|
|
|
@mock.patch('nova.network.neutron.update_instance_cache_with_nw_info')
|
|
def test_update_inst_info_cache_for_disassociated_fip_other_cell(
|
|
self, mock_update_cache):
|
|
"""Tests a scenario where a floating IP is associated to an instance
|
|
in another cell from the one in which it's currently associated
|
|
and the network info cache on the original instance is refreshed.
|
|
"""
|
|
ctxt = context.get_context()
|
|
mock_client = mock.Mock()
|
|
new_instance = fake_instance.fake_instance_obj(ctxt)
|
|
cctxt = context.get_context()
|
|
old_instance = fake_instance.fake_instance_obj(cctxt)
|
|
fip = {'id': uuids.floating_ip_id,
|
|
'port_id': uuids.old_port_id,
|
|
'floating_ip_address': '172.24.5.15'}
|
|
# Setup the mocks.
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_show_port',
|
|
return_value={
|
|
'device_id': old_instance.uuid}),
|
|
mock.patch.object(self.api, '_get_instance_by_uuid_using_api_db',
|
|
return_value=old_instance)
|
|
) as (
|
|
_show_port, _get_instance_by_uuid_using_api_db
|
|
):
|
|
# Run the code.
|
|
self.api._update_inst_info_cache_for_disassociated_fip(
|
|
ctxt, new_instance, mock_client, fip)
|
|
# Assert the calls.
|
|
_show_port.assert_called_once_with(
|
|
ctxt, uuids.old_port_id, neutron_client=mock_client)
|
|
_get_instance_by_uuid_using_api_db.assert_called_once_with(
|
|
ctxt, old_instance.uuid)
|
|
mock_update_cache.assert_called_once_with(
|
|
self.api, cctxt, old_instance)
|
|
|
|
@mock.patch('nova.network.neutron.LOG.info')
|
|
@mock.patch('nova.network.neutron.update_instance_cache_with_nw_info')
|
|
def test_update_inst_info_cache_for_disassociated_fip_inst_not_found(
|
|
self, mock_update_cache, mock_log_info):
|
|
"""Tests the case that a floating IP is re-associated to an instance
|
|
in another cell but the original instance cannot be found.
|
|
"""
|
|
ctxt = context.get_context()
|
|
mock_client = mock.Mock()
|
|
new_instance = fake_instance.fake_instance_obj(ctxt)
|
|
fip = {'id': uuids.floating_ip_id,
|
|
'port_id': uuids.old_port_id,
|
|
'floating_ip_address': '172.24.5.15'}
|
|
# Setup the mocks.
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_show_port',
|
|
return_value={
|
|
'device_id': uuids.original_inst_uuid}),
|
|
mock.patch.object(self.api,
|
|
'_get_instance_by_uuid_using_api_db',
|
|
return_value=None)
|
|
) as (
|
|
_show_port, _get_instance_by_uuid_using_api_db
|
|
):
|
|
# Run the code.
|
|
self.api._update_inst_info_cache_for_disassociated_fip(
|
|
ctxt, new_instance, mock_client, fip)
|
|
# Assert the calls.
|
|
_show_port.assert_called_once_with(
|
|
ctxt, uuids.old_port_id, neutron_client=mock_client)
|
|
_get_instance_by_uuid_using_api_db.assert_called_once_with(
|
|
ctxt, uuids.original_inst_uuid)
|
|
mock_update_cache.assert_not_called()
|
|
self.assertEqual(2, mock_log_info.call_count)
|
|
self.assertIn('If the instance still exists, its info cache may '
|
|
'be healed automatically.',
|
|
mock_log_info.call_args[0][0])
|
|
|
|
@mock.patch('nova.objects.Instance.get_by_uuid')
|
|
@mock.patch('nova.objects.InstanceMapping.get_by_instance_uuid')
|
|
def test_get_instance_by_uuid_using_api_db_current_cell(
|
|
self, mock_get_map, mock_get_inst):
|
|
"""Tests that _get_instance_by_uuid_using_api_db finds the
|
|
instance in the cell currently targeted by the context.
|
|
"""
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt)
|
|
mock_get_inst.return_value = instance
|
|
inst = self.api._get_instance_by_uuid_using_api_db(ctxt, instance.uuid)
|
|
self.assertIs(inst, instance)
|
|
mock_get_inst.assert_called_once_with(ctxt, instance.uuid)
|
|
mock_get_map.assert_not_called()
|
|
|
|
@mock.patch('nova.objects.Instance.get_by_uuid')
|
|
@mock.patch('nova.objects.InstanceMapping.get_by_instance_uuid')
|
|
def test_get_instance_by_uuid_using_api_db_other_cell(
|
|
self, mock_get_map, mock_get_inst):
|
|
"""Tests that _get_instance_by_uuid_using_api_db finds the
|
|
instance in another cell different from the currently targeted context.
|
|
"""
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt)
|
|
mock_get_map.return_value = objects.InstanceMapping(
|
|
cell_mapping=objects.CellMapping(
|
|
uuid=uuids.cell_mapping_uuid,
|
|
database_connection=
|
|
self.cell_mappings['cell1'].database_connection,
|
|
transport_url='none://fake'))
|
|
# Mock get_by_uuid to not find the instance in the first call, but
|
|
# do find it in the second. Target the instance context as well so
|
|
# we can assert that we used a different context for the 2nd call.
|
|
|
|
def stub_inst_get_by_uuid(_context, instance_uuid, *args, **kwargs):
|
|
if not mock_get_map.called:
|
|
# First call, raise InstanceNotFound.
|
|
self.assertIs(_context, ctxt)
|
|
raise exception.InstanceNotFound(instance_id=instance_uuid)
|
|
# Else return the instance with a newly targeted context.
|
|
self.assertIsNot(_context, ctxt)
|
|
instance._context = _context
|
|
return instance
|
|
mock_get_inst.side_effect = stub_inst_get_by_uuid
|
|
|
|
inst = self.api._get_instance_by_uuid_using_api_db(ctxt, instance.uuid)
|
|
# The instance's internal context should still be targeted and not
|
|
# the original context.
|
|
self.assertIsNot(inst._context, ctxt)
|
|
self.assertIsNotNone(inst._context.db_connection)
|
|
mock_get_map.assert_called_once_with(ctxt, instance.uuid)
|
|
mock_get_inst.assert_has_calls([
|
|
mock.call(ctxt, instance.uuid),
|
|
mock.call(inst._context, instance.uuid)])
|
|
|
|
@mock.patch('nova.objects.Instance.get_by_uuid',
|
|
side_effect=exception.InstanceNotFound(instance_id=uuids.inst))
|
|
@mock.patch('nova.objects.InstanceMapping.get_by_instance_uuid')
|
|
def test_get_instance_by_uuid_using_api_db_other_cell_never_found(
|
|
self, mock_get_map, mock_get_inst):
|
|
"""Tests that _get_instance_by_uuid_using_api_db does not find the
|
|
instance in either the current cell or another cell.
|
|
"""
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt, uuid=uuids.inst)
|
|
mock_get_map.return_value = objects.InstanceMapping(
|
|
cell_mapping=objects.CellMapping(
|
|
uuid=uuids.cell_mapping_uuid,
|
|
database_connection=
|
|
self.cell_mappings['cell1'].database_connection,
|
|
transport_url='none://fake'))
|
|
self.assertIsNone(
|
|
self.api._get_instance_by_uuid_using_api_db(ctxt, instance.uuid))
|
|
mock_get_map.assert_called_once_with(ctxt, instance.uuid)
|
|
mock_get_inst.assert_has_calls([
|
|
mock.call(ctxt, instance.uuid),
|
|
mock.call(test.MatchType(context.RequestContext), instance.uuid)])
|
|
|
|
@mock.patch('nova.objects.Instance.get_by_uuid',
|
|
side_effect=exception.InstanceNotFound(instance_id=uuids.inst))
|
|
@mock.patch('nova.objects.InstanceMapping.get_by_instance_uuid',
|
|
side_effect=exception.InstanceMappingNotFound(uuid=uuids.inst))
|
|
def test_get_instance_by_uuid_using_api_db_other_cell_map_not_found(
|
|
self, mock_get_map, mock_get_inst):
|
|
"""Tests that _get_instance_by_uuid_using_api_db does not find an
|
|
instance mapping for the instance.
|
|
"""
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt, uuid=uuids.inst)
|
|
self.assertIsNone(
|
|
self.api._get_instance_by_uuid_using_api_db(ctxt, instance.uuid))
|
|
mock_get_inst.assert_called_once_with(ctxt, instance.uuid)
|
|
mock_get_map.assert_called_once_with(ctxt, instance.uuid)
|
|
|
|
@mock.patch('nova.network.neutron.get_client',
|
|
new_callable=mock.NonCallableMock) # asserts not called
|
|
def test_migrate_instance_start_no_binding_ext(self, get_client_mock):
|
|
"""Tests that migrate_instance_start exits early if neutron doesn't
|
|
have the binding-extended API extension.
|
|
"""
|
|
with mock.patch.object(self.api, 'has_port_binding_extension',
|
|
return_value=False):
|
|
self.api.migrate_instance_start(
|
|
self.context, mock.sentinel.instance, {})
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_migrate_instance_start_activate(self, get_client_mock):
|
|
"""Tests the happy path for migrate_instance_start where the binding
|
|
for the port(s) attached to the instance are activated on the
|
|
destination host.
|
|
"""
|
|
binding = {'binding': {'status': 'INACTIVE'}}
|
|
mocked_client = get_client_mock.return_value
|
|
mocked_client.show_port_binding.return_value = binding
|
|
# Just create a simple instance with a single port.
|
|
instance = objects.Instance(info_cache=objects.InstanceInfoCache(
|
|
network_info=model.NetworkInfo([model.VIF(uuids.port_id)])))
|
|
migration = objects.Migration(
|
|
source_compute='source', dest_compute='dest')
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
self.api.migrate_instance_start(
|
|
self.context, instance, migration)
|
|
|
|
mocked_client.show_port_binding.assert_called_once_with(
|
|
uuids.port_id, 'dest')
|
|
mocked_client.activate_port_binding.assert_called_once_with(
|
|
uuids.port_id, 'dest')
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_migrate_instance_start_already_active(self, get_client_mock):
|
|
"""Tests the case that the destination host port binding is already
|
|
ACTIVE when migrate_instance_start is called so we don't try to
|
|
activate it again, which would result in a 409 from Neutron.
|
|
"""
|
|
binding = {'binding': {'status': 'ACTIVE'}}
|
|
mocked_client = get_client_mock.return_value
|
|
mocked_client.show_port_binding.return_value = binding
|
|
# Just create a simple instance with a single port.
|
|
instance = objects.Instance(info_cache=objects.InstanceInfoCache(
|
|
network_info=model.NetworkInfo([model.VIF(uuids.port_id)])))
|
|
migration = objects.Migration(
|
|
source_compute='source', dest_compute='dest')
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
self.api.migrate_instance_start(
|
|
self.context, instance, migration)
|
|
|
|
mocked_client.show_port_binding.assert_called_once_with(
|
|
uuids.port_id, 'dest')
|
|
mocked_client.activate_port_binding.assert_not_called()
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_migrate_instance_start_no_bindings(self, get_client_mock):
|
|
"""Tests the case that migrate_instance_start is running against new
|
|
enough neutron for the binding-extended API but the ports don't have
|
|
a binding resource against the destination host, so no activation
|
|
happens.
|
|
"""
|
|
NeutronNotFound = exceptions.NeutronClientException(status_code=404)
|
|
mocked_client = get_client_mock.return_value
|
|
mocked_client.show_port_binding.side_effect = NeutronNotFound
|
|
# Create an instance with two ports so we can test the short circuit
|
|
# when we find that the first port doesn't have a dest host binding.
|
|
instance = objects.Instance(info_cache=objects.InstanceInfoCache(
|
|
network_info=model.NetworkInfo([
|
|
model.VIF(uuids.port1), model.VIF(uuids.port2)])))
|
|
migration = objects.Migration(
|
|
source_compute='source', dest_compute='dest')
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
self.api.migrate_instance_start(
|
|
self.context, instance, migration)
|
|
|
|
mocked_client.show_port_binding.assert_called_once_with(
|
|
uuids.port1, 'dest')
|
|
mocked_client.activate_port_binding.assert_not_called()
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_migrate_instance_start_get_error(self, get_client_mock):
|
|
"""Tests the case that migrate_instance_start is running against new
|
|
enough neutron for the binding-extended API but getting the port
|
|
binding information results in an error response from neutron.
|
|
"""
|
|
NeutronError = exceptions.NeutronClientException(status_code=500)
|
|
mocked_client = get_client_mock.return_value
|
|
mocked_client.show_port_binding.side_effect = NeutronError
|
|
instance = objects.Instance(info_cache=objects.InstanceInfoCache(
|
|
network_info=model.NetworkInfo([
|
|
model.VIF(uuids.port1), model.VIF(uuids.port2)])))
|
|
migration = objects.Migration(
|
|
source_compute='source', dest_compute='dest')
|
|
|
|
with mock.patch.object(
|
|
self.api, 'has_port_binding_extension', return_value=True,
|
|
):
|
|
self.api.migrate_instance_start(
|
|
self.context, instance, migration)
|
|
|
|
self.assertEqual(2, mocked_client.show_port_binding.call_count)
|
|
mocked_client.show_port_binding.assert_has_calls([
|
|
mock.call(uuids.port1, 'dest'),
|
|
mock.call(uuids.port2, 'dest'),
|
|
])
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_requested_resource_for_instance_no_resource_request(
|
|
self, mock_get_client):
|
|
mock_client = mock_get_client.return_value
|
|
|
|
ports = {'ports': [
|
|
{
|
|
'id': uuids.port1,
|
|
'device_id': uuids.isnt1,
|
|
}
|
|
]}
|
|
mock_client.list_ports.return_value = ports
|
|
|
|
request_groups, req_lvl_params = (
|
|
self.api.get_requested_resource_for_instance(
|
|
self.context, uuids.inst1)
|
|
)
|
|
|
|
mock_client.list_ports.assert_called_with(
|
|
device_id=uuids.inst1, fields=['id', 'resource_request'])
|
|
self.assertEqual([], request_groups)
|
|
self.assertEqual([], req_lvl_params.same_subtree)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_requested_resource_for_instance_no_ports(
|
|
self, mock_get_client
|
|
):
|
|
mock_client = mock_get_client.return_value
|
|
|
|
ports = {'ports': []}
|
|
mock_client.list_ports.return_value = ports
|
|
|
|
request_groups, req_lvl_params = (
|
|
self.api.get_requested_resource_for_instance(
|
|
self.context, uuids.inst1)
|
|
)
|
|
|
|
mock_client.list_ports.assert_called_with(
|
|
device_id=uuids.inst1, fields=['id', 'resource_request'])
|
|
self.assertEqual([], request_groups)
|
|
self.assertEqual([], req_lvl_params.same_subtree)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_requested_resource_for_instance_with_multiple_ports(
|
|
self, mock_get_client):
|
|
mock_client = mock_get_client.return_value
|
|
|
|
ports = {'ports': [
|
|
{
|
|
'id': uuids.port1,
|
|
'device_id': uuids.isnt1,
|
|
'resource_request': {
|
|
'resources': {'NET_BW_EGR_KILOBIT_PER_SEC': 10000}}
|
|
},
|
|
{
|
|
'id': uuids.port2,
|
|
'device_id': uuids.isnt1,
|
|
'resource_request': {}
|
|
},
|
|
]}
|
|
mock_client.list_ports.return_value = ports
|
|
|
|
request_groups, req_lvl_params = (
|
|
self.api.get_requested_resource_for_instance(
|
|
self.context, uuids.inst1)
|
|
)
|
|
|
|
mock_client.list_ports.assert_called_with(
|
|
device_id=uuids.inst1, fields=['id', 'resource_request'])
|
|
self.assertEqual(1, len(request_groups))
|
|
self.assertEqual(
|
|
{'NET_BW_EGR_KILOBIT_PER_SEC': 10000},
|
|
request_groups[0].resources)
|
|
self.assertEqual(
|
|
uuids.port1,
|
|
request_groups[0].requester_id)
|
|
self.assertEqual([], req_lvl_params.same_subtree)
|
|
|
|
mock_get_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch(
|
|
"nova.network.neutron.API.has_extended_resource_request_extension",
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_get_requested_resource_for_instance_with_multiple_ports_extended(
|
|
self, mock_get_client):
|
|
mock_client = mock_get_client.return_value
|
|
|
|
ports = {'ports': [
|
|
{
|
|
'id': uuids.port1,
|
|
'device_id': uuids.isnt1,
|
|
'resource_request': {
|
|
'request_groups': [
|
|
{
|
|
'id': uuids.group1,
|
|
'resources': {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 10000
|
|
}
|
|
},
|
|
{
|
|
'id': uuids.group2,
|
|
'resources': {
|
|
'NET_KILOPACKET_PER_SEC': 100
|
|
}
|
|
}
|
|
],
|
|
'same_subtree': [uuids.group1, uuids.group2],
|
|
}
|
|
},
|
|
{
|
|
'id': uuids.port2,
|
|
'device_id': uuids.isnt1,
|
|
'resource_request': {
|
|
'request_groups': [
|
|
{
|
|
'id': uuids.group3,
|
|
'resources': {
|
|
'NET_BW_EGR_KILOBIT_PER_SEC': 20000
|
|
}
|
|
},
|
|
],
|
|
'same_subtree': [uuids.group3],
|
|
}
|
|
},
|
|
{
|
|
'id': uuids.port3,
|
|
'device_id': uuids.isnt1,
|
|
'resource_request': {}
|
|
},
|
|
]}
|
|
mock_client.list_ports.return_value = ports
|
|
|
|
request_groups, req_lvl_params = (
|
|
self.api.get_requested_resource_for_instance(
|
|
self.context, uuids.inst1)
|
|
)
|
|
|
|
mock_client.list_ports.assert_called_with(
|
|
device_id=uuids.inst1, fields=['id', 'resource_request'])
|
|
self.assertEqual(3, len(request_groups))
|
|
self.assertEqual(
|
|
{'NET_BW_EGR_KILOBIT_PER_SEC': 10000},
|
|
request_groups[0].resources)
|
|
self.assertEqual(
|
|
uuids.group1,
|
|
request_groups[0].requester_id)
|
|
self.assertEqual(
|
|
{'NET_KILOPACKET_PER_SEC': 100},
|
|
request_groups[1].resources)
|
|
self.assertEqual(
|
|
uuids.group2,
|
|
request_groups[1].requester_id)
|
|
self.assertEqual(
|
|
{'NET_BW_EGR_KILOBIT_PER_SEC': 20000},
|
|
request_groups[2].resources)
|
|
self.assertEqual(
|
|
uuids.group3,
|
|
request_groups[2].requester_id)
|
|
|
|
mock_get_client.assert_called_once_with(self.context, admin=True)
|
|
self.assertEqual(
|
|
[[uuids.group1, uuids.group2], [uuids.group3]],
|
|
req_lvl_params.same_subtree,
|
|
)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_ids_for_network_no_segment_ext(self, mock_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=False,
|
|
):
|
|
self.assertEqual(
|
|
[], self.api.get_segment_ids_for_network(self.context,
|
|
uuids.network_id))
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_ids_for_network_passes(self, mock_client):
|
|
subnets = {'subnets': [{'segment_id': uuids.segment_id}]}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.list_subnets.return_value = subnets
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
res = self.api.get_segment_ids_for_network(
|
|
self.context, uuids.network_id)
|
|
self.assertEqual([uuids.segment_id], res)
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
mocked_client.list_subnets.assert_called_once_with(
|
|
network_id=uuids.network_id, fields='segment_id')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_ids_for_network_with_no_segments(self, mock_client):
|
|
subnets = {'subnets': [{'segment_id': None}]}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.list_subnets.return_value = subnets
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
res = self.api.get_segment_ids_for_network(
|
|
self.context, uuids.network_id)
|
|
self.assertEqual([], res)
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
mocked_client.list_subnets.assert_called_once_with(
|
|
network_id=uuids.network_id, fields='segment_id')
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_ids_for_network_fails(self, mock_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.list_subnets.side_effect = (
|
|
exceptions.NeutronClientException(status_code=404))
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
self.assertRaises(exception.InvalidRoutedNetworkConfiguration,
|
|
self.api.get_segment_ids_for_network,
|
|
self.context, uuids.network_id)
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_id_for_subnet_no_segment_ext(self, mock_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=False,
|
|
):
|
|
self.assertIsNone(
|
|
self.api.get_segment_id_for_subnet(self.context,
|
|
uuids.subnet_id))
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_id_for_subnet_passes(self, mock_client):
|
|
subnet = {'subnet': {'segment_id': uuids.segment_id}}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.show_subnet.return_value = subnet
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
res = self.api.get_segment_id_for_subnet(
|
|
self.context, uuids.subnet_id)
|
|
self.assertEqual(uuids.segment_id, res)
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
mocked_client.show_subnet.assert_called_once_with(uuids.subnet_id)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_id_for_subnet_with_no_segment(self, mock_client):
|
|
subnet = {'subnet': {}}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.show_subnet.return_value = subnet
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
self.assertIsNone(
|
|
self.api.get_segment_id_for_subnet(self.context,
|
|
uuids.subnet_id))
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_get_segment_id_for_subnet_fails(self, mock_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_client.return_value = mocked_client
|
|
mocked_client.show_subnet.side_effect = (
|
|
exceptions.NeutronClientException(status_code=404))
|
|
with mock.patch.object(
|
|
self.api, 'has_segment_extension', return_value=True,
|
|
):
|
|
self.assertRaises(exception.InvalidRoutedNetworkConfiguration,
|
|
self.api.get_segment_id_for_subnet,
|
|
self.context, uuids.subnet_id)
|
|
mock_client.assert_called_once_with(self.context, admin=True)
|
|
|
|
@mock.patch.object(neutronapi.LOG, 'debug')
|
|
def test_get_port_pci_dev(self, mock_debug):
|
|
fake_port = {'id': uuids.fake_port_id}
|
|
request = objects.InstancePCIRequest(requester_id=uuids.fake_port_id,
|
|
request_id=uuids.pci_request_id)
|
|
bad_request = objects.InstancePCIRequest(
|
|
requester_id=uuids.wrong_port_id)
|
|
device = objects.PciDevice(request_id=uuids.pci_request_id)
|
|
bad_device = objects.PciDevice(request_id=uuids.wrong_request_id)
|
|
# Test the happy path
|
|
instance = objects.Instance(
|
|
pci_requests=objects.InstancePCIRequests(requests=[request]),
|
|
pci_devices=objects.PciDeviceList(objects=[device]))
|
|
self.assertEqual(
|
|
device,
|
|
self.api._get_port_pci_dev(instance, fake_port))
|
|
# Test not finding the request
|
|
instance = objects.Instance(
|
|
pci_requests=objects.InstancePCIRequests(
|
|
requests=[objects.InstancePCIRequest(bad_request)]))
|
|
self.assertIsNone(
|
|
self.api._get_port_pci_dev(instance, fake_port))
|
|
mock_debug.assert_called_with('No PCI request found for port %s',
|
|
uuids.fake_port_id, instance=instance)
|
|
mock_debug.reset_mock()
|
|
# Test not finding the device
|
|
instance = objects.Instance(
|
|
pci_requests=objects.InstancePCIRequests(requests=[request]),
|
|
pci_devices=objects.PciDeviceList(objects=[bad_device]))
|
|
self.assertIsNone(
|
|
self.api._get_port_pci_dev(instance, fake_port))
|
|
mock_debug.assert_called_with('No PCI device found for request %s',
|
|
uuids.pci_request_id, instance=instance)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.'
|
|
'has_extended_resource_request_extension')
|
|
def test__has_resource_request(self, mock_has_extended_res_req):
|
|
# Old format, resource_request in None. That is Neutron current
|
|
# behavior if the port has no QoS policy associated.
|
|
mock_has_extended_res_req.return_value = False
|
|
port_no_res_req = {"resource_request": None}
|
|
self.assertFalse(self.api._has_resource_request(
|
|
self.context, port_no_res_req, neutron=None)
|
|
)
|
|
# Old format, there is resource_request key but no resource is
|
|
# requested. We should never get this from Neutron today but we
|
|
# actually get it if there are QoS policy assigned to the port but
|
|
# there is no rule in that policy requesting resources.
|
|
port_old_empty_res_req = {"resource_request": {}}
|
|
self.assertFalse(self.api._has_resource_request(
|
|
self.context, port_old_empty_res_req, neutron=None)
|
|
)
|
|
# Old format, and the port has resource request.
|
|
port_old_res_req = {
|
|
"resource_request": {
|
|
"resources": {
|
|
"NET_BW_IGR_KILOBIT_PER_SEC": 1000,
|
|
}
|
|
}
|
|
}
|
|
self.assertTrue(self.api._has_resource_request(
|
|
self.context, port_old_res_req, neutron=None)
|
|
)
|
|
# New format tests are starting here
|
|
mock_has_extended_res_req.return_value = True
|
|
# New format, port has no QoS policy assigned
|
|
port_no_res_req = {"resource_request": None}
|
|
self.assertFalse(self.api._has_resource_request(
|
|
self.context, port_no_res_req, neutron=None)
|
|
)
|
|
# New format, port has a resource_request key but no resource is
|
|
# requested. Neutron never sends this, it sends None instead. We keep
|
|
# this here for completeness as the code actually handle this properly.
|
|
port_new_empty_res_req = {
|
|
"resource_request": {
|
|
"request_groups": []
|
|
}
|
|
}
|
|
self.assertFalse(self.api._has_resource_request(
|
|
self.context, port_new_empty_res_req, neutron=None)
|
|
)
|
|
# New format, port has a resource request
|
|
port_new_res_req = {
|
|
"resource_request": {
|
|
"request_groups": [
|
|
{
|
|
"resources": {
|
|
"NET_KILOPACKET_PER_SEC": 1000
|
|
}
|
|
},
|
|
],
|
|
"same_subtree": [],
|
|
}
|
|
}
|
|
self.assertTrue(self.api._has_resource_request(
|
|
self.context, port_new_res_req, neutron=None)
|
|
)
|
|
|
|
@mock.patch(
|
|
"nova.network.neutron.API.has_extended_resource_request_extension",
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch("nova.network.neutron.get_client")
|
|
def test_get_binding_profile_allocation_no_request(
|
|
self, mock_get_client
|
|
):
|
|
mock_get_client.return_value.show_port.return_value = {
|
|
"port": {
|
|
}
|
|
}
|
|
self.assertIsNone(
|
|
self.api.get_binding_profile_allocation(
|
|
self.context, uuids.port_uuid, {}))
|
|
mock_get_client.assert_has_calls(
|
|
[
|
|
mock.call(self.context, admin=True),
|
|
mock.call(self.context),
|
|
]
|
|
)
|
|
|
|
@mock.patch(
|
|
"nova.network.neutron.API.has_extended_resource_request_extension",
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
@mock.patch("nova.network.neutron.get_client")
|
|
def test_get_binding_profile_allocation_legacy_request(
|
|
self, mock_get_client
|
|
):
|
|
mock_get_client.return_value.show_port.return_value = {
|
|
"port": {
|
|
"id": uuids.port_uuid,
|
|
"resource_request": {
|
|
"resources": {
|
|
"CUSTOM_FOO": 123,
|
|
}
|
|
},
|
|
}
|
|
}
|
|
self.assertEqual(
|
|
uuids.rp,
|
|
self.api.get_binding_profile_allocation(
|
|
self.context, uuids.port_uuid,
|
|
{
|
|
uuids.port_uuid: [uuids.rp]
|
|
}
|
|
)
|
|
)
|
|
|
|
@mock.patch(
|
|
"nova.network.neutron.API.has_extended_resource_request_extension",
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch("nova.network.neutron.get_client")
|
|
def test_get_binding_profile_allocation_extended_request(
|
|
self, mock_get_client
|
|
):
|
|
mock_get_client.return_value.show_port.return_value = {
|
|
"port": {
|
|
"id": uuids.port_uuid,
|
|
"resource_request": {
|
|
"request_groups": [
|
|
{
|
|
"id": uuids.group1,
|
|
"resources": {
|
|
"CUSTOM_FOO": 123,
|
|
}
|
|
},
|
|
{
|
|
"id": uuids.group2,
|
|
"resources": {
|
|
"CUSTOM_BAR": 321,
|
|
}
|
|
},
|
|
],
|
|
},
|
|
}
|
|
}
|
|
self.assertEqual(
|
|
{
|
|
uuids.group1: uuids.rp1,
|
|
uuids.group2: uuids.rp2,
|
|
},
|
|
self.api.get_binding_profile_allocation(
|
|
self.context, uuids.port_uuid,
|
|
{
|
|
uuids.group1: [uuids.rp1],
|
|
uuids.group2: [uuids.rp2],
|
|
uuids.non_port_related_group: [uuids.rp3],
|
|
}
|
|
)
|
|
)
|
|
|
|
|
|
class TestInstanceHasExtendedResourceRequest(TestAPIBase):
|
|
def setUp(self):
|
|
super().setUp()
|
|
patcher = mock.patch.object(neutronapi, 'get_client')
|
|
self.addCleanup(patcher.stop)
|
|
self.mock_client = patcher.start().return_value
|
|
self.extension = {
|
|
'extensions': [
|
|
{
|
|
'alias': constants.RESOURCE_REQUEST_GROUPS,
|
|
}
|
|
]
|
|
}
|
|
|
|
def test_no_extension(self):
|
|
self.mock_client.list_extensions.return_value = {
|
|
"extensions": []
|
|
}
|
|
|
|
self.assertFalse(
|
|
self.api.instance_has_extended_resource_request(uuids.instance))
|
|
|
|
self.mock_client.list_extensions.assert_called_once_with()
|
|
self.mock_client.list_ports.assert_not_called()
|
|
|
|
def test_no_port(self):
|
|
self.mock_client.list_extensions.return_value = self.extension
|
|
self.mock_client.list_ports.return_value = {
|
|
"ports": []
|
|
}
|
|
|
|
self.assertFalse(
|
|
self.api.instance_has_extended_resource_request(uuids.instance))
|
|
|
|
self.mock_client.list_extensions.assert_called_once_with()
|
|
self.mock_client.list_ports.assert_called_once_with(
|
|
device_id=uuids.instance,
|
|
fields=['resource_request'])
|
|
|
|
def test_port_without_request(self):
|
|
self.mock_client.list_extensions.return_value = self.extension
|
|
self.mock_client.list_ports.return_value = {
|
|
"ports": [
|
|
{"resource_request": {}}
|
|
]
|
|
}
|
|
|
|
self.assertFalse(
|
|
self.api.instance_has_extended_resource_request(uuids.instance))
|
|
|
|
self.mock_client.list_extensions.assert_called_once_with()
|
|
self.mock_client.list_ports.assert_called_once_with(
|
|
device_id=uuids.instance,
|
|
fields=['resource_request'])
|
|
|
|
def test_port_with_request(self):
|
|
self.mock_client.list_extensions.return_value = self.extension
|
|
self.mock_client.list_ports.return_value = {
|
|
"ports": [
|
|
{
|
|
"resource_request": {
|
|
"request_groups": [
|
|
{
|
|
"CUSTOM_FOO": 1000,
|
|
}
|
|
]
|
|
}
|
|
}
|
|
]
|
|
}
|
|
|
|
self.assertTrue(
|
|
self.api.instance_has_extended_resource_request(uuids.instance))
|
|
|
|
self.mock_client.list_extensions.assert_called_once_with()
|
|
self.mock_client.list_ports.assert_called_once_with(
|
|
device_id=uuids.instance,
|
|
fields=['resource_request'])
|
|
|
|
|
|
class TestAPIModuleMethods(test.NoDBTestCase):
|
|
|
|
def test_gather_port_ids_and_networks_wrong_params(self):
|
|
api = neutronapi.API()
|
|
|
|
# Test with networks not None and port_ids is None
|
|
self.assertRaises(exception.NovaException,
|
|
api._gather_port_ids_and_networks,
|
|
'fake_context', 'fake_instance',
|
|
[{'network': {'name': 'foo'}}], None)
|
|
|
|
# Test with networks is None and port_ids not None
|
|
self.assertRaises(exception.NovaException,
|
|
api._gather_port_ids_and_networks,
|
|
'fake_context', 'fake_instance',
|
|
None, ['list', 'of', 'port_ids'])
|
|
|
|
def test_ensure_requested_network_ordering_no_preference_ids(self):
|
|
networks = [1, 2, 3]
|
|
|
|
neutronapi._ensure_requested_network_ordering(
|
|
lambda x: x,
|
|
networks,
|
|
None)
|
|
|
|
def test_ensure_requested_network_ordering_no_preference_hashes(self):
|
|
networks = [{'id': 3}, {'id': 1}, {'id': 2}]
|
|
|
|
neutronapi._ensure_requested_network_ordering(
|
|
lambda x: x['id'],
|
|
networks,
|
|
None)
|
|
|
|
self.assertEqual(networks, [{'id': 3}, {'id': 1}, {'id': 2}])
|
|
|
|
def test_ensure_requested_network_ordering_with_preference(self):
|
|
networks = [{'id': 3}, {'id': 1}, {'id': 2}]
|
|
|
|
neutronapi._ensure_requested_network_ordering(
|
|
lambda x: x['id'],
|
|
networks,
|
|
[1, 2, 3])
|
|
|
|
self.assertEqual(networks, [{'id': 1}, {'id': 2}, {'id': 3}])
|
|
|
|
@mock.patch('nova.network.neutron.LOG.info')
|
|
@mock.patch('nova.network.neutron.LOG.exception')
|
|
@mock.patch('nova.objects.instance_info_cache.InstanceInfoCache.save')
|
|
def test_update_instance_cache_with_nw_info_not_found(self, mock_save,
|
|
mock_log_exc,
|
|
mock_log_info):
|
|
"""Tests that an attempt to update (save) the instance info cache will
|
|
not log a traceback but will reraise the exception for caller handling.
|
|
"""
|
|
# Simulate the oslo.messaging created "<OriginalClass>_Remote" subclass
|
|
# type we'll be catching.
|
|
class InstanceNotFound_Remote(exception.InstanceNotFound):
|
|
|
|
def __init__(self, message=None, **kwargs):
|
|
super().__init__(message=message, **kwargs)
|
|
|
|
# Simulate a long exception message containing tracebacks because
|
|
# oslo.messaging appends them.
|
|
message = 'Instance was not found.\n'.ljust(255, '*')
|
|
mock_save.side_effect = InstanceNotFound_Remote(message=message,
|
|
instance_id=uuids.inst)
|
|
api = neutronapi.API()
|
|
ctxt = context.get_context()
|
|
instance = fake_instance.fake_instance_obj(ctxt, uuid=uuids.i)
|
|
|
|
self.assertRaises(
|
|
exception.InstanceNotFound,
|
|
neutronapi.update_instance_cache_with_nw_info, api, ctxt, instance,
|
|
nw_info=model.NetworkInfo())
|
|
|
|
# Verify we didn't log exception at level ERROR.
|
|
mock_log_exc.assert_not_called()
|
|
# Verify exception message was truncated before logging it.
|
|
self.assertLessEqual(len(mock_log_info.call_args.args[1]), 255)
|
|
|
|
|
|
class TestAPIPortbinding(TestAPIBase):
|
|
|
|
def test_allocate_for_instance_portbinding(self):
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
1, bind_host_id=self.instance.get('host'))
|
|
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_populate_neutron_extension_values_binding(self, mock_get_client):
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mock_get_client.return_value = mocked_client
|
|
mocked_client.list_extensions.return_value = {'extensions': []}
|
|
host_id = 'my_host_id'
|
|
instance = {'host': host_id}
|
|
port_req_body = {'port': {}}
|
|
self.api._populate_neutron_extension_values(
|
|
self.context, instance, None, port_req_body,
|
|
bind_host_id=host_id)
|
|
self.assertEqual(host_id,
|
|
port_req_body['port'][
|
|
constants.BINDING_HOST_ID])
|
|
self.assertFalse(port_req_body['port'].get(
|
|
constants.BINDING_PROFILE))
|
|
mock_get_client.assert_called_once_with(mock.ANY)
|
|
mocked_client.list_extensions.assert_called_once_with()
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, '_get_vf_pci_device_profile',
|
|
new=mock.Mock(return_value={
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
}))
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_populate_neutron_extension_values_binding_sriov(
|
|
self, mock_get_instance_pci_devs, mock_get_pci_device_devspec):
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
port_req_body = {'port': {}}
|
|
pci_req_id = 'my_req_id'
|
|
pci_dev = {'vendor_id': '1377',
|
|
'product_id': '0047',
|
|
'address': '0000:0a:00.1',
|
|
'card_serial_number': None,
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
PciDevice = collections.namedtuple('PciDevice',
|
|
['vendor_id', 'product_id', 'address',
|
|
'card_serial_number', 'dev_type'])
|
|
mydev = PciDevice(**pci_dev)
|
|
profile = {'pci_vendor_info': '1377:0047',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'physnet1',
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
}
|
|
|
|
mock_get_instance_pci_devs.return_value = [mydev]
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_pci_device_devspec.return_value = devspec
|
|
|
|
self.api._populate_neutron_binding_profile(
|
|
instance, pci_req_id, port_req_body, None)
|
|
|
|
self.assertEqual(profile,
|
|
port_req_body['port'][
|
|
constants.BINDING_PROFILE])
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, '_get_vf_pci_device_profile',
|
|
new=mock.Mock(return_value= {
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
'card_serial_number': 'MT2113X00000',
|
|
})
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_populate_neutron_extension_values_binding_sriov_card_serial(
|
|
self, mock_get_instance_pci_devs, mock_get_pci_device_devspec):
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
port_req_body = {'port': {}}
|
|
pci_req_id = 'my_req_id'
|
|
pci_dev = {'vendor_id': 'a2d6',
|
|
'product_id': '15b3',
|
|
'address': '0000:0a:00.1',
|
|
'card_serial_number': 'MT2113X00000',
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
PciDevice = collections.namedtuple('PciDevice',
|
|
['vendor_id', 'product_id', 'address',
|
|
'card_serial_number', 'dev_type'])
|
|
mydev = PciDevice(**pci_dev)
|
|
profile = {'pci_vendor_info': 'a2d6:15b3',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'physnet1',
|
|
# card_serial_number is a property of the object obtained
|
|
# from extra_info.
|
|
'card_serial_number': 'MT2113X00000',
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
}
|
|
|
|
mock_get_instance_pci_devs.return_value = [mydev]
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_pci_device_devspec.return_value = devspec
|
|
self.api._populate_neutron_binding_profile(
|
|
instance, pci_req_id, port_req_body, None)
|
|
|
|
self.assertEqual(profile,
|
|
port_req_body['port'][
|
|
constants.BINDING_PROFILE])
|
|
|
|
def test_populate_neutron_extension_values_binding_arq(self):
|
|
host_id = 'my_host_id'
|
|
instance = {'host': host_id}
|
|
port_req_body = {'port': {}}
|
|
profile = {'arq_uuid': self.arqs[0]['uuid'],
|
|
'pci_slot': '0000:0c:0.0',
|
|
'physical_network': 'physicalnet1'}
|
|
self.api._populate_neutron_binding_profile(instance,
|
|
pci_request_id=None,
|
|
port_req_body=port_req_body,
|
|
port_arq=self.arqs[0])
|
|
|
|
self.assertEqual(
|
|
profile,
|
|
port_req_body['port'][constants.BINDING_PROFILE])
|
|
|
|
@mock.patch.object(neutronapi.API, '_refresh_neutron_extensions_cache')
|
|
@mock.patch('nova.accelerator.cyborg._CyborgClient.get_arqs_for_instance')
|
|
def test_populate_neutron_extension_values_with_arq(self,
|
|
mock_get_arq, mock_referesh_cache):
|
|
host_id = 'my_host_id'
|
|
instance = {'host': host_id}
|
|
port_req_body = {'port': {}}
|
|
profile = {'arq_uuid': self.arqs[0]['uuid'],
|
|
'pci_slot': '0000:0c:0.0',
|
|
'physical_network': 'physicalnet1'}
|
|
mock_referesh_cache.return_value = []
|
|
instance = self._fake_instance_object(self.instance)
|
|
mock_get_arq.return_value = self.arqs
|
|
self.api._populate_neutron_extension_values(
|
|
self.context, instance, None, port_req_body,
|
|
bind_host_id=host_id,
|
|
port_arq=self.arqs[0])
|
|
|
|
self.assertEqual(
|
|
profile,
|
|
port_req_body['port'][constants.BINDING_PROFILE])
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, '_get_vf_pci_device_profile',
|
|
new=mock.Mock(return_value= {
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
})
|
|
)
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_populate_neutron_extension_values_binding_sriov_with_cap(
|
|
self, mock_get_instance_pci_devs, mock_get_pci_device_devspec):
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
port_req_body = {'port': {
|
|
constants.BINDING_PROFILE: {
|
|
'capabilities': ['switchdev']}}}
|
|
pci_req_id = 'my_req_id'
|
|
pci_dev = {'vendor_id': '1377',
|
|
'product_id': '0047',
|
|
'address': '0000:0a:00.1',
|
|
'card_serial_number': None,
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
PciDevice = collections.namedtuple('PciDevice',
|
|
['vendor_id', 'product_id', 'address',
|
|
'card_serial_number', 'dev_type'])
|
|
mydev = PciDevice(**pci_dev)
|
|
profile = {'pci_vendor_info': '1377:0047',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'physical_network': 'physnet1',
|
|
'capabilities': ['switchdev'],
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
}
|
|
|
|
mock_get_instance_pci_devs.return_value = [mydev]
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_pci_device_devspec.return_value = devspec
|
|
|
|
self.api._populate_neutron_binding_profile(
|
|
instance, pci_req_id, port_req_body, None)
|
|
|
|
self.assertEqual(profile,
|
|
port_req_body['port'][
|
|
constants.BINDING_PROFILE])
|
|
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_populate_neutron_extension_values_binding_sriov_pf(
|
|
self, mock_get_instance_pci_devs, mock_get_devspec
|
|
):
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
port_req_body = {'port': {}}
|
|
|
|
pci_dev = objects.PciDevice(
|
|
request_id=uuids.pci_req,
|
|
address='0000:01:00',
|
|
parent_addr='0000:02:00',
|
|
vendor_id='8086',
|
|
product_id='154d',
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_PF,
|
|
extra_info={'mac_address': 'b4:96:91:34:f4:36'}
|
|
)
|
|
|
|
expected_profile = {
|
|
'pci_vendor_info': '8086:154d',
|
|
'pci_slot': '0000:01:00',
|
|
'physical_network': 'physnet1',
|
|
'device_mac_address': 'b4:96:91:34:f4:36',
|
|
}
|
|
|
|
mock_get_instance_pci_devs.return_value = [pci_dev]
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_devspec.return_value = devspec
|
|
|
|
self.api._populate_neutron_binding_profile(
|
|
instance, uuids.pci_req, port_req_body, None)
|
|
|
|
self.assertEqual(
|
|
expected_profile,
|
|
port_req_body['port'][constants.BINDING_PROFILE]
|
|
)
|
|
|
|
@mock.patch.object(
|
|
pci_utils, 'get_vf_num_by_pci_address',
|
|
new=mock.MagicMock(side_effect=(lambda vf_a: 1
|
|
if vf_a == '0000:0a:00.1' else None)))
|
|
@mock.patch.object(
|
|
pci_utils, 'get_mac_by_pci_address',
|
|
new=mock.MagicMock(side_effect=(lambda vf_a: {
|
|
'0000:0a:00.0': '52:54:00:1e:59:c6'}.get(vf_a)))
|
|
)
|
|
def test__get_vf_pci_device_profile(self):
|
|
pci_dev = {'vendor_id': 'a2d6',
|
|
'product_id': '15b3',
|
|
'address': '0000:0a:00.1',
|
|
'parent_addr': '0000:0a:00.0',
|
|
'card_serial_number': 'MT2113X00000',
|
|
'sriov_cap': {
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
},
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
PciDevice = collections.namedtuple('PciDevice',
|
|
['vendor_id', 'product_id', 'address',
|
|
'card_serial_number', 'sriov_cap',
|
|
'dev_type', 'parent_addr'])
|
|
mydev = PciDevice(**pci_dev)
|
|
self.assertEqual(self.api._get_vf_pci_device_profile(mydev),
|
|
{'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
'card_serial_number': 'MT2113X00000'})
|
|
|
|
@mock.patch.object(
|
|
neutronapi.API, '_get_vf_pci_device_profile',
|
|
new=mock.MagicMock(side_effect=(
|
|
lambda dev: {'0000:0a:00.1': {
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'vf_num': 1,
|
|
'card_serial_number': 'MT2113X00000',
|
|
}}.get(dev.address)
|
|
)))
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
def test__get_pci_device_profile_vf(self, mock_get_pci_device_devspec):
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_pci_device_devspec.return_value = devspec
|
|
|
|
pci_dev = {'vendor_id': 'a2d6',
|
|
'product_id': '15b3',
|
|
'address': '0000:0a:00.1',
|
|
'card_serial_number': 'MT2113X00000',
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
PciDevice = collections.namedtuple('PciDevice',
|
|
['vendor_id', 'product_id', 'address',
|
|
'card_serial_number', 'dev_type'])
|
|
mydev = PciDevice(**pci_dev)
|
|
|
|
self.assertEqual({'card_serial_number': 'MT2113X00000',
|
|
'pci_slot': '0000:0a:00.1',
|
|
'pci_vendor_info': 'a2d6:15b3',
|
|
'pf_mac_address': '52:54:00:1e:59:c6',
|
|
'physical_network': 'physnet1',
|
|
'vf_num': 1},
|
|
self.api._get_pci_device_profile(mydev))
|
|
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
def test__get_pci_device_profile_pf(self, mock_get_pci_device_devspec):
|
|
devspec = mock.Mock()
|
|
devspec.get_tags.return_value = {'physical_network': 'physnet1'}
|
|
mock_get_pci_device_devspec.return_value = devspec
|
|
|
|
pci_dev = objects.PciDevice(
|
|
request_id=uuids.pci_req,
|
|
address='0000:0a:00.0',
|
|
parent_addr='0000:02:00',
|
|
vendor_id='a2d6',
|
|
product_id='15b3',
|
|
dev_type=obj_fields.PciDeviceType.SRIOV_PF,
|
|
extra_info={
|
|
'capabilities': jsonutils.dumps(
|
|
{'card_serial_number': 'MT2113X00000'}),
|
|
'mac_address': 'b4:96:91:34:f4:36',
|
|
},
|
|
|
|
)
|
|
self.assertEqual(
|
|
{
|
|
'pci_slot': '0000:0a:00.0',
|
|
'pci_vendor_info': 'a2d6:15b3',
|
|
'physical_network': 'physnet1',
|
|
'device_mac_address': 'b4:96:91:34:f4:36',
|
|
},
|
|
self.api._get_pci_device_profile(pci_dev),
|
|
)
|
|
|
|
@mock.patch.object(pci_whitelist.Whitelist, 'get_devspec')
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_populate_neutron_extension_values_binding_sriov_fail(
|
|
self, mock_get_instance_pci_devs, mock_get_pci_device_devspec):
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
port_req_body = {'port': {}}
|
|
pci_req_id = 'my_req_id'
|
|
pci_objs = [objects.PciDevice(vendor_id='1377',
|
|
product_id='0047',
|
|
address='0000:0a:00.1',
|
|
compute_node_id=1,
|
|
request_id='1234567890')]
|
|
|
|
mock_get_instance_pci_devs.return_value = pci_objs
|
|
mock_get_pci_device_devspec.return_value = None
|
|
|
|
self.assertRaises(
|
|
exception.PciDeviceNotFound,
|
|
self.api._populate_neutron_binding_profile,
|
|
instance, pci_req_id, port_req_body, None)
|
|
|
|
@mock.patch('nova.objects.Instance.get_pci_devices', return_value=[])
|
|
def test_populate_neutron_binding_profile_pci_dev_not_found(
|
|
self, mock_get_instance_pci_devs):
|
|
api = neutronapi.API()
|
|
instance = objects.Instance(pci_devices=objects.PciDeviceList())
|
|
port_req_body = {'port': {}}
|
|
pci_req_id = 'my_req_id'
|
|
self.assertRaises(exception.PciDeviceNotFound,
|
|
api._populate_neutron_binding_profile,
|
|
instance, pci_req_id, port_req_body, None)
|
|
mock_get_instance_pci_devs.assert_called_once_with(
|
|
request_id=pci_req_id)
|
|
|
|
@mock.patch.object(
|
|
pci_utils, 'is_physical_function',
|
|
new=mock.Mock(return_value=False)
|
|
)
|
|
@mock.patch.object(
|
|
pci_utils, 'get_vf_num_by_pci_address',
|
|
new=mock.MagicMock(
|
|
side_effect=(lambda vf_a: {'0000:0a:00.1': 1}.get(vf_a)))
|
|
)
|
|
@mock.patch.object(
|
|
pci_utils, 'get_mac_by_pci_address',
|
|
new=mock.MagicMock(side_effect=(lambda vf_a: {
|
|
'0000:0a:00.0': '52:54:00:1e:59:c6'}.get(vf_a)))
|
|
)
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
def test_pci_parse_whitelist_called_once(
|
|
self, mock_get_instance_pci_devs
|
|
):
|
|
device_spec = [
|
|
jsonutils.dumps(
|
|
{
|
|
"address": "0000:0a:00.1",
|
|
"physical_network": "default",
|
|
}
|
|
)
|
|
]
|
|
cfg.CONF.set_override(
|
|
'device_spec', device_spec, 'pci')
|
|
|
|
# NOTE(takashin): neutronapi.API must be initialized
|
|
# after the 'device_spec' is set in this test case.
|
|
api = neutronapi.API()
|
|
host_id = 'my_host_id'
|
|
instance = objects.Instance(host=host_id)
|
|
pci_req_id = 'my_req_id'
|
|
port_req_body = {'port': {}}
|
|
pci_dev = {'vendor_id': '1377',
|
|
'product_id': '0047',
|
|
'address': '0000:0a:00.1',
|
|
'parent_addr': '0000:0a:00.0',
|
|
'dev_type': obj_fields.PciDeviceType.SRIOV_VF,
|
|
}
|
|
|
|
whitelist = pci_whitelist.Whitelist(CONF.pci.device_spec)
|
|
with mock.patch.object(pci_whitelist.Whitelist,
|
|
'_parse_white_list_from_config',
|
|
wraps=whitelist._parse_white_list_from_config
|
|
) as mock_parse_whitelist:
|
|
for i in range(2):
|
|
mydev = objects.PciDevice.create(None, pci_dev)
|
|
mock_get_instance_pci_devs.return_value = [mydev]
|
|
api._populate_neutron_binding_profile(
|
|
instance, pci_req_id, port_req_body, None)
|
|
self.assertEqual(0, mock_parse_whitelist.call_count)
|
|
|
|
def _populate_pci_mac_address_fakes(self):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
pci_dev = {'vendor_id': '1377',
|
|
'product_id': '0047',
|
|
'address': '0000:0a:00.1',
|
|
'dev_type': 'type-PF'}
|
|
pf = objects.PciDevice()
|
|
vf = objects.PciDevice()
|
|
pf.update_device(pci_dev)
|
|
|
|
pci_dev['dev_type'] = 'type-VF'
|
|
vf.update_device(pci_dev)
|
|
return instance, pf, vf
|
|
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
@mock.patch.object(pci_utils, 'get_mac_by_pci_address')
|
|
def test_populate_pci_mac_address_pf(self, mock_get_mac_by_pci_address,
|
|
mock_get_instance_pci_devs):
|
|
instance, pf, vf = self._populate_pci_mac_address_fakes()
|
|
|
|
port_req_body = {'port': {}}
|
|
mock_get_instance_pci_devs.return_value = [pf]
|
|
mock_get_mac_by_pci_address.return_value = 'fake-mac-address'
|
|
expected_port_req_body = {'port': {'mac_address': 'fake-mac-address'}}
|
|
req = port_req_body.copy()
|
|
self.api._populate_pci_mac_address(instance, 0, req)
|
|
self.assertEqual(expected_port_req_body, req)
|
|
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
@mock.patch.object(pci_utils, 'get_mac_by_pci_address')
|
|
def test_populate_pci_mac_address_vf(self, mock_get_mac_by_pci_address,
|
|
mock_get_instance_pci_devs):
|
|
instance, pf, vf = self._populate_pci_mac_address_fakes()
|
|
|
|
port_req_body = {'port': {}}
|
|
mock_get_instance_pci_devs.return_value = [vf]
|
|
req = port_req_body.copy()
|
|
self.api._populate_pci_mac_address(instance, 42, port_req_body)
|
|
self.assertEqual(port_req_body, req)
|
|
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
@mock.patch.object(pci_utils, 'get_mac_by_pci_address')
|
|
def test_populate_pci_mac_address_vf_fail(self,
|
|
mock_get_mac_by_pci_address,
|
|
mock_get_instance_pci_devs):
|
|
instance, pf, vf = self._populate_pci_mac_address_fakes()
|
|
|
|
port_req_body = {'port': {}}
|
|
mock_get_instance_pci_devs.return_value = [vf]
|
|
mock_get_mac_by_pci_address.side_effect = (
|
|
exception.PciDeviceNotFoundById)
|
|
req = port_req_body.copy()
|
|
self.api._populate_pci_mac_address(instance, 42, port_req_body)
|
|
self.assertEqual(port_req_body, req)
|
|
|
|
@mock.patch('nova.objects.Instance.get_pci_devices')
|
|
@mock.patch('nova.network.neutron.LOG.error')
|
|
def test_populate_pci_mac_address_no_device(self, mock_log_error,
|
|
mock_get_instance_pci_devs):
|
|
instance, pf, vf = self._populate_pci_mac_address_fakes()
|
|
|
|
port_req_body = {'port': {}}
|
|
mock_get_instance_pci_devs.return_value = []
|
|
req = port_req_body.copy()
|
|
self.api._populate_pci_mac_address(instance, 42, port_req_body)
|
|
self.assertEqual(port_req_body, req)
|
|
self.assertEqual(42, mock_log_error.call_args[0][1])
|
|
|
|
def _test_update_port_binding_true(self, expected_bind_host,
|
|
func_name, *args):
|
|
func = getattr(self.api, func_name)
|
|
|
|
search_opts = {'device_id': self.instance['uuid'],
|
|
'tenant_id': self.instance['project_id']}
|
|
ports = {'ports': [{'id': 'test1'}]}
|
|
port_req_body = {'port':
|
|
{constants.BINDING_HOST_ID: expected_bind_host,
|
|
'device_owner': 'compute:%s' %
|
|
self.instance['availability_zone']}}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mocked_client.list_ports.return_value = ports
|
|
mocked_client.update_port.return_value = None
|
|
with mock.patch.object(neutronapi, 'get_client',
|
|
return_value=mocked_client) as mock_get_client:
|
|
func(*args)
|
|
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mocked_client.list_ports.assert_called_once_with(**search_opts)
|
|
mocked_client.update_port.assert_called_once_with(
|
|
'test1', port_req_body)
|
|
|
|
def _test_update_port_true_exception(self, expected_bind_host,
|
|
func_name, *args):
|
|
func = getattr(self.api, func_name)
|
|
|
|
search_opts = {'device_id': self.instance['uuid'],
|
|
'tenant_id': self.instance['project_id']}
|
|
ports = {'ports': [{'id': 'test1'}]}
|
|
port_req_body = {'port':
|
|
{constants.BINDING_HOST_ID: expected_bind_host,
|
|
'device_owner': 'compute:%s' %
|
|
self.instance['availability_zone']}}
|
|
mocked_client = mock.create_autospec(client.Client)
|
|
mocked_client.list_ports.return_value = ports
|
|
mocked_client.update_port.side_effect = Exception(
|
|
"fail to update port")
|
|
with mock.patch.object(neutronapi, 'get_client',
|
|
return_value=mocked_client) as mock_get_client:
|
|
self.assertRaises(NEUTRON_CLIENT_EXCEPTION,
|
|
func,
|
|
*args)
|
|
|
|
mock_get_client.assert_called_once_with(mock.ANY, admin=True)
|
|
mocked_client.list_ports.assert_called_once_with(**search_opts)
|
|
mocked_client.update_port.assert_called_once_with(
|
|
'test1', port_req_body)
|
|
|
|
def test_migrate_instance_finish_binding_true(self):
|
|
migration = objects.Migration(source_compute=self.instance.get('host'),
|
|
dest_compute='dest_host')
|
|
instance = self._fake_instance_object(self.instance)
|
|
self._test_update_port_binding_true('dest_host',
|
|
'migrate_instance_finish',
|
|
self.context,
|
|
instance,
|
|
migration,
|
|
{})
|
|
|
|
def test_migrate_instance_finish_binding_true_exception(self):
|
|
migration = objects.Migration(source_compute=self.instance.get('host'),
|
|
dest_compute='dest_host')
|
|
instance = self._fake_instance_object(self.instance)
|
|
self._test_update_port_true_exception('dest_host',
|
|
'migrate_instance_finish',
|
|
self.context,
|
|
instance,
|
|
migration,
|
|
{})
|
|
|
|
def test_setup_instance_network_on_host_true(self):
|
|
instance = self._fake_instance_object(self.instance)
|
|
self._test_update_port_binding_true('fake_host',
|
|
'setup_instance_network_on_host',
|
|
self.context,
|
|
instance,
|
|
'fake_host')
|
|
|
|
def test_setup_instance_network_on_host_exception(self):
|
|
instance = self._fake_instance_object(self.instance)
|
|
self._test_update_port_true_exception(
|
|
'fake_host', 'setup_instance_network_on_host',
|
|
self.context, instance, 'fake_host')
|
|
|
|
@mock.patch('nova.network.neutron.get_client',
|
|
new_callable=mock.NonCallableMock)
|
|
def test_bind_ports_to_host_no_ports(self, mock_client):
|
|
self.assertDictEqual({},
|
|
self.api.bind_ports_to_host(
|
|
mock.sentinel.context,
|
|
objects.Instance(info_cache=None),
|
|
'fake-host'))
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_bind_ports_to_host(self, mock_client):
|
|
"""Tests a single port happy path where everything is successful."""
|
|
def fake_create(port_id, data):
|
|
self.assertDictEqual(binding, data)
|
|
return mock.DEFAULT
|
|
|
|
nwinfo = model.NetworkInfo([model.VIF(uuids.port)])
|
|
inst = objects.Instance(
|
|
info_cache=objects.InstanceInfoCache(network_info=nwinfo))
|
|
ctxt = context.get_context()
|
|
binding = {'binding': {'host': 'fake-host',
|
|
'vnic_type': 'normal',
|
|
'profile': {'foo': 'bar'}}}
|
|
mocked_client = mock_client.return_value
|
|
mocked_client.create_port_binding.return_value = binding
|
|
mocked_client.create_port_binding.side_effect = fake_create
|
|
|
|
result = self.api.bind_ports_to_host(
|
|
ctxt, inst, 'fake-host', {uuids.port: 'normal'},
|
|
{uuids.port: {'foo': 'bar'}})
|
|
|
|
self.assertEqual(1, mocked_client.create_port_binding.call_count)
|
|
self.assertDictEqual({uuids.port: binding['binding']}, result)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_bind_ports_to_host_with_vif_profile_and_vnic(self, mock_client):
|
|
"""Tests bind_ports_to_host with default/non-default parameters."""
|
|
def fake_create(port_id, data):
|
|
self.assertDictEqual(binding, data)
|
|
return mock.DEFAULT
|
|
|
|
ctxt = context.get_context()
|
|
vif_profile = {'foo': 'default'}
|
|
nwinfo = model.NetworkInfo([model.VIF(id=uuids.port,
|
|
vnic_type="direct",
|
|
profile=vif_profile)])
|
|
inst = objects.Instance(
|
|
info_cache=objects.InstanceInfoCache(network_info=nwinfo))
|
|
binding = {'binding': {'host': 'fake-host',
|
|
'vnic_type': 'direct',
|
|
'profile': vif_profile}}
|
|
mocked_client = mock_client.return_value
|
|
mocked_client.create_port_binding.return_value = binding
|
|
mocked_client.create_port_binding.side_effect = fake_create
|
|
|
|
result = self.api.bind_ports_to_host(ctxt, inst, 'fake-host')
|
|
|
|
self.assertEqual(1, mocked_client.create_port_binding.call_count)
|
|
self.assertDictEqual({uuids.port: binding['binding']}, result)
|
|
|
|
# assert that if vnic_type and profile are set in VIF object
|
|
# the provided vnic_type and profile take precedence.
|
|
|
|
nwinfo = model.NetworkInfo([model.VIF(id=uuids.port,
|
|
vnic_type='direct',
|
|
profile=vif_profile)])
|
|
inst = objects.Instance(
|
|
info_cache=objects.InstanceInfoCache(network_info=nwinfo))
|
|
vif_profile_per_port = {uuids.port: {'foo': 'overridden'}}
|
|
vnic_type_per_port = {uuids.port: "direct-overridden"}
|
|
binding = {'binding': {'host': 'fake-host',
|
|
'vnic_type': 'direct-overridden',
|
|
'profile': {'foo': 'overridden'}}}
|
|
mocked_client = mock_client.return_value
|
|
mocked_client.create_port_binding.return_value = binding
|
|
mocked_client.create_port_binding.side_effect = fake_create
|
|
|
|
result = self.api.bind_ports_to_host(
|
|
ctxt, inst, 'fake-host', vnic_type_per_port, vif_profile_per_port)
|
|
|
|
self.assertEqual(2, mocked_client.create_port_binding.call_count)
|
|
self.assertDictEqual({uuids.port: binding['binding']}, result)
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_bind_ports_to_host_rollback(self, mock_client):
|
|
"""Tests a scenario where an instance has two ports, and binding the
|
|
first is successful but binding the second fails, so the code will
|
|
rollback the binding for the first port.
|
|
"""
|
|
nwinfo = model.NetworkInfo([
|
|
model.VIF(uuids.ok), model.VIF(uuids.fail)])
|
|
inst = objects.Instance(
|
|
info_cache=objects.InstanceInfoCache(network_info=nwinfo))
|
|
NeutronError = exceptions.NeutronClientException(status_code=500)
|
|
|
|
def fake_create(port_id, host):
|
|
if port_id == uuids.ok:
|
|
return {'binding': {'host': 'fake-host'}}
|
|
|
|
raise NeutronError
|
|
|
|
mocked_client = mock_client.return_value
|
|
mocked_client.create_port_binding.side_effect = fake_create
|
|
mocked_client.delete_port_binding.side_effect = NeutronError
|
|
|
|
self.assertRaises(exception.PortBindingFailed,
|
|
self.api.bind_ports_to_host,
|
|
self.context, inst, 'fake-host')
|
|
|
|
# assert that create was called twice and delete once
|
|
self.assertEqual(2, mocked_client.create_port_binding.call_count)
|
|
self.assertEqual(1, mocked_client.delete_port_binding.call_count)
|
|
mocked_client.delete_port_binding.assert_called_once_with(
|
|
uuids.ok, 'fake-host')
|
|
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_delete_port_binding(self, mock_client):
|
|
# Create three ports where:
|
|
# - one is successfully unbound
|
|
# - one is not found
|
|
# - one fails to be unbound
|
|
def fake_delete(port_id, host):
|
|
if port_id == uuids.ok:
|
|
return
|
|
|
|
status_code = 404 if port_id == uuids.notfound else 500
|
|
raise exceptions.NeutronClientException(status_code=status_code)
|
|
|
|
mock_client.return_value.delete_port_binding.side_effect = fake_delete
|
|
for port_id in (uuids.ok, uuids.notfound, uuids.fail):
|
|
if port_id == uuids.fail:
|
|
self.assertRaises(exception.PortBindingDeletionFailed,
|
|
self.api.delete_port_binding,
|
|
self.context, port_id, 'fake-host')
|
|
else:
|
|
self.api.delete_port_binding(self.context, port_id,
|
|
'fake-host')
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=False))
|
|
@mock.patch('nova.accelerator.cyborg._CyborgClient.delete_arqs_by_uuid')
|
|
@mock.patch('nova.network.neutron.get_binding_profile')
|
|
@mock.patch('nova.network.neutron.API._show_port')
|
|
@mock.patch('nova.network.neutron.get_client')
|
|
def test_unbind_ports_clean_arq(self, mock_neutron, mock_show,
|
|
mock_bind, mock_delete_arq):
|
|
mock_client = mock.Mock()
|
|
mock_ctx = mock.Mock(is_admin=False)
|
|
ports = ["1"]
|
|
mock_show.return_value = {'id': uuids.port}
|
|
mock_bind.return_value = {'arq_uuid': self.arqs[0]['uuid'],
|
|
'key': 'val'}
|
|
api = neutronapi.API()
|
|
api._unbind_ports(mock_ctx, ports, mock_neutron, mock_client)
|
|
mock_delete_arq.assert_called_once_with([self.arqs[0]['uuid']])
|
|
# verify binding profile key 'arq_uuid' deleted
|
|
call_args = mock_client.update_port.call_args[0][1]
|
|
self.assertEqual(call_args['port']['binding:profile'],
|
|
{'key': 'val'})
|
|
|
|
|
|
class TestAllocateForInstance(test.NoDBTestCase):
|
|
def setUp(self):
|
|
super(TestAllocateForInstance, self).setUp()
|
|
self.context = context.RequestContext('userid', uuids.my_tenant)
|
|
self.instance = objects.Instance(uuid=uuids.instance,
|
|
project_id=uuids.tenant_id, hostname="host")
|
|
|
|
def test_allocate_for_instance_raises_invalid_input(self):
|
|
api = neutronapi.API()
|
|
self.instance.project_id = ""
|
|
|
|
self.assertRaises(exception.InvalidInput,
|
|
api.allocate_for_instance, self.context, self.instance, None)
|
|
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info')
|
|
@mock.patch.object(neutronapi.API, '_update_ports_for_instance')
|
|
@mock.patch.object(neutronapi.API, '_create_ports_for_instance')
|
|
@mock.patch.object(neutronapi.API, '_process_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_clean_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_network_ids')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_port_ids')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
def test_allocate_for_instance_minimal_args(self, mock_get_client,
|
|
mock_validate_ports, mock_validate_nets, mock_clean_sg, mock_sg,
|
|
mock_create_ports, mock_update_ports, mock_gni):
|
|
|
|
api = neutronapi.API()
|
|
mock_get_client.side_effect = ["user", "admin"]
|
|
mock_validate_ports.return_value = ({}, "ordered_nets")
|
|
mock_validate_nets.return_value = "nets"
|
|
mock_clean_sg.return_value = "security_groups"
|
|
mock_sg.return_value = "security_group_ids"
|
|
mock_create_ports.return_value = "requests_and_created_ports"
|
|
mock_update_ports.return_value = (
|
|
"nets", "ports", [uuids.preexist], [uuids.created])
|
|
mock_gni.return_value = [
|
|
{"id": uuids.created}, {"id": uuids.preexist}, {"id": "foo"}
|
|
]
|
|
|
|
result = api.allocate_for_instance(self.context, self.instance, None)
|
|
|
|
self.assertEqual(len(result), 2)
|
|
self.assertEqual(result[0], {"id": uuids.created})
|
|
self.assertEqual(result[1], {"id": uuids.preexist})
|
|
|
|
mock_validate_ports.assert_called_once_with(
|
|
self.context, self.instance, "admin", None)
|
|
|
|
def test_ensure_no_port_binding_failure_raises(self):
|
|
port = {
|
|
'id': uuids.port_id,
|
|
'binding:vif_type': model.VIF_TYPE_BINDING_FAILED
|
|
}
|
|
|
|
self.assertRaises(exception.PortBindingFailed,
|
|
neutronapi._ensure_no_port_binding_failure, port)
|
|
|
|
def test_ensure_no_port_binding_failure_passes_if_no_binding(self):
|
|
port = {'id': uuids.port_id}
|
|
neutronapi._ensure_no_port_binding_failure(port)
|
|
|
|
def test_validate_requested_port_ids_no_ports(self):
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
network_list = [objects.NetworkRequest(network_id='net-1')]
|
|
requested_networks = objects.NetworkRequestList(objects=network_list)
|
|
|
|
ports, ordered_networks = api._validate_requested_port_ids(
|
|
self.context, self.instance, mock_client, requested_networks)
|
|
|
|
self.assertEqual({}, ports)
|
|
self.assertEqual(network_list, ordered_networks)
|
|
|
|
def test_validate_requested_port_ids_success(self):
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
requested_networks = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id='net-1'),
|
|
objects.NetworkRequest(port_id=uuids.port_id)])
|
|
port = {
|
|
"id": uuids.port_id,
|
|
"tenant_id": uuids.tenant_id,
|
|
"network_id": 'net-2'
|
|
}
|
|
mock_client.show_port.return_value = {"port": port}
|
|
|
|
ports, ordered_networks = api._validate_requested_port_ids(
|
|
self.context, self.instance, mock_client, requested_networks)
|
|
|
|
mock_client.show_port.assert_called_once_with(uuids.port_id)
|
|
self.assertEqual({uuids.port_id: port}, ports)
|
|
self.assertEqual(2, len(ordered_networks))
|
|
self.assertEqual(requested_networks[0], ordered_networks[0])
|
|
self.assertEqual('net-2', ordered_networks[1].network_id)
|
|
|
|
def _assert_validate_requested_port_ids_raises(self, exception, extras):
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
requested_networks = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(port_id=uuids.port_id)])
|
|
port = {
|
|
"id": uuids.port_id,
|
|
"tenant_id": uuids.tenant_id,
|
|
"network_id": 'net-2'
|
|
}
|
|
port.update(extras)
|
|
mock_client.show_port.return_value = {"port": port}
|
|
|
|
self.assertRaises(exception, api._validate_requested_port_ids,
|
|
self.context, self.instance, mock_client, requested_networks)
|
|
|
|
def test_validate_requested_port_ids_raise_not_usable(self):
|
|
self._assert_validate_requested_port_ids_raises(
|
|
exception.PortNotUsable,
|
|
{"tenant_id": "foo"})
|
|
|
|
def test_validate_requested_port_ids_raise_in_use(self):
|
|
self._assert_validate_requested_port_ids_raises(
|
|
exception.PortInUse,
|
|
{"device_id": "foo"})
|
|
|
|
def test_validate_requested_port_ids_raise_dns(self):
|
|
self._assert_validate_requested_port_ids_raises(
|
|
exception.PortNotUsableDNS,
|
|
{"dns_name": "foo"})
|
|
|
|
def test_validate_requested_port_ids_raise_binding(self):
|
|
self._assert_validate_requested_port_ids_raises(
|
|
exception.PortBindingFailed,
|
|
{"binding:vif_type": model.VIF_TYPE_BINDING_FAILED})
|
|
|
|
def test_validate_requested_network_ids_success_auto_net(self):
|
|
requested_networks = []
|
|
ordered_networks = []
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
nets = [{'id': "net1"}]
|
|
mock_client.list_networks.side_effect = [{}, {"networks": nets}]
|
|
|
|
result = api._validate_requested_network_ids(self.context,
|
|
self.instance, mock_client, requested_networks, ordered_networks)
|
|
|
|
self.assertEqual(nets, list(result.values()))
|
|
expected_call_list = [
|
|
mock.call(shared=False, tenant_id=uuids.tenant_id),
|
|
mock.call(shared=True)
|
|
]
|
|
self.assertEqual(expected_call_list,
|
|
mock_client.list_networks.call_args_list)
|
|
|
|
def test_validate_requested_network_ids_success_found_net(self):
|
|
ordered_networks = [objects.NetworkRequest(network_id="net1")]
|
|
requested_networks = objects.NetworkRequestList(ordered_networks)
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
nets = [{'id': "net1"}]
|
|
mock_client.list_networks.return_value = {"networks": nets}
|
|
|
|
result = api._validate_requested_network_ids(self.context,
|
|
self.instance, mock_client, requested_networks, ordered_networks)
|
|
|
|
self.assertEqual(nets, list(result.values()))
|
|
mock_client.list_networks.assert_called_once_with(id=['net1'])
|
|
|
|
def test_validate_requested_network_ids_success_no_nets(self):
|
|
requested_networks = []
|
|
ordered_networks = []
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
mock_client.list_networks.side_effect = [{}, {"networks": []}]
|
|
|
|
result = api._validate_requested_network_ids(self.context,
|
|
self.instance, mock_client, requested_networks, ordered_networks)
|
|
|
|
self.assertEqual({}, result)
|
|
expected_call_list = [
|
|
mock.call(shared=False, tenant_id=uuids.tenant_id),
|
|
mock.call(shared=True)
|
|
]
|
|
self.assertEqual(expected_call_list,
|
|
mock_client.list_networks.call_args_list)
|
|
|
|
def _assert_validate_requested_network_ids_raises(self, exception, nets,
|
|
requested_networks=None):
|
|
ordered_networks = []
|
|
if requested_networks is None:
|
|
requested_networks = objects.NetworkRequestList()
|
|
api = neutronapi.API()
|
|
mock_client = mock.Mock()
|
|
mock_client.list_networks.side_effect = [{}, {"networks": nets}]
|
|
|
|
self.assertRaises(exception, api._validate_requested_network_ids,
|
|
self.context, self.instance, mock_client,
|
|
requested_networks, ordered_networks)
|
|
|
|
def test_validate_requested_network_ids_raises_forbidden(self):
|
|
rules = {'network:attach_external_network': 'is_admin:True'}
|
|
policy.set_rules(oslo_policy.Rules.from_dict(rules))
|
|
self._assert_validate_requested_network_ids_raises(
|
|
exception.ExternalNetworkAttachForbidden,
|
|
[{'id': "net1", 'router:external': True, 'shared': False}])
|
|
|
|
def test_validate_requested_network_ids_raises_net_not_found(self):
|
|
requested_networks = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id="1")])
|
|
self._assert_validate_requested_network_ids_raises(
|
|
exception.NetworkNotFound,
|
|
[], requested_networks=requested_networks)
|
|
|
|
def test_validate_requested_network_ids_raises_too_many_nets(self):
|
|
self._assert_validate_requested_network_ids_raises(
|
|
exception.NetworkAmbiguous,
|
|
[{'id': "net1"}, {'id': "net2"}])
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
def test_create_ports_for_instance_no_security(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [objects.NetworkRequest(network_id=uuids.net)]
|
|
nets = {uuids.net: {"id": uuids.net, "port_security_enabled": False}}
|
|
mock_client = mock.Mock()
|
|
mock_client.create_port.return_value = {"port": {"id": uuids.port}}
|
|
|
|
result = api._create_ports_for_instance(self.context, self.instance,
|
|
ordered_networks, nets, mock_client, None)
|
|
|
|
self.assertEqual([(ordered_networks[0], uuids.port)], result)
|
|
mock_client.create_port.assert_called_once_with(
|
|
{'port': {
|
|
'network_id': uuids.net, 'tenant_id': uuids.tenant_id,
|
|
'admin_state_up': True, 'device_id': self.instance.uuid}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
def test_create_ports_for_instance_with_security_groups(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [objects.NetworkRequest(network_id=uuids.net)]
|
|
nets = {uuids.net: {"id": uuids.net, "subnets": [uuids.subnet]}}
|
|
mock_client = mock.Mock()
|
|
mock_client.create_port.return_value = {"port": {"id": uuids.port}}
|
|
security_groups = [uuids.sg]
|
|
|
|
result = api._create_ports_for_instance(self.context, self.instance,
|
|
ordered_networks, nets, mock_client, security_groups)
|
|
|
|
self.assertEqual([(ordered_networks[0], uuids.port)], result)
|
|
mock_client.create_port.assert_called_once_with(
|
|
{'port': {
|
|
'network_id': uuids.net, 'tenant_id': uuids.tenant_id,
|
|
'admin_state_up': True, 'security_groups': security_groups,
|
|
'device_id': self.instance.uuid}})
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
def test_create_ports_for_instance_with_cleanup_after_pc_failure(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [
|
|
objects.NetworkRequest(network_id=uuids.net1),
|
|
objects.NetworkRequest(network_id=uuids.net2),
|
|
objects.NetworkRequest(network_id=uuids.net3),
|
|
objects.NetworkRequest(network_id=uuids.net4)
|
|
]
|
|
nets = {
|
|
uuids.net1: {"id": uuids.net1, "port_security_enabled": False},
|
|
uuids.net2: {"id": uuids.net2, "port_security_enabled": False},
|
|
uuids.net3: {"id": uuids.net3, "port_security_enabled": False},
|
|
uuids.net4: {"id": uuids.net4, "port_security_enabled": False}
|
|
}
|
|
error = exception.PortLimitExceeded()
|
|
mock_client = mock.Mock()
|
|
mock_client.create_port.side_effect = [
|
|
{"port": {"id": uuids.port1}},
|
|
{"port": {"id": uuids.port2}},
|
|
error
|
|
]
|
|
|
|
self.assertRaises(exception.PortLimitExceeded,
|
|
api._create_ports_for_instance,
|
|
self.context, self.instance, ordered_networks, nets,
|
|
mock_client, None)
|
|
|
|
self.assertEqual([mock.call(uuids.port1), mock.call(uuids.port2)],
|
|
mock_client.delete_port.call_args_list)
|
|
self.assertEqual(3, mock_client.create_port.call_count)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False),
|
|
)
|
|
def test_create_ports_for_instance_with_cleanup_after_sg_failure(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [
|
|
objects.NetworkRequest(network_id=uuids.net1),
|
|
objects.NetworkRequest(network_id=uuids.net2),
|
|
objects.NetworkRequest(network_id=uuids.net3)
|
|
]
|
|
nets = {
|
|
uuids.net1: {"id": uuids.net1, "port_security_enabled": False},
|
|
uuids.net2: {"id": uuids.net2, "port_security_enabled": False},
|
|
uuids.net3: {"id": uuids.net3, "port_security_enabled": True}
|
|
}
|
|
mock_client = mock.Mock()
|
|
mock_client.create_port.side_effect = [
|
|
{"port": {"id": uuids.port1}},
|
|
{"port": {"id": uuids.port2}}
|
|
]
|
|
|
|
self.assertRaises(exception.SecurityGroupCannotBeApplied,
|
|
api._create_ports_for_instance,
|
|
self.context, self.instance, ordered_networks, nets,
|
|
mock_client, None)
|
|
|
|
self.assertEqual([mock.call(uuids.port1), mock.call(uuids.port2)],
|
|
mock_client.delete_port.call_args_list)
|
|
self.assertEqual(2, mock_client.create_port.call_count)
|
|
|
|
def test_create_ports_for_instance_raises_subnets_missing(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [objects.NetworkRequest(network_id=uuids.net)]
|
|
nets = {uuids.net: {"id": uuids.net, "port_security_enabled": True}}
|
|
mock_client = mock.Mock()
|
|
|
|
self.assertRaises(exception.SecurityGroupCannotBeApplied,
|
|
api._create_ports_for_instance,
|
|
self.context, self.instance,
|
|
ordered_networks, nets, mock_client, None)
|
|
|
|
self.assertFalse(mock_client.create_port.called)
|
|
|
|
def test_create_ports_for_instance_raises_security_off(self):
|
|
api = neutronapi.API()
|
|
ordered_networks = [objects.NetworkRequest(network_id=uuids.net)]
|
|
nets = {uuids.net: {
|
|
"id": uuids.net,
|
|
"port_security_enabled": False}}
|
|
mock_client = mock.Mock()
|
|
|
|
self.assertRaises(exception.SecurityGroupCannotBeApplied,
|
|
api._create_ports_for_instance,
|
|
self.context, self.instance,
|
|
ordered_networks, nets, mock_client, [uuids.sg])
|
|
|
|
self.assertFalse(mock_client.create_port.called)
|
|
|
|
@mock.patch.object(objects.VirtualInterface, "create")
|
|
def test_update_ports_for_instance_with_portbinding(self, mock_create):
|
|
api = neutronapi.API()
|
|
self.instance.availability_zone = "test_az"
|
|
mock_neutron = mock.Mock()
|
|
mock_admin = mock.Mock()
|
|
requests_and_created_ports = [
|
|
(objects.NetworkRequest(
|
|
network_id=uuids.net1), uuids.port1),
|
|
(objects.NetworkRequest(
|
|
network_id=uuids.net2, port_id=uuids.port2), None)]
|
|
net1 = {"id": uuids.net1}
|
|
net2 = {"id": uuids.net2}
|
|
nets = {uuids.net1: net1, uuids.net2: net2}
|
|
bind_host_id = "bind_host_id"
|
|
requested_ports_dict = {uuids.port1: {}, uuids.port2: {}}
|
|
|
|
mock_neutron.list_extensions.return_value = {"extensions": [
|
|
{"alias": "asdf"}]}
|
|
port1 = {"port": {"id": uuids.port1, "mac_address": "mac1r"}}
|
|
port2 = {"port": {"id": uuids.port2, "mac_address": "mac2r"}}
|
|
mock_admin.update_port.side_effect = [port1, port2]
|
|
|
|
ordered_nets, ordered_ports, preexisting_port_ids, \
|
|
created_port_ids = api._update_ports_for_instance(
|
|
self.context, self.instance,
|
|
mock_neutron, mock_admin, requests_and_created_ports, nets,
|
|
bind_host_id, requested_ports_dict, None)
|
|
|
|
self.assertEqual([net1, net2], ordered_nets, "ordered_nets")
|
|
self.assertEqual([uuids.port1, uuids.port2], ordered_ports,
|
|
"ordered_ports")
|
|
self.assertEqual([uuids.port2], preexisting_port_ids, "preexisting")
|
|
self.assertEqual([uuids.port1], created_port_ids, "created")
|
|
mock_admin.update_port.assert_called_with(uuids.port2,
|
|
{'port': {
|
|
'device_owner': 'compute:test_az',
|
|
constants.BINDING_HOST_ID: bind_host_id,
|
|
'device_id': self.instance.uuid}})
|
|
|
|
|
|
class TestAPINeutronHostnameDNS(TestAPIBase):
|
|
|
|
def test_allocate_for_instance_create_port(self):
|
|
# The port's dns_name attribute should be set by the port create
|
|
# request in allocate_for_instance
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
1, dns_extension=True)
|
|
|
|
def test_allocate_for_instance_with_requested_port(self):
|
|
# The port's dns_name attribute should be set by the port update
|
|
# request in allocate_for_instance
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=1, dns_extension=True,
|
|
requested_networks=requested_networks)
|
|
|
|
def test_allocate_for_instance_port_dns_name_preset_equal_hostname(self):
|
|
# The port's dns_name attribute should be set by the port update
|
|
# request in allocate_for_instance. The port's dns_name was preset by
|
|
# the user with a value equal to the instance's hostname
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=1, dns_extension=True,
|
|
requested_networks=requested_networks,
|
|
_dns_name='test-instance')
|
|
|
|
def test_allocate_for_instance_port_dns_name_preset_noteq_hostname(self):
|
|
# If a pre-existing port has dns_name set, an exception should be
|
|
# raised if dns_name is not equal to the instance's hostname
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance(
|
|
requested_networks=requested_networks,
|
|
exception=exception.PortNotUsableDNS,
|
|
dns_extension=True,
|
|
_break='pre_list_networks',
|
|
_dns_name='my-instance')
|
|
|
|
|
|
class TestAPINeutronHostnameDNSPortbinding(TestAPIBase):
|
|
|
|
def test_allocate_for_instance_create_port(self):
|
|
# The port's dns_name attribute should be set by the port create
|
|
# request in allocate_for_instance
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
1, dns_extension=True, bind_host_id=self.instance.get('host'))
|
|
|
|
def test_allocate_for_instance_with_requested_port(self):
|
|
# The port's dns_name attribute should be set by the port update
|
|
# request in allocate_for_instance
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=1, dns_extension=True,
|
|
bind_host_id=self.instance.get('host'),
|
|
requested_networks=requested_networks)
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_extended_resource_request_extension',
|
|
new=mock.Mock(return_value=False)
|
|
)
|
|
def test_allocate_for_instance_create_port_with_dns_domain(self):
|
|
# The port's dns_name attribute should be set by the port update
|
|
# request in _update_port_dns_name. This should happen only when the
|
|
# port binding extension is enabled and the port's network has a
|
|
# non-blank dns_domain attribute
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
11, dns_extension=True, bind_host_id=self.instance.get('host'))
|
|
|
|
@mock.patch(
|
|
'nova.network.neutron.API.has_dns_extension',
|
|
new=mock.Mock(return_value=True),
|
|
)
|
|
def test_allocate_for_instance_with_requested_port_with_dns_domain(self):
|
|
# The port's dns_name attribute should be set by the port update
|
|
# request in _update_port_dns_name. This should happen only when the
|
|
# port binding extension is enabled and the port's network has a
|
|
# non-blank dns_domain attribute
|
|
requested_networks = objects.NetworkRequestList(
|
|
objects=[objects.NetworkRequest(port_id=uuids.portid_1)])
|
|
self._test_allocate_for_instance_with_virtual_interface(
|
|
net_idx=11, dns_extension=True,
|
|
bind_host_id=self.instance.get('host'),
|
|
requested_networks=requested_networks)
|
|
|
|
|
|
class TestNeutronClientForAdminScenarios(test.NoDBTestCase):
|
|
|
|
def setUp(self):
|
|
super(TestNeutronClientForAdminScenarios, self).setUp()
|
|
# NOTE(morganfainberg): The real configuration fixture here is used
|
|
# instead o the already existing fixtures to ensure that the new
|
|
# config options are automatically deregistered at the end of the
|
|
# test run. Without the use of this fixture, the config options
|
|
# from the plugin(s) would persist for all subsequent tests from when
|
|
# these are run (due to glonal conf object) and not be fully
|
|
# representative of a "clean" slate at the start of a test.
|
|
self.config_fixture = self.useFixture(config_fixture.Config())
|
|
oslo_opts = ks_loading.get_auth_plugin_conf_options('v2password')
|
|
self.config_fixture.register_opts(oslo_opts, 'neutron')
|
|
|
|
@requests_mock.mock()
|
|
def _test_get_client_for_admin(self, req_mock,
|
|
use_id=False, admin_context=False):
|
|
token_value = uuidutils.generate_uuid(dashed=False)
|
|
auth_url = 'http://anyhost/auth'
|
|
token_resp = V2Token(token_id=token_value)
|
|
req_mock.post(auth_url + '/tokens', json=token_resp)
|
|
|
|
self.flags(endpoint_override='http://anyhost/', group='neutron')
|
|
self.flags(auth_type='v2password', group='neutron')
|
|
self.flags(auth_url=auth_url, group='neutron')
|
|
self.flags(timeout=30, group='neutron')
|
|
if use_id:
|
|
self.flags(tenant_id='tenant_id', group='neutron')
|
|
self.flags(user_id='user_id', group='neutron')
|
|
|
|
if admin_context:
|
|
my_context = context.get_admin_context()
|
|
else:
|
|
my_context = context.RequestContext('userid', uuids.my_tenant,
|
|
auth_token='token')
|
|
|
|
# clean global
|
|
neutronapi.reset_state()
|
|
|
|
if admin_context:
|
|
# Note that the context does not contain a token but is
|
|
# an admin context which will force an elevation to admin
|
|
# credentials.
|
|
context_client = neutronapi.get_client(my_context)
|
|
else:
|
|
# Note that the context is not elevated, but the True is passed in
|
|
# which will force an elevation to admin credentials even though
|
|
# the context has an auth_token.
|
|
context_client = neutronapi.get_client(my_context, True)
|
|
|
|
admin_auth = neutronapi._ADMIN_AUTH
|
|
|
|
self.assertEqual(CONF.neutron.auth_url, admin_auth.auth_url)
|
|
self.assertEqual(CONF.neutron.password, admin_auth.password)
|
|
|
|
if use_id:
|
|
self.assertEqual(CONF.neutron.tenant_id,
|
|
admin_auth.tenant_id)
|
|
self.assertEqual(CONF.neutron.user_id, admin_auth.user_id)
|
|
|
|
self.assertIsNone(admin_auth.tenant_name)
|
|
self.assertIsNone(admin_auth.username)
|
|
else:
|
|
self.assertEqual(CONF.neutron.username, admin_auth.username)
|
|
|
|
self.assertIsNone(admin_auth.tenant_id)
|
|
self.assertIsNone(admin_auth.user_id)
|
|
|
|
self.assertEqual(CONF.neutron.timeout,
|
|
neutronapi._SESSION.timeout)
|
|
|
|
self.assertEqual(
|
|
token_value,
|
|
context_client.httpclient.auth.get_token(neutronapi._SESSION))
|
|
self.assertEqual(
|
|
CONF.neutron.endpoint_override,
|
|
context_client.httpclient.get_endpoint())
|
|
|
|
def test_get_client_for_admin(self):
|
|
self._test_get_client_for_admin()
|
|
|
|
def test_get_client_for_admin_with_id(self):
|
|
self._test_get_client_for_admin(use_id=True)
|
|
|
|
def test_get_client_for_admin_context(self):
|
|
self._test_get_client_for_admin(admin_context=True)
|
|
|
|
def test_get_client_for_admin_context_with_id(self):
|
|
self._test_get_client_for_admin(use_id=True, admin_context=True)
|
|
|
|
|
|
class TestNeutronPortSecurity(test.NoDBTestCase):
|
|
|
|
def test__process_security_groups(self):
|
|
instance = objects.Instance(project_id=uuids.project_id)
|
|
mock_neutron = mock.Mock(spec=client.Client)
|
|
mock_neutron.list_security_groups.return_value = {
|
|
'security_groups': [
|
|
{
|
|
'id': uuids.sg1,
|
|
'name': 'sg1',
|
|
},
|
|
{
|
|
'id': uuids.sg2,
|
|
'name': 'sg2',
|
|
},
|
|
{
|
|
'id': uuids.sg3,
|
|
'name': 'sg3',
|
|
}
|
|
]
|
|
}
|
|
api = neutronapi.API()
|
|
api._process_security_groups(
|
|
instance, mock_neutron, ["sg1", uuids.sg2])
|
|
|
|
mock_neutron.list_security_groups.assert_called_once_with(
|
|
fields=['id', 'name'], tenant_id=uuids.project_id)
|
|
|
|
def test__process_security_groups_not_found(self):
|
|
instance = objects.Instance(project_id=uuids.project_id)
|
|
mock_neutron = mock.Mock(spec=client.Client)
|
|
mock_neutron.list_security_groups.return_value = {
|
|
'security_groups': [
|
|
{
|
|
'id': uuids.sg1,
|
|
'name': 'sg1',
|
|
},
|
|
{
|
|
'id': uuids.sg3,
|
|
'name': 'sg3',
|
|
}
|
|
]
|
|
}
|
|
api = neutronapi.API()
|
|
|
|
ex = self.assertRaises(
|
|
exception.SecurityGroupNotFound, api._process_security_groups,
|
|
instance, mock_neutron, ["sg1", uuids.sg2])
|
|
|
|
self.assertIn(uuids.sg2, str(ex))
|
|
mock_neutron.list_security_groups.assert_called_once_with(
|
|
fields=['id', 'name'], tenant_id=uuids.project_id)
|
|
|
|
def test__process_security_groups_non_unique_match(self):
|
|
instance = objects.Instance(project_id=uuids.project_id)
|
|
mock_neutron = mock.Mock(spec=client.Client)
|
|
mock_neutron.list_security_groups.return_value = {
|
|
'security_groups': [
|
|
{
|
|
'id': uuids.sg1,
|
|
'name': 'nonunique-name',
|
|
},
|
|
{
|
|
'id': uuids.sg2,
|
|
'name': 'nonunique-name',
|
|
}
|
|
]
|
|
}
|
|
api = neutronapi.API()
|
|
|
|
ex = self.assertRaises(
|
|
exception.NoUniqueMatch, api._process_security_groups,
|
|
instance, mock_neutron, ["nonunique-name", uuids.sg2])
|
|
|
|
self.assertIn("nonunique-name", str(ex))
|
|
mock_neutron.list_security_groups.assert_called_once_with(
|
|
fields=['id', 'name'], tenant_id=uuids.project_id)
|
|
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info')
|
|
@mock.patch.object(neutronapi.API, '_update_port_dns_name')
|
|
@mock.patch.object(neutronapi.API, '_create_port_minimal')
|
|
@mock.patch.object(neutronapi.API, '_populate_neutron_extension_values')
|
|
@mock.patch.object(neutronapi.API, '_check_external_network_attach')
|
|
@mock.patch.object(neutronapi.API, '_process_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_get_available_networks')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_port_ids')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def test_no_security_groups_requested(
|
|
self, mock_vif, mock_get_client,
|
|
mock_validate_requested_port_ids,
|
|
mock_get_available_networks, mock_process_security_groups,
|
|
mock_check_external_network_attach,
|
|
mock_populate_neutron_extension_values, mock_create_port,
|
|
mock_update_port_dns_name, mock_get_instance_nw_info):
|
|
nets = [
|
|
{'id': 'net1',
|
|
'name': 'net_name1',
|
|
'subnets': ['mysubnid1'],
|
|
'port_security_enabled': True},
|
|
{'id': 'net2',
|
|
'name': 'net_name2',
|
|
'subnets': ['mysubnid2'],
|
|
'port_security_enabled': True}]
|
|
onets = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id='net1'),
|
|
objects.NetworkRequest(network_id='net2')])
|
|
|
|
instance = objects.Instance(
|
|
project_id=1, availability_zone='nova', uuid=uuids.instance)
|
|
secgroups = ['default'] # Nova API provides the 'default'
|
|
|
|
mock_validate_requested_port_ids.return_value = [{}, onets]
|
|
mock_get_available_networks.return_value = nets
|
|
mock_process_security_groups.return_value = []
|
|
|
|
api = neutronapi.API()
|
|
mock_create_port.return_value = {'id': 'foo', 'mac_address': 'bar'}
|
|
api.allocate_for_instance(
|
|
mock.sentinel.context, instance, requested_networks=onets,
|
|
security_groups=secgroups)
|
|
|
|
mock_process_security_groups.assert_called_once_with(
|
|
instance, mock.ANY, [])
|
|
mock_create_port.assert_has_calls([
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net1', None, []),
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net2', None, [])],
|
|
any_order=True)
|
|
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info')
|
|
@mock.patch.object(neutronapi.API, '_update_port_dns_name')
|
|
@mock.patch.object(neutronapi.API, '_create_port_minimal')
|
|
@mock.patch.object(neutronapi.API, '_populate_neutron_extension_values')
|
|
@mock.patch.object(neutronapi.API, '_check_external_network_attach')
|
|
@mock.patch.object(neutronapi.API, '_process_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_get_available_networks')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_port_ids')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def test_security_groups_requested(
|
|
self, mock_vif, mock_get_client,
|
|
mock_validate_requested_port_ids,
|
|
mock_get_available_networks, mock_process_security_groups,
|
|
mock_check_external_network_attach,
|
|
mock_populate_neutron_extension_values, mock_create_port,
|
|
mock_update_port_dns_name, mock_get_instance_nw_info):
|
|
nets = [
|
|
{'id': 'net1',
|
|
'name': 'net_name1',
|
|
'subnets': ['mysubnid1'],
|
|
'port_security_enabled': True},
|
|
{'id': 'net2',
|
|
'name': 'net_name2',
|
|
'subnets': ['mysubnid2'],
|
|
'port_security_enabled': True}]
|
|
onets = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id='net1'),
|
|
objects.NetworkRequest(network_id='net2')])
|
|
|
|
instance = objects.Instance(
|
|
project_id=1, availability_zone='nova', uuid=uuids.instance)
|
|
secgroups = ['default', 'secgrp1', 'secgrp2']
|
|
|
|
mock_validate_requested_port_ids.return_value = [{}, onets]
|
|
mock_get_available_networks.return_value = nets
|
|
mock_process_security_groups.return_value = ['default-uuid',
|
|
'secgrp-uuid1',
|
|
'secgrp-uuid2']
|
|
|
|
api = neutronapi.API()
|
|
mock_create_port.return_value = {'id': 'foo', 'mac_address': 'bar'}
|
|
api.allocate_for_instance(
|
|
mock.sentinel.context, instance, requested_networks=onets,
|
|
security_groups=secgroups)
|
|
|
|
mock_create_port.assert_has_calls([
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net1', None,
|
|
['default-uuid', 'secgrp-uuid1', 'secgrp-uuid2']),
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net2', None,
|
|
['default-uuid', 'secgrp-uuid1', 'secgrp-uuid2'])],
|
|
any_order=True)
|
|
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info')
|
|
@mock.patch.object(neutronapi.API, '_update_port_dns_name')
|
|
@mock.patch.object(neutronapi.API, '_create_port_minimal')
|
|
@mock.patch.object(neutronapi.API, '_populate_neutron_extension_values')
|
|
@mock.patch.object(neutronapi.API, '_check_external_network_attach')
|
|
@mock.patch.object(neutronapi.API, '_process_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_get_available_networks')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_port_ids')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def test_port_security_disabled_no_security_groups_requested(
|
|
self, mock_vif, mock_get_client,
|
|
mock_validate_requested_port_ids,
|
|
mock_get_available_networks, mock_process_security_groups,
|
|
mock_check_external_network_attach,
|
|
mock_populate_neutron_extension_values, mock_create_port,
|
|
mock_update_port_dns_name, mock_get_instance_nw_info):
|
|
nets = [
|
|
{'id': 'net1',
|
|
'name': 'net_name1',
|
|
'subnets': ['mysubnid1'],
|
|
'port_security_enabled': False},
|
|
{'id': 'net2',
|
|
'name': 'net_name2',
|
|
'subnets': ['mysubnid2'],
|
|
'port_security_enabled': False}]
|
|
onets = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id='net1'),
|
|
objects.NetworkRequest(network_id='net2')])
|
|
|
|
instance = objects.Instance(
|
|
project_id=1, availability_zone='nova', uuid=uuids.instance)
|
|
secgroups = ['default'] # Nova API provides the 'default'
|
|
|
|
mock_validate_requested_port_ids.return_value = [{}, onets]
|
|
mock_get_available_networks.return_value = nets
|
|
mock_process_security_groups.return_value = []
|
|
|
|
api = neutronapi.API()
|
|
mock_create_port.return_value = {'id': 'foo', 'mac_address': 'bar'}
|
|
api.allocate_for_instance(
|
|
mock.sentinel.context, instance, requested_networks=onets,
|
|
security_groups=secgroups)
|
|
|
|
mock_process_security_groups.assert_called_once_with(
|
|
instance, mock.ANY, [])
|
|
mock_create_port.assert_has_calls([
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net1', None, []),
|
|
mock.call(
|
|
mock.sentinel.context, mock.ANY, instance, u'net2', None, [])],
|
|
any_order=True)
|
|
|
|
@mock.patch.object(neutronapi.API, 'get_instance_nw_info')
|
|
@mock.patch.object(neutronapi.API, '_update_port_dns_name')
|
|
@mock.patch.object(neutronapi.API, '_create_port_minimal')
|
|
@mock.patch.object(neutronapi.API, '_populate_neutron_extension_values')
|
|
@mock.patch.object(neutronapi.API, '_check_external_network_attach')
|
|
@mock.patch.object(neutronapi.API, '_process_security_groups')
|
|
@mock.patch.object(neutronapi.API, '_get_available_networks')
|
|
@mock.patch.object(neutronapi.API, '_validate_requested_port_ids')
|
|
@mock.patch.object(neutronapi, 'get_client')
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def test_port_security_disabled_and_security_groups_requested(
|
|
self, mock_vif, mock_get_client,
|
|
mock_validate_requested_port_ids,
|
|
mock_get_available_networks, mock_process_security_groups,
|
|
mock_check_external_network_attach,
|
|
mock_populate_neutron_extension_values, mock_create_port,
|
|
mock_update_port_dns_name, mock_get_instance_nw_info):
|
|
nets = [
|
|
{'id': 'net1',
|
|
'name': 'net_name1',
|
|
'subnets': ['mysubnid1'],
|
|
'port_security_enabled': True},
|
|
{'id': 'net2',
|
|
'name': 'net_name2',
|
|
'subnets': ['mysubnid2'],
|
|
'port_security_enabled': False}]
|
|
onets = objects.NetworkRequestList(objects=[
|
|
objects.NetworkRequest(network_id='net1'),
|
|
objects.NetworkRequest(network_id='net2')])
|
|
|
|
instance = objects.Instance(
|
|
project_id=1, availability_zone='nova', uuid=uuids.instance)
|
|
secgroups = ['default', 'secgrp1', 'secgrp2']
|
|
|
|
mock_validate_requested_port_ids.return_value = [{}, onets]
|
|
mock_get_available_networks.return_value = nets
|
|
mock_process_security_groups.return_value = ['default-uuid',
|
|
'secgrp-uuid1',
|
|
'secgrp-uuid2']
|
|
|
|
api = neutronapi.API()
|
|
self.assertRaises(
|
|
exception.SecurityGroupCannotBeApplied,
|
|
api.allocate_for_instance,
|
|
'context', instance, requested_networks=onets,
|
|
security_groups=secgroups)
|
|
|
|
mock_process_security_groups.assert_called_once_with(
|
|
instance, mock.ANY, ['default', 'secgrp1', 'secgrp2'])
|
|
|
|
|
|
class TestAPIAutoAllocateNetwork(test.NoDBTestCase):
|
|
"""Tests auto-allocation scenarios"""
|
|
|
|
def setUp(self):
|
|
super(TestAPIAutoAllocateNetwork, self).setUp()
|
|
self.api = neutronapi.API()
|
|
self.context = context.RequestContext(uuids.user_id, uuids.project_id)
|
|
|
|
def test__can_auto_allocate_network_validation_conflict(self):
|
|
# Tests that the dry-run validation with neutron fails (not ready).
|
|
ntrn = mock.Mock()
|
|
ntrn.validate_auto_allocated_topology_requirements.side_effect = \
|
|
exceptions.Conflict
|
|
self.assertFalse(self.api._can_auto_allocate_network(
|
|
self.context, ntrn))
|
|
validate = ntrn.validate_auto_allocated_topology_requirements
|
|
validate.assert_called_once_with(uuids.project_id)
|
|
|
|
def test__can_auto_allocate_network(self):
|
|
# Tests the happy path.
|
|
ntrn = mock.Mock()
|
|
self.assertTrue(self.api._can_auto_allocate_network(
|
|
self.context, ntrn))
|
|
validate = ntrn.validate_auto_allocated_topology_requirements
|
|
validate.assert_called_once_with(uuids.project_id)
|
|
|
|
def test__ports_needed_per_instance_no_reqs_no_nets(self):
|
|
# Tests no requested_networks and no available networks.
|
|
with mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[]):
|
|
self.assertEqual(
|
|
1, self.api._ports_needed_per_instance(self.context,
|
|
mock.sentinel.neutron,
|
|
None))
|
|
|
|
def test__ports_needed_per_instance_empty_reqs_no_nets(self):
|
|
# Tests empty requested_networks and no available networks.
|
|
requested_networks = objects.NetworkRequestList()
|
|
with mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[]):
|
|
self.assertEqual(
|
|
1, self.api._ports_needed_per_instance(self.context,
|
|
mock.sentinel.neutron,
|
|
requested_networks))
|
|
|
|
def test__ports_needed_per_instance_auto_reqs_no_nets_not_ready(self):
|
|
# Test for when there are no available networks and we're requested
|
|
# to auto-allocate the network but auto-allocation is not available.
|
|
net_req = objects.NetworkRequest(
|
|
network_id=net_req_obj.NETWORK_ID_AUTO)
|
|
requested_networks = objects.NetworkRequestList(objects=[net_req])
|
|
with mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[]):
|
|
with mock.patch.object(self.api, '_can_auto_allocate_network',
|
|
spec=True, return_value=False) as can_alloc:
|
|
self.assertRaises(
|
|
exception.UnableToAutoAllocateNetwork,
|
|
self.api._ports_needed_per_instance,
|
|
self.context, mock.sentinel.neutron, requested_networks)
|
|
can_alloc.assert_called_once_with(
|
|
self.context, mock.sentinel.neutron)
|
|
|
|
def test__ports_needed_per_instance_auto_reqs_no_nets_ok(self):
|
|
# Test for when there are no available networks and we're requested
|
|
# to auto-allocate the network and auto-allocation is available.
|
|
net_req = objects.NetworkRequest(
|
|
network_id=net_req_obj.NETWORK_ID_AUTO)
|
|
requested_networks = objects.NetworkRequestList(objects=[net_req])
|
|
with mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[]):
|
|
with mock.patch.object(self.api, '_can_auto_allocate_network',
|
|
spec=True, return_value=True) as can_alloc:
|
|
self.assertEqual(
|
|
1, self.api._ports_needed_per_instance(
|
|
self.context,
|
|
mock.sentinel.neutron,
|
|
requested_networks))
|
|
can_alloc.assert_called_once_with(
|
|
self.context, mock.sentinel.neutron)
|
|
|
|
def test__validate_requested_port_ids_auto_allocate(self):
|
|
# Tests that _validate_requested_port_ids doesn't really do anything
|
|
# if there is an auto-allocate network request.
|
|
net_req = objects.NetworkRequest(
|
|
network_id=net_req_obj.NETWORK_ID_AUTO)
|
|
requested_networks = objects.NetworkRequestList(objects=[net_req])
|
|
self.assertEqual(({}, []),
|
|
self.api._validate_requested_port_ids(
|
|
self.context, mock.sentinel.instance,
|
|
mock.sentinel.neutron_client, requested_networks))
|
|
|
|
def test__auto_allocate_network_conflict(self):
|
|
# Tests that we handle a 409 from Neutron when auto-allocating topology
|
|
instance = mock.Mock(project_id=self.context.project_id)
|
|
ntrn = mock.Mock()
|
|
ntrn.get_auto_allocated_topology = mock.Mock(
|
|
side_effect=exceptions.Conflict)
|
|
self.assertRaises(exception.UnableToAutoAllocateNetwork,
|
|
self.api._auto_allocate_network, instance, ntrn)
|
|
ntrn.get_auto_allocated_topology.assert_called_once_with(
|
|
instance.project_id)
|
|
|
|
def test__auto_allocate_network_network_not_found(self):
|
|
# Tests that we handle a 404 from Neutron when auto-allocating topology
|
|
instance = mock.Mock(project_id=self.context.project_id)
|
|
ntrn = mock.Mock()
|
|
ntrn.get_auto_allocated_topology.return_value = {
|
|
'auto_allocated_topology': {
|
|
'id': uuids.network_id
|
|
}
|
|
}
|
|
ntrn.show_network = mock.Mock(
|
|
side_effect=exceptions.NetworkNotFoundClient)
|
|
self.assertRaises(exception.UnableToAutoAllocateNetwork,
|
|
self.api._auto_allocate_network, instance, ntrn)
|
|
ntrn.show_network.assert_called_once_with(uuids.network_id)
|
|
|
|
def test__auto_allocate_network(self):
|
|
# Tests the happy path.
|
|
instance = mock.Mock(project_id=self.context.project_id)
|
|
ntrn = mock.Mock()
|
|
ntrn.get_auto_allocated_topology.return_value = {
|
|
'auto_allocated_topology': {
|
|
'id': uuids.network_id
|
|
}
|
|
}
|
|
ntrn.show_network.return_value = {'network': mock.sentinel.network}
|
|
self.assertEqual(mock.sentinel.network,
|
|
self.api._auto_allocate_network(instance, ntrn))
|
|
|
|
def test_allocate_for_instance_auto_allocate(self):
|
|
# Tests the happy path.
|
|
ntrn = mock.Mock()
|
|
# mock neutron.list_networks which is called from
|
|
# _get_available_networks when net_ids is empty, which it will be
|
|
# because _validate_requested_port_ids will return an empty list since
|
|
# we requested 'auto' allocation.
|
|
ntrn.list_networks.return_value = {}
|
|
|
|
fake_network = {
|
|
'id': uuids.network_id,
|
|
'subnets': [
|
|
uuids.subnet_id,
|
|
]
|
|
}
|
|
|
|
def fake_get_instance_nw_info(context, instance, **kwargs):
|
|
# assert the network and port are what was used in the test
|
|
self.assertIn('networks', kwargs)
|
|
self.assertEqual(1, len(kwargs['networks']))
|
|
self.assertEqual(uuids.network_id,
|
|
kwargs['networks'][0]['id'])
|
|
self.assertIn('port_ids', kwargs)
|
|
self.assertEqual(1, len(kwargs['port_ids']))
|
|
self.assertEqual(uuids.port_id, kwargs['port_ids'][0])
|
|
# return a fake vif
|
|
return [model.VIF(id=uuids.port_id)]
|
|
|
|
@mock.patch('nova.network.neutron.get_client', return_value=ntrn)
|
|
@mock.patch.object(self.api, '_auto_allocate_network',
|
|
return_value=fake_network)
|
|
@mock.patch.object(self.api, '_check_external_network_attach')
|
|
@mock.patch.object(self.api, '_populate_neutron_extension_values')
|
|
@mock.patch.object(self.api, '_create_port_minimal', spec=True,
|
|
return_value={'id': uuids.port_id,
|
|
'mac_address': 'foo'})
|
|
@mock.patch.object(self.api, '_update_port')
|
|
@mock.patch.object(self.api, '_update_port_dns_name')
|
|
@mock.patch.object(self.api, 'get_instance_nw_info',
|
|
fake_get_instance_nw_info)
|
|
@mock.patch('nova.objects.VirtualInterface')
|
|
def do_test(self,
|
|
mock_vif,
|
|
update_port_dsn_name_mock,
|
|
update_port_mock,
|
|
create_port_mock,
|
|
populate_ext_values_mock,
|
|
check_external_net_attach_mock,
|
|
auto_allocate_mock,
|
|
get_client_mock):
|
|
instance = fake_instance.fake_instance_obj(self.context)
|
|
net_req = objects.NetworkRequest(
|
|
network_id=net_req_obj.NETWORK_ID_AUTO)
|
|
requested_networks = objects.NetworkRequestList(objects=[net_req])
|
|
|
|
nw_info = self.api.allocate_for_instance(
|
|
self.context, instance, requested_networks)
|
|
self.assertEqual(1, len(nw_info))
|
|
self.assertEqual(uuids.port_id, nw_info[0]['id'])
|
|
# assert that we filtered available networks on admin_state_up=True
|
|
ntrn.list_networks.assert_has_calls([
|
|
mock.call(tenant_id=instance.project_id, shared=False,
|
|
admin_state_up=True),
|
|
mock.call(shared=True)])
|
|
|
|
# assert the calls to create the port are using the network that
|
|
# was auto-allocated
|
|
port_req_body = mock.ANY
|
|
create_port_mock.assert_called_once_with(
|
|
self.context, ntrn, instance, uuids.network_id,
|
|
None, # request.address (fixed IP)
|
|
[], # security_group_ids - we didn't request any
|
|
)
|
|
update_port_mock.assert_called_once_with(
|
|
ntrn, instance, uuids.port_id, port_req_body)
|
|
|
|
do_test(self)
|
|
|
|
|
|
class TestGetInstanceNetworkInfo(test.NoDBTestCase):
|
|
"""Tests rebuilding the network_info cache."""
|
|
|
|
def setUp(self):
|
|
super(TestGetInstanceNetworkInfo, self).setUp()
|
|
self.api = neutronapi.API()
|
|
self.context = context.RequestContext(uuids.user_id, uuids.project_id)
|
|
self.instance = fake_instance.fake_instance_obj(self.context)
|
|
client_mock = mock.patch('nova.network.neutron.get_client')
|
|
self.client = client_mock.start().return_value
|
|
self.addCleanup(client_mock.stop)
|
|
# This is a no-db set of tests and we don't care about refreshing the
|
|
# info_cache from the database so just mock it out.
|
|
refresh_info_cache_for_instance = mock.patch(
|
|
'nova.compute.utils.refresh_info_cache_for_instance')
|
|
refresh_info_cache_for_instance.start()
|
|
self.addCleanup(refresh_info_cache_for_instance.stop)
|
|
|
|
@staticmethod
|
|
def _get_vif_in_cache(info_cache, vif_id):
|
|
for vif in info_cache:
|
|
if vif['id'] == vif_id:
|
|
return vif
|
|
|
|
@staticmethod
|
|
def _get_fake_info_cache(vif_ids, **kwargs):
|
|
"""Returns InstanceInfoCache based on the list of provided VIF IDs"""
|
|
nwinfo = model.NetworkInfo(
|
|
[model.VIF(vif_id, **kwargs) for vif_id in vif_ids])
|
|
return objects.InstanceInfoCache(network_info=nwinfo)
|
|
|
|
@staticmethod
|
|
def _get_fake_port(port_id, **kwargs):
|
|
network_id = kwargs.get('network_id', uuids.network_id)
|
|
return {'id': port_id, 'network_id': network_id}
|
|
|
|
@staticmethod
|
|
def _get_fake_vif(context, **kwargs):
|
|
"""Returns VirtualInterface based on provided VIF ID"""
|
|
return obj_vif.VirtualInterface(context=context, **kwargs)
|
|
|
|
def test_get_nw_info_refresh_vif_id_add_vif(self):
|
|
"""Tests that a network-changed event occurred on a single port
|
|
which is not already in the cache so it's added.
|
|
"""
|
|
# The cache has one existing port.
|
|
self.instance.info_cache = self._get_fake_info_cache([uuids.old_port])
|
|
# The instance has two ports, one old, one new.
|
|
self.client.list_ports.return_value = {
|
|
'ports': [self._get_fake_port(uuids.old_port),
|
|
self._get_fake_port(uuids.new_port)]}
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[{'id': uuids.network_id}]),
|
|
mock.patch.object(self.api, '_build_vif_model',
|
|
return_value=model.VIF(uuids.new_port)),
|
|
# We should not get as far as calling _gather_port_ids_and_networks
|
|
mock.patch.object(self.api, '_gather_port_ids_and_networks',
|
|
new_callable=mock.NonCallableMock)
|
|
) as (
|
|
get_nets, build_vif, gather_ports
|
|
):
|
|
nwinfo = self.api._get_instance_nw_info(
|
|
self.context, self.instance, refresh_vif_id=uuids.new_port)
|
|
get_nets.assert_called_once_with(
|
|
self.context, self.instance.project_id,
|
|
[uuids.network_id], self.client)
|
|
# Assert that the old and new ports are in the cache.
|
|
for port_id in (uuids.old_port, uuids.new_port):
|
|
self.assertIsNotNone(self._get_vif_in_cache(nwinfo, port_id))
|
|
|
|
def test_get_nw_info_refresh_vif_id_update_vif(self):
|
|
"""Tests that a network-changed event occurred on a single port
|
|
which is already in the cache so it's updated.
|
|
"""
|
|
# The cache has two existing active VIFs.
|
|
self.instance.info_cache = self._get_fake_info_cache(
|
|
[uuids.old_port, uuids.new_port], active=True)
|
|
# The instance has two ports, one old, one new.
|
|
self.client.list_ports.return_value = {
|
|
'ports': [self._get_fake_port(uuids.old_port),
|
|
self._get_fake_port(uuids.new_port)]}
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[{'id': uuids.network_id}]),
|
|
# Fake that the port is no longer active.
|
|
mock.patch.object(self.api, '_build_vif_model',
|
|
return_value=model.VIF(
|
|
uuids.new_port, active=False)),
|
|
# We should not get as far as calling _gather_port_ids_and_networks
|
|
mock.patch.object(self.api, '_gather_port_ids_and_networks',
|
|
new_callable=mock.NonCallableMock)
|
|
) as (
|
|
get_nets, build_vif, gather_ports
|
|
):
|
|
nwinfo = self.api._get_instance_nw_info(
|
|
self.context, self.instance, refresh_vif_id=uuids.new_port)
|
|
get_nets.assert_called_once_with(
|
|
self.context, self.instance.project_id,
|
|
[uuids.network_id], self.client)
|
|
# Assert that the old and new ports are in the cache and that the
|
|
# old port is still active and the new port is not active.
|
|
old_vif = self._get_vif_in_cache(nwinfo, uuids.old_port)
|
|
self.assertIsNotNone(old_vif)
|
|
self.assertTrue(old_vif['active'])
|
|
new_vif = self._get_vif_in_cache(nwinfo, uuids.new_port)
|
|
self.assertIsNotNone(new_vif)
|
|
self.assertFalse(new_vif['active'])
|
|
|
|
def test_get_nw_info_refresh_vif_id_remove_vif(self):
|
|
"""Tests that a network-changed event occurred on a single port
|
|
which is already in the cache but not in the current list of ports
|
|
for the instance, so it's removed from the cache.
|
|
"""
|
|
# The cache has two existing VIFs.
|
|
self.instance.info_cache = self._get_fake_info_cache(
|
|
[uuids.old_port, uuids.removed_port])
|
|
# The instance has one remaining port.
|
|
self.client.list_ports.return_value = {
|
|
'ports': [self._get_fake_port(uuids.old_port)]}
|
|
# We should not get as far as calling _gather_port_ids_and_networks
|
|
with mock.patch.object(
|
|
self.api, '_gather_port_ids_and_networks',
|
|
new_callable=mock.NonCallableMock):
|
|
nwinfo = self.api._get_instance_nw_info(
|
|
self.context, self.instance, refresh_vif_id=uuids.removed_port)
|
|
# Assert that only the old port is still in the cache.
|
|
old_vif = self._get_vif_in_cache(nwinfo, uuids.old_port)
|
|
self.assertIsNotNone(old_vif)
|
|
removed_vif = self._get_vif_in_cache(nwinfo, uuids.removed_port)
|
|
self.assertIsNone(removed_vif)
|
|
|
|
def test_get_instance_nw_info_force_refresh(self):
|
|
"""Tests a full refresh of the instance info cache using information
|
|
from neutron rather than the instance's current info cache data.
|
|
"""
|
|
# Fake out an empty cache.
|
|
self.instance.info_cache = self._get_fake_info_cache([])
|
|
# The instance has one attached port in neutron.
|
|
self.client.list_ports.return_value = {
|
|
'ports': [self._get_fake_port(uuids.port_id)]}
|
|
ordered_port_list = [uuids.port_id]
|
|
|
|
with test.nested(
|
|
mock.patch.object(self.api, '_get_available_networks',
|
|
return_value=[{'id': uuids.network_id}]),
|
|
mock.patch.object(self.api, '_build_vif_model',
|
|
return_value=model.VIF(uuids.port_id)),
|
|
# We should not call _gather_port_ids_and_networks since that uses
|
|
# the existing instance.info_cache when no ports/networks are
|
|
# passed to _build_network_info_model and what we want is a full
|
|
# refresh of the ports based on what neutron says is current.
|
|
mock.patch.object(self.api, '_gather_port_ids_and_networks',
|
|
new_callable=mock.NonCallableMock),
|
|
mock.patch.object(self.api, '_get_ordered_port_list',
|
|
return_value=ordered_port_list)
|
|
) as (
|
|
get_nets, build_vif, gather_ports, mock_port_map
|
|
):
|
|
nwinfo = self.api._get_instance_nw_info(
|
|
self.context, self.instance, force_refresh=True)
|
|
get_nets.assert_called_once_with(
|
|
self.context, self.instance.project_id,
|
|
[uuids.network_id], self.client)
|
|
# Assert that the port is in the cache now.
|
|
self.assertIsNotNone(self._get_vif_in_cache(nwinfo, uuids.port_id))
|
|
|
|
def test__get_ordered_port_list(self):
|
|
"""This test if port_list is sorted by VirtualInterface id
|
|
sequence.
|
|
"""
|
|
nova_vifs = [
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_1, id=0),
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_2, id=1),
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_3, id=2),
|
|
]
|
|
# Random order.
|
|
current_neutron_ports = [
|
|
self._get_fake_port(uuids.port_id_2),
|
|
self._get_fake_port(uuids.port_id_1),
|
|
self._get_fake_port(uuids.port_id_3),
|
|
]
|
|
expected_port_list = [uuids.port_id_1,
|
|
uuids.port_id_2,
|
|
uuids.port_id_3]
|
|
with mock.patch.object(self.api, 'get_vifs_by_instance',
|
|
return_value=nova_vifs):
|
|
port_list = self.api._get_ordered_port_list(
|
|
self.context, self.instance, current_neutron_ports)
|
|
self.assertEqual(expected_port_list,
|
|
port_list)
|
|
|
|
def test__get_ordered_port_list_new_port(self):
|
|
"""This test if port_list is sorted by VirtualInterface id
|
|
sequence while new port appears.
|
|
"""
|
|
nova_vifs = [
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_1, id=0),
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_3, id=2),
|
|
]
|
|
# New port appears.
|
|
current_neutron_ports = [
|
|
self._get_fake_port(uuids.port_id_1),
|
|
self._get_fake_port(uuids.port_id_4),
|
|
self._get_fake_port(uuids.port_id_3)
|
|
]
|
|
expected_port_list = [uuids.port_id_1,
|
|
uuids.port_id_3,
|
|
uuids.port_id_4]
|
|
with mock.patch.object(self.api, 'get_vifs_by_instance',
|
|
return_value=nova_vifs):
|
|
port_list = self.api._get_ordered_port_list(
|
|
self.context, self.instance, current_neutron_ports)
|
|
self.assertEqual(expected_port_list,
|
|
port_list)
|
|
|
|
def test__get_ordered_port_list_new_port_and_deleted_vif(self):
|
|
"""This test if port_list is sorted by VirtualInterface id
|
|
sequence while new port appears along with deleted old
|
|
VirtualInterface objects.
|
|
"""
|
|
# Display also deleted VirtualInterface.
|
|
nova_vifs = [
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_1, id=0,
|
|
deleted=True),
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_2, id=3),
|
|
self._get_fake_vif(self.context,
|
|
uuid=uuids.port_id_3, id=5),
|
|
]
|
|
# Random order and new port.
|
|
current_neutron_ports = [
|
|
self._get_fake_port(uuids.port_id_4),
|
|
self._get_fake_port(uuids.port_id_3),
|
|
self._get_fake_port(uuids.port_id_2),
|
|
]
|
|
expected_port_list = [uuids.port_id_2,
|
|
uuids.port_id_3,
|
|
uuids.port_id_4]
|
|
with mock.patch.object(self.api, 'get_vifs_by_instance',
|
|
return_value=nova_vifs):
|
|
port_list = self.api._get_ordered_port_list(
|
|
self.context, self.instance, current_neutron_ports)
|
|
self.assertEqual(expected_port_list,
|
|
port_list)
|