Fix placement of lsp when external_ids not in sync
When options.requested-chassis is not in sync with external_ids.neutron:host_id it would pick both hosts, causing duplicate announcements from more than 1 host. This has been fixed in change 910305, but was left unchanged for the sync method, causing issues when the sync interval was re-evaluating all lsp's on the node. The code for determining the chassis of a port has been moved from the base_watcher to driver_utils so the logic for the event is the same as the logic when fetching the records from the northbound database. Related-bug: #2049902 Change-Id: I545d6b41fd308eb56e5295657260718dc14868f7
This commit is contained in:
parent
8f3351072c
commit
1bacff1dff
@ -40,6 +40,40 @@ def get_addr_scopes(port):
|
||||
}
|
||||
|
||||
|
||||
def get_port_chassis(port, chassis,
|
||||
default_port_type=constants.OVN_VM_VIF_PORT_TYPE):
|
||||
# row.options['requested-chassis'] superseeds the id in external_ids.
|
||||
# Since it is not used for virtual ports by ovn, this option will be
|
||||
# ignored for virtual ports.
|
||||
|
||||
# since 'old' rows could be used, it will not hold the type information
|
||||
# if that is the case, please supply a default in the arguments.
|
||||
|
||||
port_type = getattr(port, 'type', default_port_type)
|
||||
if (port_type not in [constants.OVN_VIRTUAL_VIF_PORT_TYPE] and
|
||||
hasattr(port, 'options') and
|
||||
port.options.get(constants.OVN_REQUESTED_CHASSIS)):
|
||||
|
||||
# requested-chassis can be a comma separated list,
|
||||
# so lets only return our chassis if it is a list, to be able
|
||||
# to do a == equal comparison
|
||||
req_chassis = port.options[constants.OVN_REQUESTED_CHASSIS]
|
||||
if chassis in req_chassis.split(','):
|
||||
req_chassis = chassis
|
||||
|
||||
return req_chassis.split(',')[0], constants.OVN_CHASSIS_AT_OPTIONS
|
||||
|
||||
elif (hasattr(port, 'external_ids') and
|
||||
port.external_ids.get(constants.OVN_HOST_ID_EXT_ID_KEY)):
|
||||
|
||||
return (
|
||||
port.external_ids[constants.OVN_HOST_ID_EXT_ID_KEY],
|
||||
constants.OVN_CHASSIS_AT_EXT_IDS
|
||||
)
|
||||
|
||||
return None, None
|
||||
|
||||
|
||||
def check_name_prefix(entity, prefix):
|
||||
try:
|
||||
return entity.name.startswith(prefix)
|
||||
|
@ -28,6 +28,7 @@ from ovsdbapp.schema.ovn_northbound import impl_idl as nb_impl_idl
|
||||
from ovsdbapp.schema.ovn_southbound import impl_idl as sb_impl_idl
|
||||
|
||||
from ovn_bgp_agent import constants
|
||||
from ovn_bgp_agent.drivers.openstack.utils import driver_utils
|
||||
from ovn_bgp_agent import exceptions
|
||||
from ovn_bgp_agent.utils import helpers
|
||||
|
||||
@ -414,14 +415,10 @@ class OvsdbNbOvnIdl(nb_impl_idl.OvnNbApiIdlImpl, Backend):
|
||||
ports = []
|
||||
cmd = self.db_find_rows('Logical_Switch_Port', ('up', '=', True))
|
||||
for row in cmd.execute(check_error=True):
|
||||
if (row.external_ids and
|
||||
row.external_ids.get(
|
||||
constants.OVN_HOST_ID_EXT_ID_KEY) == chassis):
|
||||
ports.append(row)
|
||||
elif (row.options and
|
||||
row.options.get(
|
||||
constants.OVN_REQUESTED_CHASSIS) == chassis):
|
||||
port_chassis, _ = driver_utils.get_port_chassis(row, chassis)
|
||||
if port_chassis == chassis:
|
||||
ports.append(row)
|
||||
|
||||
return ports
|
||||
|
||||
def get_active_cr_lrp_on_chassis(self, chassis):
|
||||
|
@ -97,30 +97,8 @@ class LSPChassisEvent(Event):
|
||||
return len(mac.strip().split(' ')) > 1
|
||||
|
||||
def _get_chassis(self, row, default_type=constants.OVN_VM_VIF_PORT_TYPE):
|
||||
# row.options['requested-chassis'] superseeds the id in external_ids.
|
||||
# Since it is not used for virtual ports by ovn, this option will be
|
||||
# ignored for virtual ports.
|
||||
|
||||
# since 'old' rows could be used, it will not hold the type information
|
||||
# if that is the case, please supply a default in the arguments.
|
||||
row_type = getattr(row, 'type', default_type)
|
||||
if (row_type not in [constants.OVN_VIRTUAL_VIF_PORT_TYPE] and
|
||||
hasattr(row, 'options') and
|
||||
row.options.get(constants.OVN_REQUESTED_CHASSIS)):
|
||||
|
||||
# requested-chassis can be a comma separated list,
|
||||
# so lets only return our chassis if it is a list, to be able to
|
||||
# do a == equal comparison
|
||||
req_chassis = row.options[constants.OVN_REQUESTED_CHASSIS]
|
||||
if self.agent.chassis in req_chassis.split(','):
|
||||
req_chassis = self.agent.chassis
|
||||
|
||||
return (req_chassis, constants.OVN_CHASSIS_AT_OPTIONS)
|
||||
elif (hasattr(row, 'external_ids') and
|
||||
row.external_ids.get(constants.OVN_HOST_ID_EXT_ID_KEY)):
|
||||
return (row.external_ids[constants.OVN_HOST_ID_EXT_ID_KEY],
|
||||
constants.OVN_CHASSIS_AT_EXT_IDS)
|
||||
return None, None
|
||||
return driver_utils.get_port_chassis(row, self.agent.chassis,
|
||||
default_port_type=default_type)
|
||||
|
||||
def _has_additional_binding(self, row):
|
||||
if (hasattr(row, 'options') and
|
||||
|
@ -68,6 +68,44 @@ class TestDriverUtils(test_base.TestCase):
|
||||
constants.IP_VERSION_6: None,
|
||||
})
|
||||
|
||||
def test_get_port_chassis_from_options(self):
|
||||
my_host = 'foo-host'
|
||||
|
||||
# it is a VM port type, should use options field.
|
||||
row = utils.create_row(
|
||||
external_ids={constants.OVN_HOST_ID_EXT_ID_KEY: 'bar-host'},
|
||||
options={constants.OVN_REQUESTED_CHASSIS: my_host})
|
||||
|
||||
self.assertEqual(driver_utils.get_port_chassis(row, chassis=my_host),
|
||||
(my_host, constants.OVN_CHASSIS_AT_OPTIONS))
|
||||
|
||||
def test_get_port_chassis_from_external_ids(self):
|
||||
my_host = 'foo-host'
|
||||
|
||||
# it is a VM port type, should use options field.
|
||||
row = utils.create_row(
|
||||
external_ids={constants.OVN_HOST_ID_EXT_ID_KEY: my_host})
|
||||
|
||||
self.assertEqual(driver_utils.get_port_chassis(row, chassis=my_host),
|
||||
(my_host, constants.OVN_CHASSIS_AT_EXT_IDS))
|
||||
|
||||
def test_get_port_chassis_from_external_ids_virtual_port(self):
|
||||
my_host = 'foo-host'
|
||||
|
||||
# it is a VM port type, should use options field.
|
||||
row = utils.create_row(
|
||||
external_ids={constants.OVN_HOST_ID_EXT_ID_KEY: my_host},
|
||||
options={constants.OVN_REQUESTED_CHASSIS: 'bar-host'},
|
||||
type=constants.OVN_VIRTUAL_VIF_PORT_TYPE)
|
||||
|
||||
self.assertEqual(driver_utils.get_port_chassis(row, chassis=my_host),
|
||||
(my_host, constants.OVN_CHASSIS_AT_EXT_IDS))
|
||||
|
||||
def test_get_port_chassis_no_information(self):
|
||||
row = utils.create_row()
|
||||
self.assertEqual(driver_utils.get_port_chassis(row, chassis='foo'),
|
||||
(None, None))
|
||||
|
||||
def test_check_name_prefix(self):
|
||||
lb = utils.create_row(name='some-name')
|
||||
self.assertTrue(driver_utils.check_name_prefix(lb, 'some'))
|
||||
|
Loading…
Reference in New Issue
Block a user