Iterate over same port_id if more than one exists

In certain cases where multiple ports can have the same
external_ids:iface_id property, the ovs agent will arbitrarily choose
one and ignore the rest. In case the chosen port isn't on the
integration bridge the ovs agent is managing, an error is returned to
the calling function. This is faulty since one of the other ports may
belong to the correct bridge and it should be chosen instead.

This is interesting for future L3 HA integration tests, where 2
different instances of l3 agents are needed to run on the same machine.
In this case, both agents will register ports which have the same
iface_id property, but obviously only one of the ports is relevant for
each agent.

Closes-bug: #1370914
Related-bug: #1374947
Change-Id: I05d70417e0f78ae51a9ce377fc93a3f12046b313
This commit is contained in:
John Schwarz 2014-09-18 11:24:53 +03:00
parent 7249b49f6b
commit 72b430b856
2 changed files with 59 additions and 32 deletions

View File

@ -22,6 +22,7 @@ from neutron.agent.linux import ip_lib
from neutron.agent.linux import utils from neutron.agent.linux import utils
from neutron.common import exceptions from neutron.common import exceptions
from neutron.openstack.common import excutils from neutron.openstack.common import excutils
from neutron.openstack.common.gettextutils import _LI, _LW
from neutron.openstack.common import jsonutils from neutron.openstack.common import jsonutils
from neutron.openstack.common import log as logging from neutron.openstack.common import log as logging
from neutron.plugins.common import constants from neutron.plugins.common import constants
@ -401,29 +402,28 @@ class OVSBridge(BaseOVS):
# an exeception which will be captured in this block. # an exeception which will be captured in this block.
# We won't deal with the possibility of ovs-vsctl return multiple # We won't deal with the possibility of ovs-vsctl return multiple
# rows since the interface identifier is unique # rows since the interface identifier is unique
data = json_result['data'][0] for data in json_result['data']:
port_name = data[name_idx] port_name = data[name_idx]
switch = get_bridge_for_iface(self.root_helper, port_name) switch = get_bridge_for_iface(self.root_helper, port_name)
if switch != self.br_name: if switch != self.br_name:
LOG.info(_("Port: %(port_name)s is on %(switch)s," continue
" not on %(br_name)s"), {'port_name': port_name, ofport = data[ofport_idx]
'switch': switch, # ofport must be integer otherwise return None
'br_name': self.br_name}) if not isinstance(ofport, int) or ofport == -1:
return LOG.warn(_LW("ofport: %(ofport)s for VIF: %(vif)s is not a"
ofport = data[ofport_idx] " positive integer"), {'ofport': ofport,
# ofport must be integer otherwise return None 'vif': port_id})
if not isinstance(ofport, int) or ofport == -1: return
LOG.warn(_("ofport: %(ofport)s for VIF: %(vif)s is not a " # Find VIF's mac address in external ids
"positive integer"), {'ofport': ofport, ext_id_dict = dict((item[0], item[1]) for item in
'vif': port_id}) data[ext_ids_idx][1])
return vif_mac = ext_id_dict['attached-mac']
# Find VIF's mac address in external ids return VifPort(port_name, ofport, port_id, vif_mac, self)
ext_id_dict = dict((item[0], item[1]) for item in LOG.info(_LI("Port %(port_id)s not present in bridge %(br_name)s"),
data[ext_ids_idx][1]) {'port_id': port_id, 'br_name': self.br_name})
vif_mac = ext_id_dict['attached-mac'] except Exception as error:
return VifPort(port_name, ofport, port_id, vif_mac, self) LOG.warn(_LW("Unable to parse interface details. Exception: %s"),
except Exception as e: error)
LOG.warn(_("Unable to parse interface details. Exception: %s"), e)
return return
def delete_ports(self, all_ports=False): def delete_ports(self, all_ports=False):

View File

@ -822,8 +822,10 @@ class OVS_Lib_Test(base.BaseTestCase):
with testtools.ExpectedException(Exception): with testtools.ExpectedException(Exception):
self.br.get_local_port_mac() self.br.get_local_port_mac()
def _test_get_vif_port_by_id(self, iface_id, data, br_name=None): def _test_get_vif_port_by_id(self, iface_id, data, br_name=None,
extra_calls_and_values=None):
headings = ['external_ids', 'name', 'ofport'] headings = ['external_ids', 'name', 'ofport']
# Each element is a tuple of (expected mock call, return_value) # Each element is a tuple of (expected mock call, return_value)
expected_calls_and_values = [ expected_calls_and_values = [
(mock.call(["ovs-vsctl", self.TO, "--format=json", (mock.call(["ovs-vsctl", self.TO, "--format=json",
@ -836,9 +838,15 @@ class OVS_Lib_Test(base.BaseTestCase):
if not br_name: if not br_name:
br_name = self.BR_NAME br_name = self.BR_NAME
# Only the last information list in 'data' is used, so if more
# than one vif is described in data, the rest must be declared
# in the argument 'expected_calls_and_values'.
if extra_calls_and_values:
expected_calls_and_values.extend(extra_calls_and_values)
expected_calls_and_values.append( expected_calls_and_values.append(
(mock.call(["ovs-vsctl", self.TO, (mock.call(["ovs-vsctl", self.TO,
"iface-to-br", data[0][headings.index('name')]], "iface-to-br", data[-1][headings.index('name')]],
root_helper=self.root_helper), root_helper=self.root_helper),
br_name)) br_name))
tools.setup_mock_calls(self.execute, expected_calls_and_values) tools.setup_mock_calls(self.execute, expected_calls_and_values)
@ -847,6 +855,15 @@ class OVS_Lib_Test(base.BaseTestCase):
tools.verify_mock_calls(self.execute, expected_calls_and_values) tools.verify_mock_calls(self.execute, expected_calls_and_values)
return vif_port return vif_port
def _assert_vif_port(self, vif_port, ofport=None, mac=None):
if not ofport or ofport == -1 or not mac:
self.assertIsNone(vif_port)
return
self.assertEqual('tap99id', vif_port.vif_id)
self.assertEqual(mac, vif_port.vif_mac)
self.assertEqual('tap99', vif_port.port_name)
self.assertEqual(ofport, vif_port.ofport)
def _test_get_vif_port_by_id_with_data(self, ofport=None, mac=None): def _test_get_vif_port_by_id_with_data(self, ofport=None, mac=None):
external_ids = [["iface-id", "tap99id"], external_ids = [["iface-id", "tap99id"],
["iface-status", "active"]] ["iface-status", "active"]]
@ -855,13 +872,7 @@ class OVS_Lib_Test(base.BaseTestCase):
data = [[["map", external_ids], "tap99", data = [[["map", external_ids], "tap99",
ofport if ofport else '["set",[]]']] ofport if ofport else '["set",[]]']]
vif_port = self._test_get_vif_port_by_id('tap99id', data) vif_port = self._test_get_vif_port_by_id('tap99id', data)
if not ofport or ofport == -1 or not mac: self._assert_vif_port(vif_port, ofport, mac)
self.assertIsNone(vif_port)
return
self.assertEqual(vif_port.vif_id, 'tap99id')
self.assertEqual(vif_port.vif_mac, 'aa:bb:cc:dd:ee:ff')
self.assertEqual(vif_port.port_name, 'tap99')
self.assertEqual(vif_port.ofport, ofport)
def test_get_vif_by_port_id_with_ofport(self): def test_get_vif_by_port_id_with_ofport(self):
self._test_get_vif_port_by_id_with_data( self._test_get_vif_port_by_id_with_data(
@ -887,6 +898,22 @@ class OVS_Lib_Test(base.BaseTestCase):
self.assertIsNone(self._test_get_vif_port_by_id('tap99id', data, self.assertIsNone(self._test_get_vif_port_by_id('tap99id', data,
"br-ext")) "br-ext"))
def test_get_vif_by_port_id_multiple_vifs(self):
external_ids = [["iface-id", "tap99id"],
["iface-status", "active"],
["attached-mac", "de:ad:be:ef:13:37"]]
data = [[["map", external_ids], "dummytap", 1],
[["map", external_ids], "tap99", 1337]]
extra_calls_and_values = [
(mock.call(["ovs-vsctl", self.TO,
"iface-to-br", "dummytap"],
root_helper=self.root_helper),
"br-ext")]
vif_port = self._test_get_vif_port_by_id(
'tap99id', data, extra_calls_and_values=extra_calls_and_values)
self._assert_vif_port(vif_port, ofport=1337, mac="de:ad:be:ef:13:37")
class TestDeferredOVSBridge(base.BaseTestCase): class TestDeferredOVSBridge(base.BaseTestCase):