OpenStack Networking (Neutron)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

3470 lines
161 KiB

# Copyright (c) 2013 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 copy
import functools
from unittest import mock
import weakref
import fixtures
import netaddr
from neutron_lib.agent import constants as agent_consts
from neutron_lib.api.definitions import availability_zone as az_def
from neutron_lib.api.definitions import external_net as extnet_apidef
from neutron_lib.api.definitions import multiprovidernet as mpnet_apidef
from neutron_lib.api.definitions import portbindings
from neutron_lib.api.definitions import provider_net as pnet
from neutron_lib.api import validators
from neutron_lib.callbacks import events
from neutron_lib.callbacks import exceptions as c_exc
from neutron_lib.callbacks import registry
from neutron_lib.callbacks import resources
from neutron_lib import constants
from neutron_lib import context
from neutron_lib.db import api as db_api
from neutron_lib import exceptions as exc
from neutron_lib import fixture
from neutron_lib.plugins import constants as plugin_constants
from neutron_lib.plugins import directory
from neutron_lib.plugins.ml2 import api as driver_api
from neutron_lib.plugins import utils as p_utils
from oslo_config import cfg
from oslo_db import exception as db_exc
from oslo_utils import uuidutils
import testtools
import webob
from neutron._i18n import _
from neutron.common import utils
from neutron.db import agents_db
from neutron.db import provisioning_blocks
from neutron.db import securitygroups_db as sg_db
from neutron.db import segments_db
from neutron.objects import base as base_obj
from neutron.objects import ports as port_obj
from neutron.objects import router as l3_obj
from neutron.plugins.ml2.common import exceptions as ml2_exc
from neutron.plugins.ml2 import db as ml2_db
from neutron.plugins.ml2 import driver_context
from neutron.plugins.ml2.drivers import type_vlan
from neutron.plugins.ml2 import managers
from neutron.plugins.ml2 import models
from neutron.plugins.ml2 import plugin as ml2_plugin
from neutron import quota
from neutron.services.revisions import revision_plugin
from neutron.services.segments import db as segments_plugin_db
from neutron.services.segments import plugin as segments_plugin
from neutron.tests.common import helpers
from neutron.tests.unit import _test_extension_portbindings as test_bindings
from neutron.tests.unit.agent import test_securitygroups_rpc as test_sg_rpc
from neutron.tests.unit.db import test_allowedaddresspairs_db as test_pair
from neutron.tests.unit.db import test_db_base_plugin_v2 as test_plugin
from neutron.tests.unit.db import test_ipam_pluggable_backend as test_ipam
from neutron.tests.unit.extensions import test_extra_dhcp_opt as test_dhcpopts
from neutron.tests.unit.plugins.ml2.drivers import mechanism_logger as \
mech_logger
from neutron.tests.unit.plugins.ml2.drivers import mechanism_test as mech_test
cfg.CONF.import_opt('network_vlan_ranges',
'neutron.plugins.ml2.drivers.type_vlan',
group='ml2_type_vlan')
PLUGIN_NAME = 'ml2'
DEVICE_OWNER_COMPUTE = constants.DEVICE_OWNER_COMPUTE_PREFIX + 'fake'
HOST = 'fake_host'
TEST_ROUTER_ID = 'router_id'
# TODO(marun) - Move to somewhere common for reuse
class PluginConfFixture(fixtures.Fixture):
"""Plugin configuration shared across the unit and functional tests."""
def __init__(self, plugin_name, parent_setup=None):
super(PluginConfFixture, self).__init__()
self.plugin_name = plugin_name
self.parent_setup = parent_setup
def _setUp(self):
if self.parent_setup:
self.parent_setup()
class Ml2ConfFixture(PluginConfFixture):
def __init__(self, parent_setup=None):
super(Ml2ConfFixture, self).__init__(PLUGIN_NAME, parent_setup)
class Ml2PluginV2TestCase(test_plugin.NeutronDbPluginV2TestCase):
_mechanism_drivers = ['logger', 'test']
l3_plugin = ('neutron.tests.unit.extensions.test_l3.'
'TestL3NatServicePlugin')
def get_additional_service_plugins(self):
"""Subclasses can return a dictionary of service plugins to load."""
return {}
def setup_parent(self):
"""Perform parent setup with the common plugin configuration class."""
service_plugins = {'l3_plugin_name': self.l3_plugin}
service_plugins.update(self.get_additional_service_plugins())
# Ensure that the parent setup can be called without arguments
# by the common configuration setUp.
parent_setup = functools.partial(
super(Ml2PluginV2TestCase, self).setUp,
plugin=PLUGIN_NAME,
service_plugins=service_plugins,
)
self.useFixture(Ml2ConfFixture(parent_setup))
self.port_create_status = 'DOWN'
def setUp(self):
self.ovo_push_interface_p = mock.patch(
'neutron.plugins.ml2.ovo_rpc.OVOServerRpcInterface')
self.ovo_push_interface_p.start()
# Enable the test mechanism driver to ensure that
# we can successfully call through to all mechanism
# driver apis.
cfg.CONF.set_override('mechanism_drivers',
self._mechanism_drivers,
group='ml2')
self.physnet = 'physnet1'
self.vlan_range = '1:100'
self.vlan_range2 = '200:300'
self.physnet2 = 'physnet2'
self.phys_vrange = ':'.join([self.physnet, self.vlan_range])
self.phys2_vrange = ':'.join([self.physnet2, self.vlan_range2])
cfg.CONF.set_override('network_vlan_ranges',
[self.phys_vrange, self.phys2_vrange],
group='ml2_type_vlan')
self.setup_parent()
self.driver = directory.get_plugin()
self.context = context.get_admin_context()
class TestMl2BulkToggleWithoutBulkless(Ml2PluginV2TestCase):
_mechanism_drivers = ['logger', 'test']
def test_bulk_enabled_with_bulk_drivers(self):
self.assertFalse(self._skip_native_bulk)
class TestMl2BasicGet(test_plugin.TestBasicGet,
Ml2PluginV2TestCase):
pass
class TestMl2V2HTTPResponse(test_plugin.TestV2HTTPResponse,
Ml2PluginV2TestCase):
pass
class TestMl2NetworksV2(test_plugin.TestNetworksV2,
Ml2PluginV2TestCase):
def setUp(self, plugin=None):
super(TestMl2NetworksV2, self).setUp()
# provider networks
self.pnets = [{'name': 'net1',
pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: 1,
'tenant_id': 'tenant_one'},
{'name': 'net2',
pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet2',
pnet.SEGMENTATION_ID: 210,
'tenant_id': 'tenant_one'},
{'name': 'net3',
pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet2',
pnet.SEGMENTATION_ID: 220,
'tenant_id': 'tenant_one'}
]
# multiprovider networks
self.mp_nets = [{'name': 'net4',
mpnet_apidef.SEGMENTS:
[{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet2',
pnet.SEGMENTATION_ID: 1},
{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet2',
pnet.SEGMENTATION_ID: 202}],
'tenant_id': 'tenant_one'}
]
self.nets = self.mp_nets + self.pnets
def test_network_after_create_callback(self):
after_create = mock.Mock()
registry.subscribe(after_create, resources.NETWORK,
events.AFTER_CREATE)
with self.network() as n:
after_create.assert_called_once_with(
resources.NETWORK, events.AFTER_CREATE, mock.ANY,
context=mock.ANY, network=mock.ANY)
kwargs = after_create.mock_calls[0][2]
self.assertEqual(n['network']['id'],
kwargs['network']['id'])
def test_network_precommit_create_callback(self):
precommit_create = mock.Mock()
registry.subscribe(precommit_create, resources.NETWORK,
events.PRECOMMIT_CREATE)
with self.network():
precommit_create.assert_called_once_with(
resources.NETWORK, events.PRECOMMIT_CREATE, mock.ANY,
context=mock.ANY, network=mock.ANY, request=mock.ANY)
def test_network_precommit_create_callback_aborts(self):
precommit_create = mock.Mock()
registry.subscribe(precommit_create, resources.NETWORK,
events.PRECOMMIT_CREATE)
precommit_create.side_effect = exc.InvalidInput(error_message='x')
data = {'network': {'tenant_id': 'sometenant', 'name': 'dummy',
'admin_state_up': True, 'shared': False}}
req = self.new_create_request('networks', data)
res = req.get_response(self.api)
self.assertEqual(400, res.status_int)
def test_network_precommit_update_includes_req(self):
precommit_update = mock.Mock()
registry.subscribe(precommit_update, resources.NETWORK,
events.PRECOMMIT_UPDATE)
with self.network() as n:
data = {'network': {'name': 'updated'}}
req = self.new_update_request('networks', data, n['network']['id'])
self.deserialize(self.fmt, req.get_response(self.api))
precommit_update.assert_called_once_with(
resources.NETWORK, events.PRECOMMIT_UPDATE, mock.ANY,
payload=mock.ANY)
self.assertEqual(
'updated',
precommit_update.call_args[1]['payload'].desired_state['name'])
def test_network_after_update_callback(self):
after_update = mock.Mock()
registry.subscribe(after_update, resources.NETWORK,
events.AFTER_UPDATE)
with self.network() as n:
data = {'network': {'name': 'updated'}}
req = self.new_update_request('networks', data, n['network']['id'])
self.deserialize(self.fmt, req.get_response(self.api))
after_update.assert_called_once_with(
resources.NETWORK, events.AFTER_UPDATE, mock.ANY,
context=mock.ANY, network=mock.ANY, original_network=mock.ANY)
kwargs = after_update.mock_calls[0][2]
self.assertEqual(n['network']['name'],
kwargs['original_network']['name'])
self.assertEqual('updated', kwargs['network']['name'])
def test_network_after_delete_callback(self):
after_delete = mock.Mock()
registry.subscribe(after_delete, resources.NETWORK,
events.AFTER_DELETE)
with self.network() as n:
req = self.new_delete_request('networks', n['network']['id'])
req.get_response(self.api)
after_delete.assert_called_once_with(
resources.NETWORK, events.AFTER_DELETE, mock.ANY,
context=mock.ANY, network=mock.ANY)
kwargs = after_delete.mock_calls[0][2]
self.assertEqual(n['network']['id'],
kwargs['network']['id'])
def test_create_port_obj_bulk(self):
cfg.CONF.set_override('base_mac', "12:34:56:00")
test_mac = "00-12-34-56-78-90"
num_ports = 4
plugin = directory.get_plugin()
# Most of the plugin methods are undefined in a weakproxy. This is not
# the case most fo the time - Ml2Plugin is typically the plugin here -
# but the IPAM classes that inherit this test have a weakproxy here and
# thus fail. This avoids that error.
if isinstance(plugin, weakref.ProxyTypes):
self.skipTest("Bulk port method tests do not apply to IPAM plugin")
tenant_id = 'some_tenant'
device_owner = "me"
ctx = context.Context('', tenant_id)
with self.network(tenant_id=tenant_id) as network_to_use:
net_id = network_to_use['network']['id']
port = {'port': {'name': 'port',
'network_id': net_id,
'mac_address': constants.ATTR_NOT_SPECIFIED,
'fixed_ips': constants.ATTR_NOT_SPECIFIED,
'admin_state_up': True,
'device_id': 'device_id',
'device_owner': device_owner,
'tenant_id': tenant_id}}
ports = [copy.deepcopy(port) for x in range(num_ports)]
ports[1]['port']['mac_address'] = test_mac
port_data = plugin.create_port_obj_bulk(ctx, ports)
self.assertEqual(num_ports, len(port_data))
result_macs = []
for port in port_data:
port_mac = str(port.get('mac_address'))
self.assertIsNone(validators.validate_mac_address(port_mac))
result_macs.append(port_mac)
for ip_addr in port.get('fixed_ips'):
self.assertIsNone(validators.validate_ip_address(ip_addr))
self.assertTrue(test_mac in result_macs)
def test_bulk_network_before_and_after_events_outside_of_txn(self):
# capture session states during each before and after event
before = []
after = []
b_func = lambda *a, **k: before.append(k['context'].session.is_active)
a_func = lambda *a, **k: after.append(k['context'].session.is_active)
registry.subscribe(b_func, resources.NETWORK, events.BEFORE_CREATE)
registry.subscribe(a_func, resources.NETWORK, events.AFTER_CREATE)
data = [{'tenant_id': self._tenant_id}] * 4
self._create_bulk_from_list(
self.fmt, 'network', data, context=context.get_admin_context())
# ensure events captured
self.assertTrue(before)
self.assertTrue(after)
# ensure session was closed for all
self.assertFalse(any(before))
self.assertFalse(any(after))
def _create_and_verify_networks(self, networks):
for net_idx, net in enumerate(networks):
# create
req = self.new_create_request('networks',
{'network': net})
# verify
network = self.deserialize(self.fmt,
req.get_response(self.api))['network']
if mpnet_apidef.SEGMENTS not in net:
for k, v in net.items():
self.assertEqual(net[k], network[k])
self.assertNotIn(mpnet_apidef.SEGMENTS, network)
else:
segments = network[mpnet_apidef.SEGMENTS]
expected_segments = net[mpnet_apidef.SEGMENTS]
self.assertEqual(len(expected_segments), len(segments))
for expected, actual in zip(expected_segments, segments):
self.assertEqual(expected, actual)
def _lookup_network_by_segmentation_id(self, seg_id, num_expected_nets):
params_str = "%s=%s" % (pnet.SEGMENTATION_ID, seg_id)
net_req = self.new_list_request('networks', None,
params=params_str)
networks = self.deserialize(self.fmt, net_req.get_response(self.api))
if num_expected_nets:
self.assertIsNotNone(networks)
self.assertEqual(num_expected_nets, len(networks['networks']))
else:
self.assertIsNone(networks)
return networks
def test_list_networks_with_segmentation_id(self):
self._create_and_verify_networks(self.pnets)
# verify we can find the network that we expect
lookup_vlan_id = 1
expected_net = [n for n in self.pnets
if n[pnet.SEGMENTATION_ID] == lookup_vlan_id].pop()
networks = self._lookup_network_by_segmentation_id(lookup_vlan_id, 1)
# verify all provider attributes
network = networks['networks'][0]
for attr in pnet.ATTRIBUTES:
self.assertEqual(expected_net[attr], network[attr])
def test_list_mpnetworks_with_segmentation_id(self):
self._create_and_verify_networks(self.nets)
# get all networks with seg_id=1 (including multisegment networks)
lookup_vlan_id = 1
networks = self._lookup_network_by_segmentation_id(lookup_vlan_id, 2)
# get the mpnet
networks = [n for n in networks['networks']
if mpnet_apidef.SEGMENTS in n]
network = networks.pop()
# verify attributes of the looked up item
segments = network[mpnet_apidef.SEGMENTS]
expected_segments = self.mp_nets[0][mpnet_apidef.SEGMENTS]
self.assertEqual(len(expected_segments), len(segments))
for expected, actual in zip(expected_segments, segments):
self.assertEqual(expected, actual)
def test_create_network_segment_allocation_fails(self):
plugin = directory.get_plugin()
retry_fixture = fixture.DBRetryErrorsFixture(max_retries=2)
retry_fixture.setUp()
with mock.patch.object(
plugin.type_manager, 'create_network_segments',
side_effect=db_exc.RetryRequest(ValueError())
) as f:
data = {'network': {'tenant_id': 'sometenant', 'name': 'dummy',
'admin_state_up': True, 'shared': False}}
req = self.new_create_request('networks', data)
res = req.get_response(self.api)
self.assertEqual(500, res.status_int)
# 1 + retry count
self.assertEqual(3, f.call_count)
retry_fixture.cleanUp()
def test__update_provider_network_attributes_update_attrs(self):
plugin = directory.get_plugin()
kwargs = {'arg_list': (pnet.NETWORK_TYPE, ),
pnet.NETWORK_TYPE: 'vlan'}
with self.network(**kwargs) as net:
for attribute in set(pnet.ATTRIBUTES) - {pnet.SEGMENTATION_ID}:
net_data = {attribute: net['network'][attribute]}
self.assertIsNone(
plugin._update_provider_network_attributes(
self.context, net['network'], net_data))
net_data = {attribute: 'other_value'}
self.assertRaises(
exc.InvalidInput,
plugin._update_provider_network_attributes,
self.context, net['network'], net_data)
def test__update_provider_network_attributes_segmentation_id(self):
plugin = directory.get_plugin()
with self.network() as net:
with mock.patch.object(plugin, '_update_segmentation_id') as \
mock_update_segmentation_id:
net_data = {pnet.SEGMENTATION_ID: 1000}
plugin._update_provider_network_attributes(
self.context, net['network'], net_data)
mock_update_segmentation_id.assert_called_once_with(
self.context, net['network'], net_data)
def test__update_segmentation_id_multisegment_network(self):
plugin = directory.get_plugin()
segments = [{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: 1},
{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: 2}]
with self.network(**{'arg_list': (mpnet_apidef.SEGMENTS, ),
mpnet_apidef.SEGMENTS: segments}) as net:
self.assertRaises(
exc.InvalidInput, plugin._update_segmentation_id, self.context,
net['network'], {})
def test__update_segmentation_id_ports_wrong_vif_type(self):
plugin = directory.get_plugin()
with self.network() as net:
with mock.patch.object(
port_obj.Port, 'check_network_ports_by_binding_types',
return_value=True):
self.assertRaises(
exc.InvalidInput, plugin._update_segmentation_id,
self.context, net['network'], {})
def test__update_segmentation_id_agentless_mech_drivers(self):
plugin = directory.get_plugin()
segments = [{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: 1}]
mech_drivers = plugin.mechanism_manager.ordered_mech_drivers
for mech_driver in (md.obj for md in mech_drivers if
hasattr(md.obj, 'agent_type')):
mock.patch.object(type(mech_driver), 'agent_type',
new_callable=mock.PropertyMock(return_value=None)).start()
with self.network(**{'arg_list': (mpnet_apidef.SEGMENTS, ),
mpnet_apidef.SEGMENTS: segments}) as net, \
mock.patch.object(
port_obj.Port, 'check_network_ports_by_binding_types',
return_value=False) as check_network_ports_mock, \
mock.patch.object(plugin.type_manager,
'update_network_segment'), \
mock.patch.object(plugin, 'get_agents') as mock_get_agents:
net_data = {pnet.SEGMENTATION_ID: 1000}
plugin._update_segmentation_id(self.context, net['network'],
net_data)
mock_get_agents.assert_not_called()
check_network_ports_mock.assert_called_once_with(
self.context, net['network']['id'],
[portbindings.VIF_TYPE_UNBOUND,
portbindings.VIF_TYPE_BINDING_FAILED],
negative_search=True)
def test_update_network_with_empty_body(self):
with self.network() as network:
network_id = network["network"]["id"]
network_req = self.new_update_request("networks", None,
network_id)
res = network_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("network", res.json['NeutronError']['message'])
def test_update_network_with_incorrect_resource_body(self):
with self.network() as network:
network_id = network["network"]["id"]
incorrect_body = {"incorrect": {}}
network_req = self.new_update_request("networks",
incorrect_body,
network_id)
res = network_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("network", res.json['NeutronError']['message'])
class TestMl2NetworksV2AgentMechDrivers(Ml2PluginV2TestCase):
_mechanism_drivers = ['logger', 'test', 'test_with_agent']
def test__update_segmentation_id_ports(self):
plugin = directory.get_plugin()
segments = [{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: 1}]
with self.network(**{'arg_list': (mpnet_apidef.SEGMENTS, ),
mpnet_apidef.SEGMENTS: segments}) as net, \
mock.patch.object(
port_obj.Port, 'check_network_ports_by_binding_types',
return_value=False) as check_network_ports_mock, \
mock.patch.object(plugin.type_manager,
'update_network_segment'), \
mock.patch.object(plugin, 'get_agents',
return_value=[mock.ANY]):
net_data = {pnet.SEGMENTATION_ID: 1000}
plugin._update_segmentation_id(self.context, net['network'],
net_data)
check_network_ports_mock.assert_called_once_with(
self.context, net['network']['id'],
[portbindings.VIF_TYPE_UNBOUND,
portbindings.VIF_TYPE_BINDING_FAILED,
mech_test.VIF_TYPE_TEST],
negative_search=True)
class TestExternalNetwork(Ml2PluginV2TestCase):
def _create_external_network(self):
data = {'network': {'name': 'net1',
'router:external': 'True',
'tenant_id': 'tenant_one'}}
network_req = self.new_create_request('networks', data)
network = self.deserialize(self.fmt,
network_req.get_response(self.api))
return network
def test_external_network_type_none(self):
cfg.CONF.set_default('external_network_type',
None,
group='ml2')
network = self._create_external_network()
# For external network, expected network type to be
# tenant_network_types which is by default 'local'.
self.assertEqual(constants.TYPE_LOCAL,
network['network'][pnet.NETWORK_TYPE])
# No physical network specified, expected 'None'.
self.assertIsNone(network['network'][pnet.PHYSICAL_NETWORK])
# External network will not have a segmentation id.
self.assertIsNone(network['network'][pnet.SEGMENTATION_ID])
# External network will not have multiple segments.
self.assertNotIn(mpnet_apidef.SEGMENTS, network['network'])
def test_external_network_type_vlan(self):
cfg.CONF.set_default('external_network_type',
constants.TYPE_VLAN,
group='ml2')
network = self._create_external_network()
# For external network, expected network type to be 'vlan'.
self.assertEqual(constants.TYPE_VLAN,
network['network'][pnet.NETWORK_TYPE])
# Physical network is expected.
self.assertIsNotNone(network['network'][pnet.PHYSICAL_NETWORK])
# External network will have a segmentation id.
self.assertIsNotNone(network['network'][pnet.SEGMENTATION_ID])
# External network will not have multiple segments.
self.assertNotIn(mpnet_apidef.SEGMENTS, network['network'])
class TestMl2NetworksWithVlanTransparencyBase(TestMl2NetworksV2):
data = {'network': {'name': 'net1',
mpnet_apidef.SEGMENTS:
[{pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1'}],
'tenant_id': 'tenant_one',
'vlan_transparent': 'True'}}
def setUp(self, plugin=None):
cfg.CONF.set_override('vlan_transparent', True)
super(TestMl2NetworksWithVlanTransparencyBase, self).setUp(plugin)
class TestMl2NetworksWithVlanTransparency(
TestMl2NetworksWithVlanTransparencyBase):
_mechanism_drivers = ['test']
def test_create_network_vlan_transparent_fail(self):
with mock.patch.object(mech_test.TestMechanismDriver,
'check_vlan_transparency',
return_value=False):
network_req = self.new_create_request('networks', self.data)
res = network_req.get_response(self.api)
self.assertEqual(500, res.status_int)
error_result = self.deserialize(self.fmt, res)['NeutronError']
self.assertEqual("VlanTransparencyDriverError",
error_result['type'])
def test_create_network_vlan_transparent(self):
with mock.patch.object(mech_test.TestMechanismDriver,
'check_vlan_transparency',
return_value=True):
network_req = self.new_create_request('networks', self.data)
res = network_req.get_response(self.api)
self.assertEqual(201, res.status_int)
network = self.deserialize(self.fmt, res)['network']
self.assertIn('vlan_transparent', network)
class TestMl2NetworksWithVlanTransparencyAndMTU(
TestMl2NetworksWithVlanTransparencyBase):
_mechanism_drivers = ['test']
def test_create_network_vlan_transparent_and_mtu(self):
with mock.patch.object(mech_test.TestMechanismDriver,
'check_vlan_transparency',
return_value=True):
cfg.CONF.set_override('path_mtu', 1000, group='ml2')
cfg.CONF.set_override('global_physnet_mtu', 1000)
network_req = self.new_create_request('networks', self.data)
res = network_req.get_response(self.api)
self.assertEqual(201, res.status_int)
network = self.deserialize(self.fmt, res)['network']
self.assertEqual(1000, network['mtu'])
self.assertIn('vlan_transparent', network)
self.assertTrue(network['vlan_transparent'])
self.assertTrue(network['vlan_transparent'])
class TestMl2NetworksWithAvailabilityZone(TestMl2NetworksV2):
def test_create_network_availability_zone(self):
az_hints = ['az1', 'az2']
data = {'network': {'name': 'net1',
az_def.AZ_HINTS: az_hints,
'tenant_id': 'tenant_one'}}
with mock.patch.object(agents_db.AgentAvailabilityZoneMixin,
'validate_availability_zones'):
network_req = self.new_create_request('networks', data)
res = network_req.get_response(self.api)
self.assertEqual(201, res.status_int)
network = self.deserialize(self.fmt, res)['network']
self.assertEqual(az_hints, network[az_def.AZ_HINTS])
class TestMl2SubnetsV2(test_plugin.TestSubnetsV2,
Ml2PluginV2TestCase):
def test_subnet_before_create_callback(self):
before_create = mock.Mock()
registry.subscribe(before_create, resources.SUBNET,
events.BEFORE_CREATE)
with self.subnet() as s:
before_create.assert_called_once_with(
resources.SUBNET, events.BEFORE_CREATE, mock.ANY,
context=mock.ANY, subnet=mock.ANY)
kwargs = before_create.mock_calls[0][2]
self.assertEqual(s['subnet']['cidr'], kwargs['subnet']['cidr'])
self.assertEqual(s['subnet']['network_id'],
kwargs['subnet']['network_id'])
def test_subnet_after_create_callback(self):
after_create = mock.Mock()
registry.subscribe(after_create, resources.SUBNET, events.AFTER_CREATE)
with self.subnet() as s:
after_create.assert_called_once_with(
resources.SUBNET, events.AFTER_CREATE, mock.ANY,
context=mock.ANY, subnet=mock.ANY)
kwargs = after_create.mock_calls[0][2]
self.assertEqual(s['subnet']['id'], kwargs['subnet']['id'])
def test_port_create_subnetnotfound(self):
with self.network() as n:
with self.subnet(network=n, cidr='1.1.1.0/24') as s1,\
self.subnet(network=n, cidr='1.1.2.0/24') as s2,\
self.subnet(network=n, cidr='1.1.3.0/24') as s3:
fixed_ips = [{'subnet_id': s1['subnet']['id']},
{'subnet_id': s2['subnet']['id']},
{'subnet_id': s3['subnet']['id']}]
plugin = directory.get_plugin()
origin_create_port_db = plugin.create_port_db
def create_port(ctx, port):
plugin.create_port_db = origin_create_port_db
second_ctx = context.get_admin_context()
with db_api.CONTEXT_WRITER.using(second_ctx):
setattr(second_ctx, 'GUARD_TRANSACTION', False)
plugin.delete_subnet(second_ctx, s2['subnet']['id'])
return plugin.create_port_db(ctx, port)
plugin.create_port_db = create_port
res = self._create_port(
self.fmt, s2['subnet']['network_id'], fixed_ips=fixed_ips,
device_owner=constants.DEVICE_OWNER_DHCP, subnet=s1)
self.assertIn('Subnet %s could not be found.'
% s2['subnet']['id'], res.text)
def test_update_subnet_with_empty_body(self):
with self.subnet() as subnet:
subnet_id = subnet["subnet"]["id"]
subnet_req = self.new_update_request("subnets", None,
subnet_id)
res = subnet_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("subnet", res.json['NeutronError']['message'])
def test_update_subnet_with_incorrect_resource_body(self):
with self.subnet() as subnet:
subnet_id = subnet["subnet"]["id"]
incorrect_body = {"incorrect": {}}
subnet_req = self.new_update_request("subnets", incorrect_body,
subnet_id)
res = subnet_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("subnet", res.json['NeutronError']['message'])
def test_subnet_after_update_callback(self):
after_update = mock.Mock()
registry.subscribe(after_update, resources.SUBNET, events.AFTER_UPDATE)
with self.subnet() as s:
data = {'subnet': {'name': 'updated'}}
req = self.new_update_request('subnets', data, s['subnet']['id'])
self.deserialize(self.fmt, req.get_response(self.api))
after_update.assert_called_once_with(
resources.SUBNET, events.AFTER_UPDATE, mock.ANY,
context=mock.ANY, subnet=mock.ANY,
original_subnet=mock.ANY)
kwargs = after_update.mock_calls[0][2]
self.assertEqual(s['subnet']['name'],
kwargs['original_subnet']['name'])
self.assertEqual('updated', kwargs['subnet']['name'])
def test_subnet_after_delete_callback(self):
after_delete = mock.Mock()
registry.subscribe(after_delete, resources.SUBNET, events.AFTER_DELETE)
with self.subnet() as s:
req = self.new_delete_request('subnets', s['subnet']['id'])
req.get_response(self.api)
after_delete.assert_called_once_with(
resources.SUBNET, events.AFTER_DELETE, mock.ANY,
context=mock.ANY, subnet=mock.ANY)
kwargs = after_delete.mock_calls[0][2]
self.assertEqual(s['subnet']['id'], kwargs['subnet']['id'])
def test_delete_subnet_race_with_dhcp_port_creation(self):
with self.network() as network:
with self.subnet(network=network) as subnet:
subnet_id = subnet['subnet']['id']
attempt = [0]
def create_dhcp_port(*args, **kwargs):
"""A method to emulate race condition.
Adds dhcp port in the middle of subnet delete
"""
if attempt[0] > 0:
return False
attempt[0] += 1
data = {'port': {'network_id': network['network']['id'],
'tenant_id':
network['network']['tenant_id'],
'name': 'port1',
'admin_state_up': 1,
'device_id': '',
'device_owner':
constants.DEVICE_OWNER_DHCP,
'fixed_ips': [{'subnet_id': subnet_id}]}}
plugin = directory.get_plugin()
plugin.create_port(context.get_admin_context(), data)
# we mock _subnet_check_ip_allocations with method
# that creates DHCP port 'in the middle' of subnet_delete
# causing retry this way subnet is deleted on the
# second attempt
registry.subscribe(create_dhcp_port, resources.SUBNET,
events.PRECOMMIT_DELETE)
req = self.new_delete_request('subnets', subnet_id)
res = req.get_response(self.api)
self.assertEqual(204, res.status_int)
self.assertEqual(1, attempt[0])
def test_create_subnet_check_mtu_in_mech_context(self):
plugin = directory.get_plugin()
plugin.mechanism_manager.create_subnet_precommit = mock.Mock()
net_arg = {pnet.NETWORK_TYPE: 'vxlan',
pnet.SEGMENTATION_ID: '1'}
network = self._make_network(self.fmt, 'net1', True,
arg_list=(pnet.NETWORK_TYPE,
pnet.SEGMENTATION_ID,),
**net_arg)
with self.subnet(network=network):
mock_subnet_pre = plugin.mechanism_manager.create_subnet_precommit
observerd_mech_context = mock_subnet_pre.call_args_list[0][0][0]
self.assertEqual(network['network']['mtu'],
observerd_mech_context.network.current['mtu'])
class TestMl2DbOperationBounds(test_plugin.DbOperationBoundMixin,
Ml2PluginV2TestCase):
"""Test cases to assert constant query count for list operations.
These test cases assert that an increase in the number of objects
does not result in an increase of the number of db operations. All
database lookups during a list operation should be performed in bulk
so the number of queries required for 2 objects instead of 1 should
stay the same.
"""
def setUp(self):
super(TestMl2DbOperationBounds, self).setUp()
self.kwargs = self.get_api_kwargs()
def make_network(self):
return self._make_network(self.fmt, 'name', True, **self.kwargs)
def make_subnet(self):
net = self.make_network()
setattr(self, '_subnet_count', getattr(self, '_subnet_count', 0) + 1)
cidr = '1.%s.0.0/24' % self._subnet_count
return self._make_subnet(self.fmt, net, None, cidr, **self.kwargs)
def make_port(self):
net = self.make_network()
return self._make_port(self.fmt, net['network']['id'], **self.kwargs)
def test_network_list_queries_constant(self):
self._assert_object_list_queries_constant(self.make_network,
'networks')
def test_subnet_list_queries_constant(self):
self._assert_object_list_queries_constant(self.make_subnet, 'subnets')
def test_port_list_queries_constant(self):
self._assert_object_list_queries_constant(self.make_port, 'ports')
self._assert_object_list_queries_constant(self.make_port, 'ports',
filters=['device_id'])
self._assert_object_list_queries_constant(self.make_port, 'ports',
filters=['device_id',
'device_owner'])
self._assert_object_list_queries_constant(self.make_port, 'ports',
filters=['tenant_id',
'name',
'device_id'])
class TestMl2DbOperationBoundsTenant(TestMl2DbOperationBounds):
admin = False
class TestMl2DbOperationBoundsTenantRbac(TestMl2DbOperationBoundsTenant):
def make_port_in_shared_network(self):
context_ = self._get_context()
# create shared network owned by the tenant; we use direct driver call
# because default policy does not allow users to create shared networks
net = self.driver.create_network(
context.get_admin_context(),
{'network': {'name': 'net1',
'tenant_id': context_.project_id,
'admin_state_up': True,
'shared': True}})
# create port that belongs to another tenant
return self._make_port(
self.fmt, net['id'],
set_context=True, tenant_id='fake_tenant')
def test_port_list_in_shared_network_queries_constant(self):
self._assert_object_list_queries_constant(
self.make_port_in_shared_network, 'ports')
class TestMl2RevivedAgentsBindPorts(Ml2PluginV2TestCase):
_mechanism_drivers = ['openvswitch', 'logger']
def _test__retry_binding_revived_agents(self, event, agent_status,
admin_state_up, agent_type,
ports, should_bind_ports):
plugin = directory.get_plugin()
context = mock.Mock()
agent = {
'agent_status': agent_status,
'admin_state_up': admin_state_up,
'agent_type': agent_type}
host = "test_host"
binding = mock.MagicMock(
vif_type=portbindings.VIF_TYPE_BINDING_FAILED, host=host)
for port in ports:
port.bindings = [binding]
with mock.patch(
'neutron.objects.ports.Port.get_ports_by_binding_type_and_host',
return_value=ports
) as get_ports_by_binding_type_and_host, mock.patch.object(
plugin, 'get_network', return_value=mock.Mock()
) as get_network, mock.patch(
'neutron.plugins.ml2.db.get_binding_level_objs',
return_value=None
) as get_binding_level_objs, mock.patch(
'neutron.plugins.ml2.driver_context.PortContext'
) as port_context, mock.patch.object(
plugin, '_bind_port_if_needed'
) as bind_port_if_needed:
plugin._retry_binding_revived_agents(
resources.AGENT, event, plugin,
events.DBEventPayload(
context=context, metadata={'host': host}, states=(agent,),
desired_state=agent
)
)
if (agent_status == agent_consts.AGENT_ALIVE or
not admin_state_up or
agent_type not in plugin._rebind_on_revive_agent_types):
get_ports_by_binding_type_and_host.assert_not_called()
else:
get_ports_by_binding_type_and_host.assert_called_once_with(
context, portbindings.VIF_TYPE_BINDING_FAILED, host)
if should_bind_ports:
get_network_expected_calls = [
mock.call(context, port.network_id) for port in ports]
get_network.assert_has_calls(get_network_expected_calls)
get_binding_level_expected_calls = [
mock.call(context, port.id, host) for port in ports]
get_binding_level_objs.assert_has_calls(
get_binding_level_expected_calls)
bind_port_if_needed.assert_called_once_with(port_context())
else:
get_network.assert_not_called()
get_binding_level_objs.assert_not_called()
bind_port_if_needed.assert_not_called()
def test__retry_binding_revived_agents(self):
port = mock.MagicMock(
id=uuidutils.generate_uuid())
self._test__retry_binding_revived_agents(
events.AFTER_UPDATE, agent_consts.AGENT_REVIVED, True,
constants.AGENT_TYPE_OVS, [port],
should_bind_ports=True)
def test__retry_binding_revived_agents_no_binding_failed_ports(self):
self._test__retry_binding_revived_agents(
events.AFTER_UPDATE, agent_consts.AGENT_REVIVED, True,
constants.AGENT_TYPE_OVS, [],
should_bind_ports=False)
def test__retry_binding_revived_agents_alive_agent(self):
port = mock.MagicMock(
id=uuidutils.generate_uuid())
self._test__retry_binding_revived_agents(
events.AFTER_UPDATE, agent_consts.AGENT_ALIVE, True,
constants.AGENT_TYPE_OVS, [port],
should_bind_ports=False)
def test__retry_binding_revived_agents_not_binding_agent(self):
port = mock.MagicMock(
id=uuidutils.generate_uuid())
self._test__retry_binding_revived_agents(
events.AFTER_UPDATE, agent_consts.AGENT_REVIVED, True,
"Other agent which don't support binding", [port],
should_bind_ports=False)
def test__retry_binding_revived_agents_agent_admin_state_down(self):
port = mock.MagicMock(
id=uuidutils.generate_uuid())
self._test__retry_binding_revived_agents(
events.AFTER_UPDATE, agent_consts.AGENT_REVIVED, False,
constants.AGENT_TYPE_OVS, [port],
should_bind_ports=False)
class TestMl2PortsV2(test_plugin.TestPortsV2, Ml2PluginV2TestCase):
def test__port_provisioned_with_blocks(self):
plugin = directory.get_plugin()
ups = mock.patch.object(plugin, 'update_port_status').start()
with self.port() as port:
mock.patch('neutron.plugins.ml2.plugin.db.get_port').start()
provisioning_blocks.add_provisioning_component(
self.context, port['port']['id'], 'port', 'DHCP')
plugin._port_provisioned(
'port', 'evt', 'trigger',
payload=events.DBEventPayload(
context, resource_id=port['port']['id']))
self.assertFalse(ups.called)
def test__port_provisioned_no_binding(self):
device_id = uuidutils.generate_uuid()
plugin = directory.get_plugin()
with self.network() as net:
net_id = net['network']['id']
port_id = uuidutils.generate_uuid()
port_obj.Port(self.context,
id=port_id, project_id='tenant', network_id=net_id,
mac_address=netaddr.EUI('08-00-01-02-03-04'),
admin_state_up=True, status='ACTIVE',
device_id=device_id,
device_owner=DEVICE_OWNER_COMPUTE).create()
self.assertIsNone(plugin._port_provisioned(
'port', 'evt', 'trigger', payload=events.DBEventPayload(
self.context, resource_id=port_id)))
def test__port_provisioned_port_admin_state_down(self):
plugin = directory.get_plugin()
ups = mock.patch.object(plugin, 'update_port_status').start()
port_id = 'fake_port_id'
def getitem(key):
return constants.ACTIVE
binding = mock.MagicMock(vif_type=portbindings.VIF_TYPE_OVS)
binding.__getitem__.side_effect = getitem
port = mock.MagicMock(
id=port_id, admin_state_up=False, port_binding=[binding])
with mock.patch('neutron.plugins.ml2.plugin.db.get_port',
return_value=port):
plugin._port_provisioned('port', 'evt', 'trigger',
payload=events.DBEventPayload(
self.context, resource_id=port_id))
self.assertFalse(ups.called)
def test_port_after_create_outside_transaction(self):
self.tx_open = True
receive = lambda *a, **k: setattr(self, 'tx_open',
k['context'].session.is_active)
registry.subscribe(receive, resources.PORT, events.AFTER_CREATE)
with self.port():
self.assertFalse(self.tx_open)
def test_port_after_update_outside_transaction(self):
self.tx_open = True
receive = lambda *a, **k: setattr(self, 'tx_open',
k['context'].session.is_active)
with self.port() as p:
registry.subscribe(receive, resources.PORT, events.AFTER_UPDATE)
self._update('ports', p['port']['id'],
{'port': {'name': 'update'}})
self.assertFalse(self.tx_open)
def test_port_after_delete_outside_transaction(self):
self.tx_open = True
receive = lambda *a, **k: setattr(self, 'tx_open',
k['context'].session.is_active)
with self.port() as p:
registry.subscribe(receive, resources.PORT, events.AFTER_DELETE)
self._delete('ports', p['port']['id'])
self.assertFalse(self.tx_open)
def test_create_router_port_and_fail_create_postcommit(self):
with mock.patch.object(managers.MechanismManager,
'create_port_postcommit',
side_effect=ml2_exc.MechanismDriverError(
method='create_port_postcommit')):
l3_plugin = directory.get_plugin(plugin_constants.L3)
data = {'router': {'name': 'router', 'admin_state_up': True,
'tenant_id': 'fake_tenant'}}
r = l3_plugin.create_router(self.context, data)
with self.subnet() as s:
data = {'subnet_id': s['subnet']['id']}
self.assertRaises(ml2_exc.MechanismDriverError,
l3_plugin.add_router_interface,
self.context, r['id'], data)
res_ports = self._list('ports')['ports']
self.assertEqual([], res_ports)
def test_create_router_port_and_fail_bind_port_if_needed(self):
with mock.patch.object(ml2_plugin.Ml2Plugin, '_bind_port_if_needed',
side_effect=ml2_exc.MechanismDriverError(
method='_bind_port_if_needed')):
l3_plugin = directory.get_plugin(plugin_constants.L3)
data = {'router': {'name': 'router', 'admin_state_up': True,
'tenant_id': 'fake_tenant'}}
r = l3_plugin.create_router(self.context, data)
with self.subnet() as s:
data = {'subnet_id': s['subnet']['id']}
self.assertRaises(ml2_exc.MechanismDriverError,
l3_plugin.add_router_interface,
self.context, r['id'], data)
res_ports = self._list('ports')['ports']
self.assertEqual([], res_ports)
def test_update_port_with_empty_body(self):
with self.port() as port:
port_id = port["port"]["id"]
port_req = self.new_update_request("ports", None, port_id)
res = port_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("port", res.json['NeutronError']['message'])
def test_update_port_with_incorrect_resource_body(self):
with self.port() as port:
port_id = port["port"]["id"]
incorrect_body = {"incorrect": {}}
port_req = self.new_update_request("ports", incorrect_body,
port_id)
res = port_req.get_response(self.api)
self.assertEqual(webob.exc.HTTPBadRequest.code, res.status_int)
self.assertIn("port", res.json['NeutronError']['message'])
def test_update_port_status_build(self):
with self.port() as port:
self.assertEqual('DOWN', port['port']['status'])
self.assertEqual('DOWN', self.port_create_status)
def test_notify_port_updated_for_status_change(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.port() as port:
with mock.patch.object(self.plugin,
'_notify_port_updated') as notify_mock:
port['port']['status'] = constants.PORT_STATUS_ACTIVE
plugin.update_port(ctx, port['port']['id'], port)
self.assertTrue(notify_mock.called)
def test_update_port_status_short_id(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.port() as port:
with mock.patch.object(ml2_db, 'get_binding_level_objs',
return_value=[]) as mock_gbl:
port_id = port['port']['id']
short_id = port_id[:11]
plugin.update_port_status(ctx, short_id, 'UP')
mock_gbl.assert_called_once_with(mock.ANY, port_id, mock.ANY)
def test_update_port_with_empty_data(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.port() as port:
port_id = port['port']['id']
new_port = plugin.update_port(ctx, port_id, {"port": {}})
self.assertEqual(port["port"], new_port)
def _add_fake_dhcp_agent(self):
agent = mock.Mock()
plugin = directory.get_plugin()
self.get_dhcp_mock = mock.patch.object(
plugin, 'get_dhcp_agents_hosting_networks',
return_value=[agent]).start()
def test_dhcp_provisioning_blocks_inserted_on_create_with_agents(self):
self._add_fake_dhcp_agent()
with mock.patch.object(provisioning_blocks,
'add_provisioning_component') as ap:
with self.port():
self.assertTrue(ap.called)
def test_dhcp_provisioning_blocks_skipped_on_create_with_no_dhcp(self):
self._add_fake_dhcp_agent()
with self.subnet(enable_dhcp=False) as subnet:
with mock.patch.object(provisioning_blocks,
'add_provisioning_component') as ap:
with self.port(subnet=subnet):
self.assertFalse(ap.called)
def _test_dhcp_provisioning_blocks_inserted_on_update(self, update_dict,
expected_block):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
self._add_fake_dhcp_agent()
with self.port() as port:
with mock.patch.object(provisioning_blocks,
'add_provisioning_component') as ap:
port['port'].update(update_dict)
plugin.update_port(ctx, port['port']['id'], port)
self.assertEqual(expected_block, ap.called)
def test_dhcp_provisioning_blocks_not_inserted_on_no_addr_change(self):
update = {'binding:host_id': 'newhost'}
self._test_dhcp_provisioning_blocks_inserted_on_update(update, False)
def test_dhcp_provisioning_blocks_inserted_on_addr_change(self):
update = {'binding:host_id': 'newhost',
'mac_address': '11:22:33:44:55:66'}
self._test_dhcp_provisioning_blocks_inserted_on_update(update, True)
def test_dhcp_provisioning_blocks_removed_without_dhcp_agents(self):
with mock.patch.object(provisioning_blocks,
'remove_provisioning_component') as cp:
with self.port():
self.assertTrue(cp.called)
def test_create_update_get_port_same_fixed_ips_order(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
initial_fixed_ips = [{'ip_address': '10.0.0.5'},
{'ip_address': '10.0.0.7'},
{'ip_address': '10.0.0.6'}]
with self.port(fixed_ips=initial_fixed_ips) as port:
show = plugin.get_port(ctx, port['port']['id'])
self.assertEqual(port['port']['fixed_ips'], show['fixed_ips'])
new_fixed_ips = list(reversed(initial_fixed_ips))
port['port']['fixed_ips'] = new_fixed_ips
updated = plugin.update_port(ctx, port['port']['id'], port)
self.assertEqual(show['fixed_ips'], updated['fixed_ips'])
updated = plugin.get_port(ctx, port['port']['id'])
self.assertEqual(show['fixed_ips'], updated['fixed_ips'])
def test_update_port_fixed_ip_changed(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
fixed_ip_data = [{'ip_address': '10.0.0.4'}]
with self.port(fixed_ips=fixed_ip_data) as port,\
mock.patch.object(
plugin.notifier,
'security_groups_member_updated') as sg_member_update:
port['port']['fixed_ips'][0]['ip_address'] = '10.0.0.3'
plugin.update_port(ctx, port['port']['id'], port)
self.assertTrue(sg_member_update.called)
def test_update_port_name_do_not_notify_sg(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
port_name = "port_name"
with self.port(name=port_name) as port,\
mock.patch.object(
plugin.notifier,
'security_groups_member_updated') as sg_member_update:
port['port']['name'] = 'new_port_name'
plugin.update_port(ctx, port['port']['id'], port)
self.assertFalse(sg_member_update.called)
def test_update_port_status_with_network(self):
registry.clear() # don't care about callback behavior
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.port() as port:
net = plugin.get_network(ctx, port['port']['network_id'])
with mock.patch.object(plugin, 'get_networks') as get_nets:
plugin.update_port_status(ctx, port['port']['id'], 'UP',
network=net)
self.assertFalse(get_nets.called)
def test_update_port_mac(self):
self.check_update_port_mac(
host_arg={portbindings.HOST_ID: HOST},
arg_list=(portbindings.HOST_ID,))
def test_update_port_regenerate_mac(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
data = {'port': {'mac_address': None}}
with self.port() as port:
current_mac = port['port']['mac_address']
req = self.new_update_request('ports', data, port['port']['id'])
self.assertEqual(200, req.get_response(self.api).status_int)
new_mac = plugin.get_port(ctx, port['port']['id'])['mac_address']
self.assertNotEqual(current_mac, new_mac)
self.assertTrue(netaddr.valid_mac(new_mac))
def test_update_port_mac_does_not_change(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
data = {'port': {'description': 'Port Description'}}
with self.port() as port:
current_mac = port['port']['mac_address']
req = self.new_update_request('ports', data, port['port']['id'])
self.assertEqual(200, req.get_response(self.api).status_int)
new_mac = plugin.get_port(ctx, port['port']['id'])['mac_address']
self.assertEqual(current_mac, new_mac)
def test_update_non_existent_port(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
data = {'port': {'admin_state_up': False}}
self.assertRaises(exc.PortNotFound, plugin.update_port, ctx,
'invalid-uuid', data)
def test_delete_non_existent_port(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with mock.patch.object(ml2_plugin.LOG, 'debug') as log_debug:
plugin.delete_port(ctx, 'invalid-uuid', l3_port_check=False)
log_debug.assert_has_calls([
mock.call(_("Deleting port %s"), 'invalid-uuid'),
mock.call(_("The port '%s' was deleted"), 'invalid-uuid')
])
def test_l3_cleanup_on_net_delete(self):
l3plugin = directory.get_plugin(plugin_constants.L3)
kwargs = {'arg_list': (extnet_apidef.EXTERNAL,),
extnet_apidef.EXTERNAL: True}
with self.network(**kwargs) as n:
with self.subnet(network=n, cidr='200.0.0.0/22'):
l3plugin.create_floatingip(
context.get_admin_context(),
{'floatingip': {'floating_network_id': n['network']['id'],
'tenant_id': n['network']['tenant_id'],
'dns_name': '', 'dns_domain': ''}}
)
self._delete('networks', n['network']['id'])
flips = l3plugin.get_floatingips(context.get_admin_context())
self.assertFalse(flips)
def test_create_ports_bulk_port_binding_failure(self):
ctx = context.get_admin_context()
with self.network() as net:
plugin = directory.get_plugin()
with mock.patch.object(plugin, '_process_port_binding',
side_effect=ml2_exc.MechanismDriverError(
method='create_port_bulk')) as _process_port_binding:
res = self._create_port_bulk(self.fmt, 2, net['network']['id'],
'test', True, context=ctx)
self.assertTrue(_process_port_binding.called)
# We expect a 500 as we injected a fault in the plugin
self._validate_behavior_on_bulk_failure(
res, 'ports', webob.exc.HTTPServerError.code)
def test_create_ports_bulk_with_sec_grp(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.network() as net,\
mock.patch.object(plugin.notifier,
'security_groups_member_updated') as m_upd:
res = self._create_port_bulk(self.fmt, 3, net['network']['id'],
'test', True, context=ctx)
ports = self.deserialize(self.fmt, res)
if 'ports' in ports:
used_sg = ports['ports'][0]['security_groups']
m_upd.assert_has_calls(
[mock.call(ctx, [sg]) for sg in used_sg], any_order=True)
else:
self.assertTrue('ports' in ports)
def test_create_ports_bulk_with_sec_grp_member_provider_update(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
bulk_mock_name = "security_groups_member_updated"
with self.network() as net,\
mock.patch.object(plugin.notifier, bulk_mock_name) as m_upd:
net_id = net['network']['id']
data = [{
'network_id': net_id,
'tenant_id': self._tenant_id
},
{
'network_id': net_id,
'tenant_id': self._tenant_id,
'device_owner': constants.DEVICE_OWNER_DHCP
}
]
res = self._create_bulk_from_list(self.fmt, 'port',
data, context=ctx)
ports = self.deserialize(self.fmt, res)
used_sg = ports['ports'][0]['security_groups']
m_upd.assert_called_with(ctx, used_sg)
m_upd.reset_mock()
data[0]['device_owner'] = constants.DEVICE_OWNER_DHCP
self._create_bulk_from_list(self.fmt, 'port',
data, context=ctx)
self.assertFalse(m_upd.called)
def test_create_ports_bulk_with_sec_grp_provider_update_ipv6(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
fake_prefix = '2001:db8::/64'
fake_gateway = 'fe80::1'
with self.network() as net:
with self.subnet(net,
gateway_ip=fake_gateway,
cidr=fake_prefix,
ip_version=constants.IP_VERSION_6) as snet_v6,\
mock.patch.object(
plugin.notifier,
'security_groups_member_updated') as m_upd:
net_id = net['network']['id']
data = [{
'network_id': net_id,
'tenant_id': self._tenant_id,
'fixed_ips': [{'subnet_id': snet_v6['subnet']['id']}],
'device_owner': constants.DEVICE_OWNER_ROUTER_INTF
}
]
self._create_bulk_from_list(self.fmt, 'port',
data, context=ctx)
self.assertFalse(m_upd.called)
def test_delete_port_no_notify_in_disassociate_floatingips(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
l3plugin = directory.get_plugin(plugin_constants.L3)
with self.port() as port,\
mock.patch.object(
l3plugin,
'disassociate_floatingips') as disassociate_floatingips,\
mock.patch.object(registry, 'notify') as notify:
port_id = port['port']['id']
plugin.delete_port(ctx, port_id)
# check that no notification was requested while under
# transaction
disassociate_floatingips.assert_has_calls([
mock.call(ctx, port_id, do_notify=False)
])
# check that notifier was still triggered
self.assertTrue(notify.call_counts)
def test_registry_notify_before_after_port_binding(self):
plugin = directory.get_plugin()
ctx = context.get_admin_context()
b_update_events = []
a_update_events = []
b_receiver = lambda *a, **k: b_update_events.append(k)
a_receiver = lambda *a, **k: a_update_events.append(k['port'])
registry.subscribe(b_receiver, resources.PORT,
events.BEFORE_UPDATE)
registry.subscribe(a_receiver, resources.PORT,
events.AFTER_UPDATE)
with self.port() as p:
port = {'port': {'binding:host_id': 'newhost'}}
plugin.update_port(ctx, p['port']['id'], port)
# updating in the host should result in two AFTER_UPDATE events.
# one to change the host_id, the second to commit a binding
self.assertEqual(2, len(b_update_events))
self.assertEqual({'context': ctx,
'port': {'binding:host_id': 'newhost'},
'original_port': mock.ANY},
b_update_events[0])
self.assertIn('orig_binding', b_update_events[1])
self.assertIn('new_binding', b_update_events[1])
self.assertDictContainsSubset({'context': ctx}, b_update_events[1])
self.assertDictContainsSubset({
'admin_state_up': True,
'binding:host_id': 'newhost',
'binding:vif_type': 'unbound',
'binding:vnic_type': u'normal',
'status': 'DOWN'},
b_update_events[1]['port'])
self.assertEqual('newhost', a_update_events[0]['binding:host_id'])
self.assertEqual('unbound', a_update_events[0]['binding:vif_type'])
self.assertEqual('newhost', a_update_events[1]['binding:host_id'])
self.assertNotEqual('unbound', a_update_events[1]['binding:vif_type'])
def test_check_if_compute_port_serviced_by_dvr(self):
self.assertTrue(utils.is_dvr_serviced(DEVICE_OWNER_COMPUTE))
def test_check_if_dhcp_port_serviced_by_dvr(self):
self.assertTrue(utils.is_dvr_serviced(constants.DEVICE_OWNER_DHCP))
def test_check_if_port_not_serviced_by_dvr(self):
self.assertFalse(utils.is_dvr_serviced(
constants.DEVICE_OWNER_ROUTER_INTF))
def test_disassociate_floatingips_do_notify_returns_nothing(self):
ctx = context.get_admin_context()
l3plugin = directory.get_plugin(plugin_constants.L3)
with self.port() as port:
port_id = port['port']['id']
# check that nothing is returned when notifications are handled
# by the called method
self.assertIsNone(l3plugin.disassociate_floatingips(ctx, port_id))
def test_create_port_tolerates_db_deadlock(self):
plugin = directory.get_plugin()
with self.network() as net:
with self.subnet(network=net) as subnet:
_orig = plugin._get_port
self._failed = False
def fail_once(*args, **kwargs):
if not self._failed:
self._failed = True
raise db_exc.DBDeadlock()
return _orig(*args, **kwargs)
with mock.patch.object(plugin, '_get_port',
side_effect=fail_once) as get_port_mock:
port_kwargs = {portbindings.HOST_ID: 'host1',
'subnet': subnet,
'device_id': 'deadlocktest'}
with self.port(arg_list=(portbindings.HOST_ID,),
**port_kwargs) as port:
self.assertTrue(port['port']['id'])
self.assertTrue(get_port_mock.called)
# make sure that we didn't create more than one port on
# the retry
query_params = "network_id=%s" % net['network']['id']
query_params += "&device_id=%s" % 'deadlocktest'
ports = self._list('ports', query_params=query_params)
self.assertEqual(1, len(ports['ports']))
def test_delete_port_tolerates_db_deadlock(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
with self.port() as port:
port_db = plugin._get_port(ctx, port['port']['id'])
with mock.patch.object(plugin, '_get_port') as gp:
gp.side_effect = [db_exc.DBDeadlock] + [port_db] * 3
req = self.new_delete_request('ports', port['port']['id'])
res = req.get_response(self.api)
self.assertEqual(204, res.status_int)
self.assertGreater(gp.call_count, 1)
self.assertRaises(
exc.PortNotFound, plugin.get_port, ctx, port['port']['id'])
def test_port_create_resillient_to_duplicate_records(self):
def make_port():
with self.port():
pass
self._test_operation_resillient_to_ipallocation_failure(make_port)
def test_port_update_resillient_to_duplicate_records(self):
cidr = '10.0.0.0/24'
allocation_pools = [{'start': '10.0.0.2', 'end': '10.0.0.8'}]
with self.subnet(cidr=cidr,
allocation_pools=allocation_pools) as subnet:
with self.port(subnet=subnet) as p:
data = {'port': {'fixed_ips': [{'ip_address': '10.0.0.9'}]}}
req = self.new_update_request('ports', data, p['port']['id'])
def do_request():
self.assertEqual(200,
req.get_response(self.api).status_int)
self._test_operation_resillient_to_ipallocation_failure(
do_request)
def _test_operation_resillient_to_ipallocation_failure(self, func):
class IPAllocationsGrenade(object):
insert_ip_called = False
except_raised = False
def execute(self, con, curs, stmt, *args, **kwargs):
if 'INSERT INTO ipallocations' in stmt:
self.insert_ip_called = True
def commit(self, con):
# we blow up on commit to simulate another thread/server
# stealing our IP before our transaction was done
if self.insert_ip_called and not self.except_raised:
self.except_raised = True
raise db_exc.DBDuplicateEntry()
listener = IPAllocationsGrenade()
engine = db_api.CONTEXT_WRITER.get_engine()
db_api.sqla_listen(engine, 'before_cursor_execute',
listener.execute)
db_api.sqla_listen(engine, 'commit', listener.commit)
func()
# make sure that the grenade went off during the commit
self.assertTrue(listener.except_raised)
def test_list_ports_filtered_by_fixed_ip_substring(self):
# for this test we need to enable overlapping ips
cfg.CONF.set_default('allow_overlapping_ips', True)
with self.port() as port1, self.port():
fixed_ips = port1['port']['fixed_ips'][0]
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=subnet_id%%3D%s
""".strip() % (fixed_ips['ip_address'][:-1],
fixed_ips['subnet_id'])
self._test_list_resources('port', [port1],
query_params=query_params)
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=subnet_id%%3D%s
""".strip() % (fixed_ips['ip_address'][1:],
fixed_ips['subnet_id'])
self._test_list_resources('port', [port1],
query_params=query_params)
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=subnet_id%%3D%s
""".strip() % ('192.168.',
fixed_ips['subnet_id'])
self._test_list_resources('port', [],
query_params=query_params)
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=subnet_id%%3D%s&limit=1
""".strip() % ('192.168.',
fixed_ips['subnet_id'])
self._test_list_resources('port', [],
query_params=query_params)
def test_list_ports_filtered_by_fixed_ip_substring_dual_stack(self):
with self.subnet() as subnet:
# Get a IPv4 and IPv6 address
tenant_id = subnet['subnet']['tenant_id']
net_id = subnet['subnet']['network_id']
res = self._create_subnet(
self.fmt,
tenant_id=tenant_id,
net_id=net_id,
cidr='2607:f0d0:1002:51::/124',
ip_version=constants.IP_VERSION_6,
gateway_ip=constants.ATTR_NOT_SPECIFIED)
subnet2 = self.deserialize(self.fmt, res)
kwargs = {"fixed_ips":
[{'subnet_id': subnet['subnet']['id']},
{'subnet_id': subnet2['subnet']['id']}]}
res = self._create_port(self.fmt, net_id=net_id, **kwargs)
port1 = self.deserialize(self.fmt, res)
res = self._create_port(self.fmt, net_id=net_id, **kwargs)
port2 = self.deserialize(self.fmt, res)
fixed_ips = port1['port']['fixed_ips']
self.assertEqual(2, len(fixed_ips))
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=ip_address%%3D%s
""".strip() % (fixed_ips[0]['ip_address'][:-1],
fixed_ips[1]['ip_address'])
self._test_list_resources('port', [port1],
query_params=query_params)
query_params = """
fixed_ips=ip_address_substr%%3D%s&fixed_ips=ip_address%%3D%s
""".strip() % ('192.168.',
fixed_ips[1]['ip_address'])
self._test_list_resources('port', [],
query_params=query_params)
self._delete('ports', port1['port']['id'])
self._delete('ports', port2['port']['id'])
def test_list_ports_filtered_by_security_groups(self):
# for this test we need to enable overlapping ips
cfg.CONF.set_default('allow_overlapping_ips', True)
ctx = context.get_admin_context()
with self.port() as port1, self.port() as port2:
query_params = "security_groups=%s" % (
port1['port']['security_groups'][0])
ports_data = self._list('ports', query_params=query_params)
self.assertEqual(set([port1['port']['id'], port2['port']['id']]),
set([port['id'] for port in ports_data['ports']]))
self.assertEqual(2, len(ports_data['ports']))
query_params = "security_groups=%s&limit=1" % (
port1['port']['security_groups'][0])
ports_data = self._list('ports', query_params=query_params)
self.assertIn(ports_data['ports'][0]['id'],
[port1['port']['id'], port2['port']['id']])
self.assertEqual(1, len(ports_data['ports']))
query_params = "security_groups=%s&id=%s" % (
port1['port']['security_groups'][0],
port1['port']['id'])
ports_data = self._list('ports', query_params=query_params)
self.assertEqual(port1['port']['id'], ports_data['ports'][0]['id'])
self.assertEqual(1, len(ports_data['ports']))
temp_sg = {'security_group': {'tenant_id': 'some_tenant',
'name': '', 'description': 's'}}
sg_dbMixin = sg_db.SecurityGroupDbMixin()
sg = sg_dbMixin.create_security_group(ctx, temp_sg)
sg_dbMixin._delete_port_security_group_bindings(
ctx, port2['port']['id'])
sg_dbMixin._create_port_security_group_binding(
ctx, port2['port']['id'], sg['id'])
port2['port']['security_groups'][0] = sg['id']
query_params = "security_groups=%s&id=%s" % (
port1['port']['security_groups'][0],
port1['port']['id'])
ports_data = self._list('ports', query_params=query_params)
self.assertEqual(port1['port']['id'], ports_data['ports'][0]['id'])
self.assertEqual(1, len(ports_data['ports']))
query_params = "security_groups=%s&id=%s" % (
(port2['port']['security_groups'][0],
port2['port']['id']))
ports_data = self._list('ports', query_params=query_params)
self.assertEqual(port2['port']['id'], ports_data['ports'][0]['id'])
self.assertEqual(1, len(ports_data['ports']))
class TestMl2PortsV2WithRevisionPlugin(Ml2PluginV2TestCase):
def setUp(self):
super(TestMl2PortsV2WithRevisionPlugin, self).setUp()
self.revision_plugin = revision_plugin.RevisionPlugin()
def test_update_port_status_bumps_revision(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
host_arg = {portbindings.HOST_ID: HOST}
with self.port(arg_list=(portbindings.HOST_ID,),
**host_arg) as port:
port = plugin.get_port(ctx, port['port']['id'])
updated_ports = []
receiver = lambda *a, **k: updated_ports.append(k['port'])
registry.subscribe(receiver, resources.PORT,
events.AFTER_UPDATE)
plugin.update_port_status(
ctx, port['id'],
constants.PORT_STATUS_ACTIVE, host=HOST)
self.assertGreater(updated_ports[0]['revision_number'],
port['revision_number'])
def test_bind_port_bumps_revision(self):
updated_ports = []
created_ports = []
ureceiver = lambda *a, **k: updated_ports.append(k['port'])
creceiver = lambda *a, **k: created_ports.append(k['port'])
registry.subscribe(ureceiver, resources.PORT,
events.AFTER_UPDATE)
registry.subscribe(creceiver, resources.PORT,
events.AFTER_CREATE)
host_arg = {portbindings.HOST_ID: HOST}
with self.port(arg_list=(portbindings.HOST_ID,),
**host_arg):
self.assertGreater(updated_ports[0]['revision_number'],
created_ports[0]['revision_number'])
def test_update_port_status_dvr_port_no_update_on_same_status(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
# enable subscription for events
p_update_receiver = mock.Mock()
registry.subscribe(p_update_receiver, resources.PORT,
events.AFTER_UPDATE)
host_arg = {portbindings.HOST_ID: HOST}
with self.port(device_owner=constants.DEVICE_OWNER_DVR_INTERFACE,
device_id=TEST_ROUTER_ID,
arg_list=(portbindings.HOST_ID,),
**host_arg) as port:
ml2_db.ensure_distributed_port_binding(ctx, port['port']['id'],
HOST)
p_update_receiver.reset_mock()
plugin.update_port_status(
ctx, port['port']['id'],
constants.PORT_STATUS_ACTIVE, host=HOST)
self.assertTrue(p_update_receiver.called)
after_1 = plugin.get_port(ctx, port['port']['id'])
p_update_receiver.reset_mock()
plugin.update_port_status(
ctx, port['port']['id'],
constants.PORT_STATUS_ACTIVE, host=HOST)
self.assertFalse(p_update_receiver.called)
after_2 = plugin.get_port(ctx, port['port']['id'])
self.assertEqual(after_1['revision_number'],
after_2['revision_number'])
class TestMl2PortsV2WithL3(test_plugin.TestPortsV2, Ml2PluginV2TestCase):
"""For testing methods that require the L3 service plugin."""
l3_plugin = 'neutron.services.l3_router.l3_router_plugin.L3RouterPlugin'
def get_additional_service_plugins(self):
return {'flavors': 'flavors'}
def test_update_port_status_notify_port_event_after_update(self):
ctx = context.get_admin_context()
plugin = directory.get_plugin()
l3plugin = directory.get_plugin(plugin_constants.L3)
host_arg = {portbindings.HOST_ID: HOST}
with mock.patch.object(l3plugin.l3_rpc_notifier,
'routers_updated_on_host') as mock_updated:
with self.port(device_owner=constants.DEVICE_OWNER_ROUTER_HA_INTF,
device_id=TEST_ROUTER_ID,
arg_list=(portbindings.HOST_ID,),
**host_arg) as port:
plugin.update_port_status(
ctx, port['port']['id'],
constants.PORT_STATUS_ACTIVE, host=HOST)
mock_updated.assert_called_once_with(
mock.ANY, [TEST_ROUTER_ID], HOST)
class TestMl2PluginOnly(Ml2PluginV2TestCase):
"""For testing methods that don't call drivers"""
def test__verify_service_plugins_requirements(self):
plugin = directory.get_plugin()
with mock.patch.dict(ml2_plugin.SERVICE_PLUGINS_REQUIRED_DRIVERS,
{self.l3_plugin: self._mechanism_drivers}),\
mock.patch.object(plugin.extension_manager,
'names',
return_value=self._mechanism_drivers):
plugin._verify_service_plugins_requirements()
def test__verify_service_plugins_requirements_missing_driver(self):
plugin = directory.get_plugin()
with mock.patch.dict(ml2_plugin.SERVICE_PLUGINS_REQUIRED_DRIVERS,
{self.l3_plugin: ['test_required_driver']}),\
mock.patch.object(plugin.extension_manager,
'names',
return_value=self._mechanism_drivers):
self.assertRaises(
ml2_exc.ExtensionDriverNotFound,
plugin._verify_service_plugins_requirements
)
def _test_check_mac_update_allowed(self, vif_type, expect_change=True):
plugin = directory.get_plugin()
port = {'mac_address': "fake_mac", 'id': "fake_id"}
if expect_change:
new_attrs = {"mac_address": "dummy_mac"}
else:
new_attrs = {"mac_address": port['mac_address']}
binding = mock.Mock()
binding.vif_type = vif_type
mac_changed = plugin._check_mac_update_allowed(port, new_attrs,
binding)
self.assertEqual(expect_change, mac_changed)
def test_check_mac_update_allowed_if_no_mac_change(self):
self._test_check_mac_update_allowed(portbindings.VIF_TYPE_UNBOUND,
expect_change=False)
def test_check_mac_update_allowed_unless_bound(self):
with testtools.ExpectedException(exc.PortBound):
self._test_check_mac_update_allowed(portbindings.VIF_TYPE_OVS)
def _test_reset_mac_for_direct_physical(self, direct_physical=True,
unbinding=True):
plugin = directory.get_plugin()
port = {'device_id': '123', 'device_owner': 'compute:nova'}
new_attrs = ({'device_id': '', 'device_owner': ''} if unbinding else
{'name': 'new'})
binding = mock.Mock()
binding.vnic_type = (
portbindings.VNIC_DIRECT_PHYSICAL if direct_physical else
portbindings.VNIC_NORMAL)
new_mac = plugin._reset_mac_for_direct_physical(
port, new_attrs, binding)
if direct_physical and unbinding:
self.assertTrue(new_mac)
self.assertIsNotNone(new_attrs.get('mac_address'))
else:
self.assertFalse(new_mac)
self.assertIsNone(new_attrs.get('mac_address'))
def test_reset_mac_for_direct_physical(self):
self._test_reset_mac_for_direct_physical()
def test_reset_mac_for_direct_physical_not_physycal(self):
self._test_reset_mac_for_direct_physical(False, True)
def test_reset_mac_for_direct_physical_no_unbinding(self):
self._test_reset_mac_for_direct_physical(True, False)
def test_reset_mac_for_direct_physical_no_unbinding_not_physical(self):
self._test_reset_mac_for_direct_physical(False, False)
def test__device_to_port_id_prefix_names(self):
input_output = [('sg-abcdefg', 'abcdefg'),
('tap123456', '123456'),
('qvo567890', '567890')]
for device, expected in input_output:
self.assertEqual(expected,
ml2_plugin.Ml2Plugin._device_to_port_id(
self.context, device))
def test__device_to_port_id_mac_address(self):
with self.port() as p:
mac = p['port']['mac_address']
port_id = p['port']['id']
self.assertEqual(port_id,
ml2_plugin.Ml2Plugin._device_to_port_id(
self.context, mac))
def test__device_to_port_id_not_uuid_not_mac(self):
dev = '1234567'
self.assertEqual(dev, ml2_plugin.Ml2Plugin._device_to_port_id(
self.context, dev))
def test__device_to_port_id_UUID(self):
port_id = uuidutils.generate_uuid()
self.assertEqual(port_id, ml2_plugin.Ml2Plugin._device_to_port_id(
self.context, port_id))
@mock.patch.object(ml2_db, 'clear_binding_levels')
@mock.patch.object(port_obj.PortBinding, 'delete_objects')
def test_delete_port_binding_delete_binding_and_levels(
self,
clear_bl_mock,
delete_port_binding_mock):
port_id = uuidutils.generate_uuid()
host = 'fake-host'
plugin = directory.get_plugin()
plugin.delete_port_binding(self.context, host, port_id)
self.assertTrue(clear_bl_mock.called_with(self.context,
port_id=port_id,
host=host))
self.assertTrue(delete_port_binding_mock.called_with(self.context,
host=host,
port_id=port_id))
class Test_GetNetworkMtu(Ml2PluginV2TestCase):
def test_get_mtu_with_physical_net(self):
plugin = directory.get_plugin()
mock_type_driver = mock.MagicMock()
plugin.type_manager.drivers['driver1'] = mock.Mock()
plugin.type_manager.drivers['driver1'].obj = mock_type_driver
net = {
'name': 'net1',
'network_type': 'driver1',
'physical_network': 'physnet1',
}
plugin._get_network_mtu(net)
mock_type_driver.get_mtu.assert_called_once_with('physnet1')
def _register_type_driver_with_mtu(self, driver, mtu):
plugin = directory.get_plugin()
class FakeDriver