[OVN] Fix updating network segmentation ID

The ML2/OVN driver wasn't handling updates to the segmentation ID for a
given network. This patch fixes this problem.

This patch extends the _update_segmentation_id() method to check on
drivers which does not inherits from AgentMechanismDriverBase, which
is the case of OVN (which inherits from MechanismDriver). A new method
is now called for those drivers to get a list of supported VIF types,
called get_supported_vif_types().

Closes-Bug: #1944708
Change-Id: Ibe08bfbc2efc55b9d628cdd0605941b7486186b6
Signed-off-by: Lucas Alvares Gomes <lucasagomes@gmail.com>
This commit is contained in:
Lucas Alvares Gomes 2021-09-23 16:04:00 +01:00
parent 492b68493f
commit 982c22dd46
6 changed files with 128 additions and 19 deletions

View File

@ -188,6 +188,16 @@ class OVNMechanismDriver(api.MechanismDriver):
def sb_ovn(self, val): def sb_ovn(self, val):
self._sb_ovn = val self._sb_ovn = val
def get_supported_vif_types(self):
vif_types = set()
for ch in self.sb_ovn.chassis_list().execute(check_error=True):
dp_type = ch.external_ids.get('datapath-type', '')
if dp_type == ovn_const.CHASSIS_DATAPATH_NETDEV:
vif_types.add(portbindings.VIF_TYPE_VHOST_USER)
else:
vif_types.add(portbindings.VIF_TYPE_OVS)
return list(vif_types)
def check_vlan_transparency(self, context): def check_vlan_transparency(self, context):
"""OVN driver vlan transparency support.""" """OVN driver vlan transparency support."""
vlan_transparency_network_types = [ vlan_transparency_network_types = [
@ -222,6 +232,10 @@ class OVNMechanismDriver(api.MechanismDriver):
def supported_extensions(self, extensions): def supported_extensions(self, extensions):
return set(ovn_extensions.ML2_SUPPORTED_API_EXTENSIONS) & extensions return set(ovn_extensions.ML2_SUPPORTED_API_EXTENSIONS) & extensions
@staticmethod
def provider_network_attribute_updates_supported():
return [provider_net.SEGMENTATION_ID]
def subscribe(self): def subscribe(self):
registry.subscribe(self.pre_fork_initialize, registry.subscribe(self.pre_fork_initialize,
resources.PROCESS, resources.PROCESS,

View File

@ -1965,6 +1965,15 @@ class OVNClient(object):
if ovn_ls_azs != neutron_net_azs: if ovn_ls_azs != neutron_net_azs:
self.sync_ha_chassis_group(context, network['id'], txn) self.sync_ha_chassis_group(context, network['id'], txn)
# Update the segment tags, if any
segments = segments_db.get_network_segments(context, network['id'])
for segment in segments:
tag = segment.get(segment_def.SEGMENTATION_ID)
tag = [] if tag is None else tag
lport_name = utils.ovn_provnet_port_name(segment['id'])
txn.add(self._nb_idl.set_lswitch_port(lport_name=lport_name,
tag=tag, if_exists=True))
self._qos_driver.update_network(txn, network, original_network) self._qos_driver.update_network(txn, network, original_network)
if check_rev_cmd.result == ovn_const.TXN_COMMITTED: if check_rev_cmd.result == ovn_const.TXN_COMMITTED:

View File

@ -949,16 +949,26 @@ class Ml2Plugin(db_base_plugin_v2.NeutronDbPluginV2,
vif_types = [portbindings.VIF_TYPE_UNBOUND, vif_types = [portbindings.VIF_TYPE_UNBOUND,
portbindings.VIF_TYPE_BINDING_FAILED] portbindings.VIF_TYPE_BINDING_FAILED]
for mech_driver in self.mechanism_manager.ordered_mech_drivers: for mech_driver in self.mechanism_manager.ordered_mech_drivers:
if (isinstance(mech_driver.obj, if (provider_net.SEGMENTATION_ID in mech_driver.obj.
mech_agent.AgentMechanismDriverBase) and
provider_net.SEGMENTATION_ID in mech_driver.obj.
provider_network_attribute_updates_supported()): provider_network_attribute_updates_supported()):
agent_type = mech_driver.obj.agent_type if isinstance(mech_driver.obj,
agents = self.get_agents( mech_agent.AgentMechanismDriverBase):
context, filters={'agent_type': [agent_type]}) agent_type = mech_driver.obj.agent_type
for agent in agents: agents = self.get_agents(
vif_types.append( context, filters={'agent_type': [agent_type]})
mech_driver.obj.get_supported_vif_type(agent)) for agent in agents:
vif_types.append(
mech_driver.obj.get_supported_vif_type(agent))
else:
# For non-AgentMechanismDriverBase drivers such as OVN
# which inherits from MechanismDriver
# TODO(lucasagomes): Perhaps we should
# include get_supported_vif_types() as part of the
# MechanismDriver's api in neutron-lib and remove this
# check in the future ?
if getattr(mech_driver.obj, 'get_supported_vif_types'):
vif_types.extend(
mech_driver.obj.get_supported_vif_types())
if ports_obj.Port.check_network_ports_by_binding_types( if ports_obj.Port.check_network_ports_by_binding_types(
context, network['id'], vif_types, negative_search=True): context, network['id'], vif_types, negative_search=True):

View File

@ -2644,6 +2644,40 @@ class TestOVNMechanismDriver(TestOVNMechanismDriverBase):
self.assertFalse( self.assertFalse(
self.mech_driver.responsible_for_ports_allocation(context)) self.mech_driver.responsible_for_ports_allocation(context))
def test_update_network_segmentation_id(self):
new_vlan_tag = 123
net_arg = {pnet.NETWORK_TYPE: 'vlan',
pnet.PHYSICAL_NETWORK: 'physnet1',
pnet.SEGMENTATION_ID: '1'}
net = self._make_network(self.fmt, 'net1', True,
arg_list=(pnet.NETWORK_TYPE,
pnet.PHYSICAL_NETWORK,
pnet.SEGMENTATION_ID,),
**net_arg)['network']
# Make sure the network was created with 1 VLAN segment
segments = segments_db.get_network_segments(self.context, net['id'])
segment = segments[0]
self.assertEqual(len(segments), 1)
self.assertEqual(segment['segmentation_id'], 1)
# Issue an update to the network changing the segmentation_id
data = {'network': {pnet.SEGMENTATION_ID: new_vlan_tag}}
req = self.new_update_request('networks', data, net['id'])
res = self.deserialize(self.fmt, req.get_response(self.api))
self.assertEqual(new_vlan_tag, res['network'][pnet.SEGMENTATION_ID])
# Assert the tag was changed in the Neutron database
segments = segments_db.get_network_segments(self.context, net['id'])
segment = segments[0]
self.assertEqual(len(segments), 1)
self.assertEqual(segment['segmentation_id'], new_vlan_tag)
# Assert the tag was changed in the OVN database
expected_call = mock.call(
lport_name=ovn_utils.ovn_provnet_port_name(segment['id']),
tag=new_vlan_tag, if_exists=True)
self.nb_ovn.set_lswitch_port.assert_has_calls([expected_call])
class OVNMechanismDriverTestCase(MechDriverSetupBase, class OVNMechanismDriverTestCase(MechDriverSetupBase,
test_plugin.Ml2PluginV2TestCase): test_plugin.Ml2PluginV2TestCase):
@ -2989,6 +3023,10 @@ class TestOVNMechanismDriverSegment(MechDriverSetupBase,
segment_id=self.seg_2['id']) as subnet: segment_id=self.seg_2['id']) as subnet:
self.sub_2 = subnet self.sub_2 = subnet
# TODO(lucasagomes): This test should use <mock>.assert_has_calls()
# to validate if the method was called with the correct values instead
# of peeking at call_args_list otherwise every time there's a new
# call to the mocked method the indexes will change
def test_create_segments_subnet_metadata_ip_allocation(self): def test_create_segments_subnet_metadata_ip_allocation(self):
self._test_segments_helper() self._test_segments_helper()
ovn_nb_api = self.mech_driver.nb_ovn ovn_nb_api = self.mech_driver.nb_ovn
@ -2997,34 +3035,34 @@ class TestOVNMechanismDriverSegment(MechDriverSetupBase,
# created subnet. # created subnet.
self.assertIn( self.assertIn(
'10.0.1.2', '10.0.1.2',
ovn_nb_api.set_lswitch_port.call_args_list[0][1]['addresses'][0]) ovn_nb_api.set_lswitch_port.call_args_list[2][1]['addresses'][0])
# Assert that the second subnet address also has been allocated for # Assert that the second subnet address also has been allocated for
# metadata port. # metadata port.
self.assertIn( self.assertIn(
'10.0.2.2', '10.0.2.2',
ovn_nb_api.set_lswitch_port.call_args_list[1][1]['addresses'][0]) ovn_nb_api.set_lswitch_port.call_args_list[6][1]['addresses'][0])
# Assert also that the first subnet address is in this update # Assert also that the first subnet address is in this update
self.assertIn( self.assertIn(
'10.0.1.2', '10.0.1.2',
ovn_nb_api.set_lswitch_port.call_args_list[1][1]['addresses'][0]) ovn_nb_api.set_lswitch_port.call_args_list[6][1]['addresses'][0])
self.assertEqual( self.assertEqual(
ovn_nb_api.set_lswitch_port.call_count, 2) ovn_nb_api.set_lswitch_port.call_count, 7)
# Make sure both updates where on same metadata port # Make sure both updates where on same metadata port
args_list = ovn_nb_api.set_lswitch_port.call_args_list args_list = ovn_nb_api.set_lswitch_port.call_args_list
self.assertEqual( self.assertEqual(
'ovnmeta-%s' % self.net['network']['id'], 'ovnmeta-%s' % self.net['network']['id'],
args_list[1][1]['external_ids']['neutron:device_id']) args_list[6][1]['external_ids']['neutron:device_id'])
self.assertEqual( self.assertEqual(
args_list[1][1]['external_ids']['neutron:device_id'], args_list[6][1]['external_ids']['neutron:device_id'],
args_list[0][1]['external_ids']['neutron:device_id']) args_list[2][1]['external_ids']['neutron:device_id'])
self.assertEqual( self.assertEqual(
args_list[1][1]['external_ids']['neutron:device_owner'], args_list[6][1]['external_ids']['neutron:device_owner'],
args_list[0][1]['external_ids']['neutron:device_owner']) args_list[2][1]['external_ids']['neutron:device_owner'])
self.assertEqual( self.assertEqual(
const.DEVICE_OWNER_DISTRIBUTED, const.DEVICE_OWNER_DISTRIBUTED,
args_list[1][1]['external_ids']['neutron:device_owner']) args_list[6][1]['external_ids']['neutron:device_owner'])
def test_create_segments_mixed_allocation_prohibited(self): def test_create_segments_mixed_allocation_prohibited(self):
self._test_segments_helper() self._test_segments_helper()

View File

@ -533,6 +533,37 @@ class TestMl2NetworksV2(test_plugin.TestNetworksV2,
portbindings.VIF_TYPE_BINDING_FAILED], portbindings.VIF_TYPE_BINDING_FAILED],
negative_search=True) negative_search=True)
@mock.patch.object(ml2_plugin, 'isinstance')
@mock.patch.object(port_obj.Port, 'check_network_ports_by_binding_types')
def test__update_segmentation_id_non_AgentMechanismDriverBase(
self, mock_check_network, mock_isinstance):
plugin = directory.get_plugin()
mock_check_network.return_value = False
mock_isinstance.return_value = False
mock_update_net_segment = mock.patch.object(
plugin.type_manager, 'update_network_segment').start()
fake_vif_types = ['fake-vif-type0', 'fake-vif-type1']
fake_driver = mock.Mock()
fake_driver.obj.get_supported_vif_types.return_value = (
fake_vif_types)
fake_driver.obj.provider_network_attribute_updates_supported.\
return_value = {pnet.SEGMENTATION_ID: 1000}
plugin.mechanism_manager.ordered_mech_drivers = [fake_driver]
with self.network() as net:
net_data = {pnet.SEGMENTATION_ID: 1000}
plugin._update_segmentation_id(self.context, net['network'],
net_data)
# Assert the new method get_supported_vif_types() has been called
fake_driver.obj.get_supported_vif_types.assert_called_once_with()
# Assert the update method has been called
mock_update_net_segment.assert_called_once_with(
self.context, net['network'], net_data, mock.ANY)
def test_update_network_with_empty_body(self): def test_update_network_with_empty_body(self):
with self.network() as network: with self.network() as network:
network_id = network["network"]["id"] network_id = network["network"]["id"]

View File

@ -0,0 +1,7 @@
---
fixes:
- |
Fixes an issue in the ML2/OVN driver where the network
segment tag was not being updated in the OVN Northbound
database. For more information, see bug `1944708
<https://bugs.launchpad.net/neutron/+bug/1944708>`_.