neutron/neutron/tests/functional/agent/linux/test_bridge_lib.py

281 lines
12 KiB
Python

# Copyright (c) 2015 Thales Services SAS
#
# 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 random
import netaddr
from neutron_lib import constants
from oslo_utils import uuidutils
import testscenarios
from neutron.agent.linux import bridge_lib
from neutron.agent.linux import ip_lib
from neutron.privileged.agent.linux import ip_lib as priv_ip_lib
from neutron.tests.common import net_helpers
from neutron.tests.functional import base
MAC_ALL_NODES_ADDRESS = '33:33:00:00:00:01'
class BridgeLibTestCase(base.BaseSudoTestCase):
def setUp(self):
super(BridgeLibTestCase, self).setUp()
self.bridge, self.port_fixture = self.create_bridge_port_fixture()
def create_bridge_port_fixture(self):
bridge = self.useFixture(
net_helpers.LinuxBridgeFixture(namespace=None)).bridge
port_fixture = self.useFixture(
net_helpers.LinuxBridgePortFixture(
bridge, port_id=uuidutils.generate_uuid()))
return bridge, port_fixture
def test_is_bridged_interface_and_remove(self):
bridge = bridge_lib.BridgeDevice(self.bridge.name)
bridge_port = self.port_fixture.br_port.name
self.assertTrue(bridge_lib.is_bridged_interface(bridge_port))
bridge.delif(bridge_port)
self.assertFalse(bridge_lib.is_bridged_interface(bridge_port))
def test_is_not_bridged_interface(self):
self.assertFalse(
bridge_lib.is_bridged_interface(self.port_fixture.port.name))
def test_delete_bridge(self):
bridge = bridge_lib.BridgeDevice(self.bridge.name)
self.assertTrue(bridge.exists())
bridge.delbr()
self.assertFalse(bridge.exists())
def test_get_bridge_names(self):
self.assertIn(self.bridge.name, bridge_lib.get_bridge_names())
def test_get_interface_ifindex(self):
port = self.port_fixture.br_port
t1 = bridge_lib.get_interface_ifindex(str(port))
self.port_fixture.veth_fixture.destroy()
self.port_fixture.veth_fixture._setUp()
t2 = bridge_lib.get_interface_ifindex(str(port))
self.assertIsNotNone(t1)
self.assertIsNotNone(t2)
self.assertGreaterEqual(t2, t1)
def test_get_interface_bridge(self):
bridge = bridge_lib.BridgeDevice.get_interface_bridge(
self.port_fixture.br_port.name)
self.assertEqual(self.bridge.name, bridge.name)
def test_get_interface_no_bridge(self):
bridge = bridge_lib.BridgeDevice.get_interface_bridge(
self.port_fixture.port.name)
self.assertIsNone(bridge)
def test_get_interfaces(self):
self.assertEqual(
[self.port_fixture.br_port.name], self.bridge.get_interfaces())
def test_get_interfaces_no_bridge(self):
bridge = bridge_lib.BridgeDevice('--fake--')
self.assertEqual([], bridge.get_interfaces())
def test_disable_ipv6(self):
sysfs_path = ("/proc/sys/net/ipv6/conf/%s/disable_ipv6" %
self.bridge.name)
# first, make sure it's enabled
with open(sysfs_path, 'r') as sysfs_disable_ipv6_file:
sysfs_disable_ipv6 = sysfs_disable_ipv6_file.read()
self.assertEqual("0\n", sysfs_disable_ipv6)
self.assertEqual(0, self.bridge.disable_ipv6())
with open(sysfs_path, 'r') as sysfs_disable_ipv6_file:
sysfs_disable_ipv6 = sysfs_disable_ipv6_file.read()
self.assertEqual("1\n", sysfs_disable_ipv6)
def _get_bridge_info(self):
for device in (device for device in ip_lib.get_devices_info(
self.bridge.namespace) if device['name'] == self.bridge.name):
return device
self.fail('Bridge %s not present' % self.bridge.name)
def test_set_forward_delay(self):
bridge = bridge_lib.BridgeDevice(self.bridge.name)
for fd in (10, 200, 3000, 40000):
bridge.setfd(fd)
br_info = self._get_bridge_info()
self.assertEqual(fd, br_info['forward_delay'])
def test_enable_and_disable_stp(self):
bridge = bridge_lib.BridgeDevice(self.bridge.name)
bridge.disable_stp()
br_info = self._get_bridge_info()
self.assertEqual(0, br_info['stp'])
bridge.enable_stp()
br_info = self._get_bridge_info()
self.assertEqual(1, br_info['stp'])
class FdbInterfaceTestCase(testscenarios.WithScenarios, base.BaseSudoTestCase):
MAC1 = 'ca:fe:ca:fe:ca:fe'
MAC2 = 'ca:fe:ca:fe:ca:01'
scenarios = [
('namespace', {'namespace': 'ns_' + uuidutils.generate_uuid()}),
('no_namespace', {'namespace': None})
]
def setUp(self):
super(FdbInterfaceTestCase, self).setUp()
_uuid = uuidutils.generate_uuid()
self.device = ('int_' + _uuid)[:constants.DEVICE_NAME_MAX_LEN]
self.device_vxlan = ('vxlan_' + _uuid)[:constants.DEVICE_NAME_MAX_LEN]
self.ip = '10.220.0.1/24'
self.ip_vxlan = '10.221.0.1/24'
if self.namespace:
priv_ip_lib.create_netns(self.namespace)
else:
self._cleanup()
self.addCleanup(self._cleanup)
vni = random.randint(1, 2**24 - 1)
ip_wrapper = ip_lib.IPWrapper(self.namespace)
ip_wrapper.add_dummy(self.device)
ip_wrapper.add_vxlan(self.device_vxlan, vni, dev=self.device)
ip_device = ip_lib.IPDevice(self.device, self.namespace)
ip_device.link.set_up()
ip_device.addr.add(self.ip)
ip_device_vxlan = ip_lib.IPDevice(self.device_vxlan, self.namespace)
ip_device_vxlan.link.set_up()
ip_device_vxlan.addr.add(self.ip_vxlan)
def _cleanup(self):
if self.namespace:
priv_ip_lib.remove_netns(self.namespace)
else:
for device in (self.device_vxlan, self.device):
try:
priv_ip_lib.delete_interface(device, None)
except priv_ip_lib.NetworkInterfaceNotFound:
pass
def _assert_mac(self, mac_address, device, present=True):
msg = ('MAC address %(mac_address)s %(present)s in the FDB table for '
'the device %(device)s in namespace %(namespace)s' %
{'mac_address': mac_address, 'device': device,
'namespace': self.namespace,
'present': 'not present' if present else 'present'})
for _device, fdbs in bridge_lib.FdbInterface.show(
dev=device, namespace=self.namespace).items():
self.assertEqual(device, _device)
macs = [fdb['mac'] for fdb in fdbs]
if ((mac_address in macs and not present) or
(mac_address not in macs and present)):
self.fail(msg)
def _assert_ip(self, mac_address, ip_address, device):
msg = ('Destination IP address %(ip_address)s not present in the FDB '
'table for the MAC address %(mac_address)s and device '
'%(device)s in namespace %(namespace)s' %
{'mac_address': mac_address, 'device': device,
'namespace': self.namespace, 'ip_address': ip_address})
for _device, fdbs in bridge_lib.FdbInterface.show(
dev=device, namespace=self.namespace).items():
self.assertEqual(device, _device)
for _ in (fdb for fdb in fdbs if fdb['mac'] == mac_address and
fdb['dst_ip'] == ip_address):
return
self.fail(msg)
def test_add_delete(self):
self._assert_mac(self.MAC1, self.device, present=False)
bridge_lib.FdbInterface.add(self.MAC1, self.device,
namespace=self.namespace)
self._assert_mac(self.MAC1, self.device)
bridge_lib.FdbInterface.delete(self.MAC1, self.device,
namespace=self.namespace)
self._assert_mac(self.MAC1, self.device, present=False)
def test_add_delete_dst(self):
self._assert_mac(self.MAC1, self.device_vxlan, present=False)
bridge_lib.FdbInterface.add(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip=str(netaddr.IPNetwork(self.ip).ip))
self._assert_ip(self.MAC1, str(netaddr.IPNetwork(self.ip).ip),
self.device_vxlan)
bridge_lib.FdbInterface.delete(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip=str(netaddr.IPNetwork(self.ip).ip))
self._assert_mac(self.MAC1, self.device_vxlan, present=False)
def test_append(self):
self._assert_mac(self.MAC1, self.device, present=False)
bridge_lib.FdbInterface.append(self.MAC1, self.device,
namespace=self.namespace)
self._assert_mac(self.MAC1, self.device)
def test_append_dst(self):
self._assert_mac(self.MAC1, self.device_vxlan)
bridge_lib.FdbInterface.append(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip=str(netaddr.IPNetwork(self.ip).ip))
self._assert_ip(self.MAC1, str(netaddr.IPNetwork(self.ip).ip),
self.device_vxlan)
def test_replace(self):
self._assert_mac(self.MAC1, self.device, present=False)
bridge_lib.FdbInterface.add(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip=str(netaddr.IPNetwork(self.ip).ip))
self._assert_ip(self.MAC1, str(netaddr.IPNetwork(self.ip).ip),
self.device_vxlan)
bridge_lib.FdbInterface.replace(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip='1.1.1.1')
self._assert_ip(self.MAC1, '1.1.1.1', self.device_vxlan)
def test_show(self):
ip_str = str(netaddr.IPNetwork(self.ip).ip)
bridge_lib.FdbInterface.add(
self.MAC1, self.device_vxlan, namespace=self.namespace,
dst_ip=ip_str)
rules = bridge_lib.FdbInterface.show(dev=self.device_vxlan,
namespace=self.namespace)
self.assertEqual(1, len(rules))
self.assertEqual(1, len(rules[self.device_vxlan]))
self.assertEqual(self.MAC1, rules[self.device_vxlan][0]['mac'])
self.assertEqual(ip_str, rules[self.device_vxlan][0]['dst_ip'])
_uuid = uuidutils.generate_uuid()
bridge_name = ('br_' + _uuid)[:constants.DEVICE_NAME_MAX_LEN]
priv_ip_lib.create_interface(bridge_name, self.namespace, 'bridge')
bridge = bridge_lib.BridgeDevice(bridge_name, namespace=self.namespace)
bridge.addif(self.device)
rules = bridge_lib.FdbInterface.show(dev=bridge_name,
namespace=self.namespace)
self.assertEqual(1, len(rules))
self._assert_mac(MAC_ALL_NODES_ADDRESS, bridge_name)
rules = bridge_lib.FdbInterface.show(dev=self.device,
namespace=self.namespace)
mac_address = ip_lib.IPDevice(self.device, self.namespace).link.address
for rule in (rule for rule in rules[self.device] if
rule['mac'] == mac_address):
self.assertEqual(bridge_name, rule['master'])
self.assertIn(rule['vlan'], (1, None))