[OVN] DB sync host/physnet - filter on agent_type
When syncing hostname and physical networks, filter neutron hosts on agent_type. Only segmenthostmappings for hosts with agent 'OVN Controller agent' should be cleaned up. Since change: I935186b6ee95f0cae8dc05869d9742c8fb3353c3 there is de-duplication of segmenthostmapping updates from agents. If the OVN DB sync clears/deletes mappings for hosts owned by other agents/plugins the mappings are never re-created. Closes-Bug: #2040172 Change-Id: Iaf15e560e1b1ec31618b2ebc6206a938463c1094 Signed-off-by: Harald Jensås <hjensas@redhat.com>
This commit is contained in:
parent
d2b5ad4ac5
commit
3aafeefc85
@ -1390,7 +1390,8 @@ class OvnSbSynchronizer(OvnDbSynchronizer):
|
|||||||
str(datetime.now()))
|
str(datetime.now()))
|
||||||
host_phynets_map = self.ovn_api.get_chassis_hostname_and_physnets()
|
host_phynets_map = self.ovn_api.get_chassis_hostname_and_physnets()
|
||||||
current_hosts = set(host_phynets_map)
|
current_hosts = set(host_phynets_map)
|
||||||
previous_hosts = segments_db.get_hosts_mapped_with_segments(ctx)
|
previous_hosts = segments_db.get_hosts_mapped_with_segments(
|
||||||
|
ctx, include_agent_types={ovn_const.OVN_CONTROLLER_AGENT})
|
||||||
|
|
||||||
stale_hosts = previous_hosts - current_hosts
|
stale_hosts = previous_hosts - current_hosts
|
||||||
for host in stale_hosts:
|
for host in stale_hosts:
|
||||||
|
@ -1820,6 +1820,22 @@ class TestOvnSbSync(base.TestOVNFunctionalBase):
|
|||||||
def _sync_resources(self):
|
def _sync_resources(self):
|
||||||
self.sb_synchronizer.sync_hostname_and_physical_networks(self.ctx)
|
self.sb_synchronizer.sync_hostname_and_physical_networks(self.ctx)
|
||||||
|
|
||||||
|
def create_agent(self, host, bridge_mappings=None, agent_type=None):
|
||||||
|
if agent_type is None:
|
||||||
|
agent_type = ovn_const.OVN_CONTROLLER_AGENT
|
||||||
|
if bridge_mappings is None:
|
||||||
|
bridge_mappings = {}
|
||||||
|
agent = {
|
||||||
|
'host': host,
|
||||||
|
'agent_type': agent_type,
|
||||||
|
'binary': '/bin/test',
|
||||||
|
'topic': 'test_topic',
|
||||||
|
'configurations': {'bridge_mappings': bridge_mappings}
|
||||||
|
}
|
||||||
|
_, status = self.plugin.create_or_update_agent(self.context, agent)
|
||||||
|
|
||||||
|
return status['id']
|
||||||
|
|
||||||
def create_segment(self, network_id, physical_network, segmentation_id):
|
def create_segment(self, network_id, physical_network, segmentation_id):
|
||||||
segment_data = {'network_id': network_id,
|
segment_data = {'network_id': network_id,
|
||||||
'physical_network': physical_network,
|
'physical_network': physical_network,
|
||||||
@ -1860,6 +1876,7 @@ class TestOvnSbSync(base.TestOVNFunctionalBase):
|
|||||||
segment = self.create_segment(network_id, 'physnet1', 50)
|
segment = self.create_segment(network_id, 'physnet1', 50)
|
||||||
segments_db.update_segment_host_mapping(
|
segments_db.update_segment_host_mapping(
|
||||||
self.ctx, 'host1', {segment['id']})
|
self.ctx, 'host1', {segment['id']})
|
||||||
|
_ = self.create_agent('host1', bridge_mappings={'physnet1': 'eth0'})
|
||||||
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
self.assertEqual({'host1'}, segment_hosts)
|
self.assertEqual({'host1'}, segment_hosts)
|
||||||
# Since there is no chassis in the sb DB, host1 is the stale host
|
# Since there is no chassis in the sb DB, host1 is the stale host
|
||||||
@ -1868,6 +1885,36 @@ class TestOvnSbSync(base.TestOVNFunctionalBase):
|
|||||||
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
self.assertFalse(segment_hosts)
|
self.assertFalse(segment_hosts)
|
||||||
|
|
||||||
|
def test_ovn_sb_sync_host_with_no_agent_not_deleted(self):
|
||||||
|
with self.network() as network:
|
||||||
|
network_id = network['network']['id']
|
||||||
|
segment = self.create_segment(network_id, 'physnet1', 50)
|
||||||
|
segments_db.update_segment_host_mapping(
|
||||||
|
self.ctx, 'host1', {segment['id']})
|
||||||
|
_ = self.create_agent('host1', bridge_mappings={'physnet1': 'eth0'},
|
||||||
|
agent_type="Not OVN Agent")
|
||||||
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
|
self.assertEqual({'host1'}, segment_hosts)
|
||||||
|
# There is no chassis in the sb DB, host1 does not have an agent
|
||||||
|
# so it is not deleted.
|
||||||
|
self._sync_resources()
|
||||||
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
|
self.assertEqual({'host1'}, segment_hosts)
|
||||||
|
|
||||||
|
def test_ovn_sb_sync_host_with_other_agent_type_not_deleted(self):
|
||||||
|
with self.network() as network:
|
||||||
|
network_id = network['network']['id']
|
||||||
|
segment = self.create_segment(network_id, 'physnet1', 50)
|
||||||
|
segments_db.update_segment_host_mapping(
|
||||||
|
self.ctx, 'host1', {segment['id']})
|
||||||
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
|
self.assertEqual({'host1'}, segment_hosts)
|
||||||
|
# There is no chassis in the sb DB, host1 does not have an agent
|
||||||
|
# so it is not deleted.
|
||||||
|
self._sync_resources()
|
||||||
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
|
self.assertEqual({'host1'}, segment_hosts)
|
||||||
|
|
||||||
def test_ovn_sb_sync(self):
|
def test_ovn_sb_sync(self):
|
||||||
with self.network() as network:
|
with self.network() as network:
|
||||||
network_id = network['network']['id']
|
network_id = network['network']['id']
|
||||||
@ -1880,6 +1927,9 @@ class TestOvnSbSync(base.TestOVNFunctionalBase):
|
|||||||
segments_db.update_segment_host_mapping(
|
segments_db.update_segment_host_mapping(
|
||||||
self.ctx, 'host3', {seg1['id']})
|
self.ctx, 'host3', {seg1['id']})
|
||||||
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
segment_hosts = segments_db.get_hosts_mapped_with_segments(self.ctx)
|
||||||
|
_ = self.create_agent('host1')
|
||||||
|
_ = self.create_agent('host2', bridge_mappings={'physnet2': 'eth0'})
|
||||||
|
_ = self.create_agent('host3', bridge_mappings={'physnet3': 'eth0'})
|
||||||
self.assertEqual({'host1', 'host2', 'host3'}, segment_hosts)
|
self.assertEqual({'host1', 'host2', 'host3'}, segment_hosts)
|
||||||
self.add_fake_chassis('host2', ['physnet2'])
|
self.add_fake_chassis('host2', ['physnet2'])
|
||||||
self.add_fake_chassis('host3', ['physnet3'])
|
self.add_fake_chassis('host3', ['physnet3'])
|
||||||
|
@ -1169,8 +1169,10 @@ class TestOvnSbSyncML2(test_mech_driver.OVNMechanismDriverTestCase):
|
|||||||
|
|
||||||
with mock.patch.object(ovn_db_sync.segments_db,
|
with mock.patch.object(ovn_db_sync.segments_db,
|
||||||
'get_hosts_mapped_with_segments',
|
'get_hosts_mapped_with_segments',
|
||||||
return_value=hosts_in_neutron):
|
return_value=hosts_in_neutron) as mock_ghmws:
|
||||||
ovn_sb_synchronizer.sync_hostname_and_physical_networks(mock.ANY)
|
ovn_sb_synchronizer.sync_hostname_and_physical_networks(mock.ANY)
|
||||||
|
mock_ghmws.assert_called_once_with(
|
||||||
|
mock.ANY, include_agent_types={ovn_const.OVN_CONTROLLER_AGENT})
|
||||||
all_hosts = set(hostname_with_physnets.keys()) | hosts_in_neutron
|
all_hosts = set(hostname_with_physnets.keys()) | hosts_in_neutron
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
len(all_hosts),
|
len(all_hosts),
|
||||||
|
@ -0,0 +1,10 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
When synchronizing the OVN databases, either when running the migration
|
||||||
|
command or during startup, the code responsible for synchronization will
|
||||||
|
only clean up segment-to-host mappings for hosts with agent_type
|
||||||
|
``OVN Controller agent``. Before, the synchronization would clean up
|
||||||
|
(delete) segment-to-host mappings for non-OVN hosts. Fixes bug:
|
||||||
|
`2040172 <https://bugs.launchpad.net/neutron/+bug/2040172>`_.
|
||||||
|
|
Loading…
Reference in New Issue
Block a user