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.
330 lines
15 KiB
330 lines
15 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 mock
|
|
from neutron_lib.api.definitions import portbindings
|
|
from neutron_lib import constants as const
|
|
from neutron_lib import context
|
|
from neutron_lib.plugins import directory
|
|
from oslo_serialization import jsonutils
|
|
|
|
from neutron.conf.plugins.ml2.drivers import driver_type
|
|
from neutron.plugins.ml2 import config
|
|
from neutron.plugins.ml2 import driver_context
|
|
from neutron.plugins.ml2 import models as ml2_models
|
|
from neutron.tests.unit.db import test_db_base_plugin_v2 as test_plugin
|
|
|
|
|
|
class PortBindingTestCase(test_plugin.NeutronDbPluginV2TestCase):
|
|
|
|
def setUp(self):
|
|
# Enable the test mechanism driver to ensure that
|
|
# we can successfully call through to all mechanism
|
|
# driver apis.
|
|
config.cfg.CONF.set_override('mechanism_drivers',
|
|
['logger', 'test'],
|
|
'ml2')
|
|
|
|
# NOTE(dasm): ml2_type_vlan requires to be registered before used.
|
|
# This piece was refactored and removed from .config, so it causes
|
|
# a problem, when tests are executed with pdb.
|
|
# There is no problem when tests are running without debugger.
|
|
driver_type.register_ml2_drivers_vlan_opts()
|
|
config.cfg.CONF.set_override('network_vlan_ranges',
|
|
['physnet1:1000:1099'],
|
|
group='ml2_type_vlan')
|
|
super(PortBindingTestCase, self).setUp('ml2')
|
|
self.port_create_status = 'DOWN'
|
|
self.plugin = directory.get_plugin()
|
|
self.plugin.start_rpc_listeners()
|
|
|
|
def _check_response(self, port, vif_type, has_port_filter, bound, status):
|
|
self.assertEqual(vif_type, port[portbindings.VIF_TYPE])
|
|
vif_details = port[portbindings.VIF_DETAILS]
|
|
port_status = port['status']
|
|
if bound:
|
|
# TODO(rkukura): Replace with new VIF security details
|
|
self.assertEqual(has_port_filter,
|
|
vif_details[portbindings.CAP_PORT_FILTER])
|
|
self.assertEqual(status or 'DOWN', port_status)
|
|
else:
|
|
self.assertEqual('DOWN', port_status)
|
|
|
|
def _test_port_binding(self, host, vif_type, has_port_filter, bound,
|
|
status=None, network_type='local'):
|
|
mac_address = 'aa:aa:aa:aa:aa:aa'
|
|
host_arg = {portbindings.HOST_ID: host,
|
|
'mac_address': mac_address}
|
|
with self.port(name='name', arg_list=(portbindings.HOST_ID,),
|
|
**host_arg) as port:
|
|
self._check_response(port['port'], vif_type, has_port_filter,
|
|
bound, status)
|
|
port_id = port['port']['id']
|
|
neutron_context = context.get_admin_context()
|
|
details = self.plugin.endpoints[0].get_device_details(
|
|
neutron_context, agent_id="theAgentId", device=port_id)
|
|
if bound:
|
|
self.assertEqual(network_type, details['network_type'])
|
|
self.assertEqual(mac_address, details['mac_address'])
|
|
else:
|
|
self.assertNotIn('network_type', details)
|
|
self.assertNotIn('mac_address', details)
|
|
|
|
def test_unbound(self):
|
|
self._test_port_binding("",
|
|
portbindings.VIF_TYPE_UNBOUND,
|
|
False, False)
|
|
|
|
def test_binding_failed(self):
|
|
self._test_port_binding("host-fail",
|
|
portbindings.VIF_TYPE_BINDING_FAILED,
|
|
False, False)
|
|
|
|
def test_binding_no_filter(self):
|
|
self._test_port_binding("host-ovs-no_filter",
|
|
portbindings.VIF_TYPE_OVS,
|
|
False, True)
|
|
|
|
def test_binding_filter(self):
|
|
self._test_port_binding("host-bridge-filter",
|
|
portbindings.VIF_TYPE_BRIDGE,
|
|
True, True)
|
|
|
|
def test_binding_status_active(self):
|
|
self._test_port_binding("host-ovs-filter-active",
|
|
portbindings.VIF_TYPE_OVS,
|
|
True, True, 'ACTIVE')
|
|
|
|
def test_update_port_binding_no_binding(self):
|
|
ctx = context.get_admin_context()
|
|
with self.port(name='name') as port:
|
|
# emulating concurrent binding deletion
|
|
with ctx.session.begin():
|
|
for item in (ctx.session.query(ml2_models.PortBinding).
|
|
filter_by(port_id=port['port']['id'])):
|
|
ctx.session.delete(item)
|
|
self.assertIsNone(
|
|
self.plugin.get_bound_port_context(ctx, port['port']['id']))
|
|
|
|
def test_hierarchical_binding(self):
|
|
self._test_port_binding("host-hierarchical",
|
|
portbindings.VIF_TYPE_OVS,
|
|
False, True, network_type='vlan')
|
|
|
|
def test_get_bound_port_context_cache_hit(self):
|
|
ctx = context.get_admin_context()
|
|
with self.port(name='name') as port:
|
|
cached_network_id = port['port']['network_id']
|
|
some_network = {'id': cached_network_id}
|
|
cached_networks = {cached_network_id: some_network}
|
|
self.plugin.get_network = mock.Mock(return_value=some_network)
|
|
self.plugin.get_bound_port_context(ctx, port['port']['id'],
|
|
cached_networks=cached_networks)
|
|
self.assertFalse(self.plugin.get_network.called)
|
|
|
|
def _test_update_port_binding(self, host, new_host=None):
|
|
with mock.patch.object(self.plugin,
|
|
'_notify_port_updated') as notify_mock:
|
|
host_arg = {portbindings.HOST_ID: host}
|
|
update_body = {'name': 'test_update'}
|
|
if new_host is not None:
|
|
update_body[portbindings.HOST_ID] = new_host
|
|
with self.port(name='name', arg_list=(portbindings.HOST_ID,),
|
|
**host_arg) as port:
|
|
neutron_context = context.get_admin_context()
|
|
updated_port = self._update('ports', port['port']['id'],
|
|
{'port': update_body},
|
|
neutron_context=neutron_context)
|
|
port_data = updated_port['port']
|
|
if new_host is not None:
|
|
self.assertEqual(new_host,
|
|
port_data[portbindings.HOST_ID])
|
|
else:
|
|
self.assertEqual(host, port_data[portbindings.HOST_ID])
|
|
if new_host is not None and new_host != host:
|
|
notify_mock.assert_called_once_with(mock.ANY)
|
|
else:
|
|
self.assertFalse(notify_mock.called)
|
|
|
|
def test_update_with_new_host_binding_notifies_agent(self):
|
|
self._test_update_port_binding('host-ovs-no_filter',
|
|
'host-bridge-filter')
|
|
|
|
def test_update_with_same_host_binding_does_not_notify(self):
|
|
self._test_update_port_binding('host-ovs-no_filter',
|
|
'host-ovs-no_filter')
|
|
|
|
def test_update_without_binding_does_not_notify(self):
|
|
self._test_update_port_binding('host-ovs-no_filter')
|
|
|
|
def testt_update_from_empty_to_host_binding_notifies_agent(self):
|
|
self._test_update_port_binding('', 'host-ovs-no_filter')
|
|
|
|
def test_update_from_host_to_empty_binding_notifies_agent(self):
|
|
self._test_update_port_binding('host-ovs-no_filter', '')
|
|
|
|
def test_process_binding_port_host_id_changed(self):
|
|
ctx = context.get_admin_context()
|
|
plugin = directory.get_plugin()
|
|
host_id = {portbindings.HOST_ID: 'host1'}
|
|
with self.port(**host_id) as port:
|
|
# Since the port is DOWN at first
|
|
# It's necessary to make its status ACTIVE for this test
|
|
plugin.update_port_status(ctx, port['port']['id'],
|
|
const.PORT_STATUS_ACTIVE)
|
|
|
|
attrs = port['port']
|
|
attrs['status'] = const.PORT_STATUS_ACTIVE
|
|
original_port = attrs.copy()
|
|
attrs['binding:host_id'] = 'host2'
|
|
updated_port = attrs.copy()
|
|
network = {'id': attrs['network_id']}
|
|
binding = ml2_models.PortBinding(
|
|
port_id=original_port['id'],
|
|
host=original_port['binding:host_id'],
|
|
vnic_type=original_port['binding:vnic_type'],
|
|
profile=jsonutils.dumps(original_port['binding:profile']),
|
|
vif_type=original_port['binding:vif_type'],
|
|
vif_details=original_port['binding:vif_details'])
|
|
levels = []
|
|
mech_context = driver_context.PortContext(
|
|
plugin, ctx, updated_port, network, binding, levels,
|
|
original_port=original_port)
|
|
|
|
plugin._process_port_binding(mech_context, port['port'])
|
|
self.assertEqual(const.PORT_STATUS_DOWN, updated_port['status'])
|
|
port_dict = plugin.get_port(ctx, port['port']['id'])
|
|
self.assertEqual(const.PORT_STATUS_DOWN, port_dict['status'])
|
|
|
|
def test_distributed_binding(self):
|
|
ctx = context.get_admin_context()
|
|
with self.port(device_owner=const.DEVICE_OWNER_DVR_INTERFACE) as port:
|
|
port_id = port['port']['id']
|
|
|
|
# Verify port's VIF type and status.
|
|
self.assertEqual(portbindings.VIF_TYPE_DISTRIBUTED,
|
|
port['port'][portbindings.VIF_TYPE])
|
|
self.assertEqual('DOWN', port['port']['status'])
|
|
|
|
# Update port to bind for a host.
|
|
self.plugin.update_distributed_port_binding(ctx, port_id, {'port':
|
|
{portbindings.HOST_ID: 'host-ovs-no_filter',
|
|
'device_id': 'router1'}})
|
|
|
|
# Get port and verify VIF type and status unchanged.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual(portbindings.VIF_TYPE_DISTRIBUTED,
|
|
port['port'][portbindings.VIF_TYPE])
|
|
self.assertEqual('DOWN', port['port']['status'])
|
|
|
|
# Get and verify binding details for host
|
|
details = self.plugin.endpoints[0].get_device_details(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
self.assertEqual('local', details['network_type'])
|
|
|
|
# Get port and verify VIF type and changed status.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual(portbindings.VIF_TYPE_DISTRIBUTED,
|
|
port['port'][portbindings.VIF_TYPE])
|
|
self.assertEqual('BUILD', port['port']['status'])
|
|
|
|
# Mark device up.
|
|
self.plugin.endpoints[0].update_device_up(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
|
|
# Get port and verify VIF type and changed status.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual(portbindings.VIF_TYPE_DISTRIBUTED,
|
|
port['port'][portbindings.VIF_TYPE])
|
|
self.assertEqual('ACTIVE', port['port']['status'])
|
|
|
|
# Mark device down.
|
|
self.plugin.endpoints[0].update_device_down(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
|
|
# Get port and verify VIF type and changed status.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual(portbindings.VIF_TYPE_DISTRIBUTED,
|
|
port['port'][portbindings.VIF_TYPE])
|
|
self.assertEqual('DOWN', port['port']['status'])
|
|
|
|
def test_distributed_binding_multi_host_status(self):
|
|
ctx = context.get_admin_context()
|
|
with self.port(device_owner=const.DEVICE_OWNER_DVR_INTERFACE) as port:
|
|
port_id = port['port']['id']
|
|
|
|
# Update port to bind for 1st host.
|
|
self.plugin.update_distributed_port_binding(ctx, port_id, {'port':
|
|
{portbindings.HOST_ID: 'host-ovs-no_filter',
|
|
'device_id': 'router1'}})
|
|
|
|
# Mark 1st device up.
|
|
self.plugin.endpoints[0].update_device_up(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
|
|
# Get port and verify status is ACTIVE.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual('ACTIVE', port['port']['status'])
|
|
|
|
# Update port to bind for a 2nd host.
|
|
self.plugin.update_distributed_port_binding(ctx, port_id, {'port':
|
|
{portbindings.HOST_ID: 'host-bridge-filter',
|
|
'device_id': 'router1'}})
|
|
|
|
# Mark 2nd device up.
|
|
self.plugin.endpoints[0].update_device_up(
|
|
ctx, agent_id="the2ndAgentId", device=port_id,
|
|
host='host-bridge-filter')
|
|
|
|
# Get port and verify status unchanged.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual('ACTIVE', port['port']['status'])
|
|
|
|
# Mark 1st device down.
|
|
self.plugin.endpoints[0].update_device_down(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
|
|
# Get port and verify status unchanged.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual('ACTIVE', port['port']['status'])
|
|
|
|
# Mark 2nd device down.
|
|
self.plugin.endpoints[0].update_device_down(
|
|
ctx, agent_id="the2ndAgentId", device=port_id,
|
|
host='host-bridge-filter')
|
|
|
|
# Get port and verify status is DOWN.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual('DOWN', port['port']['status'])
|
|
|
|
def test_distributed_binding_update_unbound_host(self):
|
|
ctx = context.get_admin_context()
|
|
with self.port(device_owner=const.DEVICE_OWNER_DVR_INTERFACE) as port:
|
|
port_id = port['port']['id']
|
|
|
|
# Mark device up without first binding on host.
|
|
self.plugin.endpoints[0].update_device_up(
|
|
ctx, agent_id="theAgentId", device=port_id,
|
|
host='host-ovs-no_filter')
|
|
|
|
# Get port and verify status is still DOWN.
|
|
port = self._show('ports', port_id)
|
|
self.assertEqual('DOWN', port['port']['status'])
|