DVR: fix csnat port missing after router update
When updating router gateway to another external network, original csnat ports are deleted in db, but no new ports are created because the _create_snat_interfaces_after_change method did't take the network change into account. To reduce the complexity of operation taken by l3 agent caused by the csnat ports change, we don't need to delete original csnat ports. This patch will skip the csnat port deletion if it is a network change. Change-Id: Id8e89daa339424bb3c5efdffd0a59cfed005bb5d Closes-Bug: #1651813
This commit is contained in:
parent
69259e9525
commit
38c070bde4
@ -395,6 +395,7 @@ class L3_NAT_dbonly_mixin(l3.RouterPluginBase,
|
|||||||
context=context,
|
context=context,
|
||||||
router=router,
|
router=router,
|
||||||
network_id=old_network_id,
|
network_id=old_network_id,
|
||||||
|
new_network_id=new_network_id,
|
||||||
gateway_ips=gw_ips)
|
gateway_ips=gw_ips)
|
||||||
|
|
||||||
def _delete_router_gw_port_db(self, context, router):
|
def _delete_router_gw_port_db(self, context, router):
|
||||||
|
@ -166,7 +166,8 @@ class L3_NAT_with_dvr_db_mixin(l3_db.L3_NAT_db_mixin,
|
|||||||
return router_db
|
return router_db
|
||||||
|
|
||||||
def _delete_dvr_internal_ports(self, event, trigger, resource,
|
def _delete_dvr_internal_ports(self, event, trigger, resource,
|
||||||
context, router, network_id, **kwargs):
|
context, router, network_id,
|
||||||
|
new_network_id, **kwargs):
|
||||||
"""
|
"""
|
||||||
GW port AFTER_DELETE event handler to cleanup DVR ports.
|
GW port AFTER_DELETE event handler to cleanup DVR ports.
|
||||||
|
|
||||||
@ -177,7 +178,9 @@ class L3_NAT_with_dvr_db_mixin(l3_db.L3_NAT_db_mixin,
|
|||||||
|
|
||||||
if not is_distributed_router(router):
|
if not is_distributed_router(router):
|
||||||
return
|
return
|
||||||
self.delete_csnat_router_interface_ports(context.elevated(), router)
|
if not new_network_id:
|
||||||
|
self.delete_csnat_router_interface_ports(context.elevated(),
|
||||||
|
router)
|
||||||
# NOTE(Swami): Delete the Floatingip agent gateway port
|
# NOTE(Swami): Delete the Floatingip agent gateway port
|
||||||
# on all hosts when it is the last gateway port in the
|
# on all hosts when it is the last gateway port in the
|
||||||
# given external network.
|
# given external network.
|
||||||
|
@ -270,8 +270,8 @@ class L3DvrTestCase(test_db_base_plugin_v2.NeutronDbPluginV2TestCase):
|
|||||||
self._helper_delete_floatingip_agent_gateway_port(
|
self._helper_delete_floatingip_agent_gateway_port(
|
||||||
'foo_host')
|
'foo_host')
|
||||||
|
|
||||||
def _setup_delete_current_gw_port_deletes_fip_agent_gw_port(
|
def _setup_delete_current_gw_port_deletes_dvr_internal_ports(
|
||||||
self, port=None, gw_port=True):
|
self, port=None, gw_port=True, new_network_id='ext_net_id_2'):
|
||||||
router = mock.MagicMock()
|
router = mock.MagicMock()
|
||||||
router.extra_attributes.distributed = True
|
router.extra_attributes.distributed = True
|
||||||
if gw_port:
|
if gw_port:
|
||||||
@ -307,17 +307,15 @@ class L3DvrTestCase(test_db_base_plugin_v2.NeutronDbPluginV2TestCase):
|
|||||||
plugin.get_ports.return_value = port
|
plugin.get_ports.return_value = port
|
||||||
grtr.return_value = router
|
grtr.return_value = router
|
||||||
self.mixin._delete_current_gw_port(
|
self.mixin._delete_current_gw_port(
|
||||||
self.ctx, router['id'], router, 'ext_network_id')
|
self.ctx, router['id'], router, new_network_id)
|
||||||
return router, plugin, del_csnat_port, del_agent_gw_port, del_fip
|
return router, plugin, del_csnat_port, del_agent_gw_port, del_fip
|
||||||
|
|
||||||
def test_delete_current_gw_port_deletes_fip_agent_gw_port_and_fipnamespace(
|
def test_delete_current_gw_port_deletes_fip_agent_gw_port_and_fipnamespace(
|
||||||
self):
|
self):
|
||||||
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
||||||
self._setup_delete_current_gw_port_deletes_fip_agent_gw_port())
|
self._setup_delete_current_gw_port_deletes_dvr_internal_ports())
|
||||||
self.assertTrue(d_csnat_port.called)
|
self.assertFalse(d_csnat_port.called)
|
||||||
self.assertTrue(d_agent_gw_port.called)
|
self.assertTrue(d_agent_gw_port.called)
|
||||||
d_csnat_port.assert_called_once_with(
|
|
||||||
mock.ANY, rtr)
|
|
||||||
d_agent_gw_port.assert_called_once_with(mock.ANY, None, 'ext_net_id')
|
d_agent_gw_port.assert_called_once_with(mock.ANY, None, 'ext_net_id')
|
||||||
del_fip.assert_called_once_with(mock.ANY, 'ext_net_id')
|
del_fip.assert_called_once_with(mock.ANY, 'ext_net_id')
|
||||||
|
|
||||||
@ -333,22 +331,30 @@ class L3DvrTestCase(test_db_base_plugin_v2.NeutronDbPluginV2TestCase):
|
|||||||
'device_owner': const.DEVICE_OWNER_ROUTER_GW
|
'device_owner': const.DEVICE_OWNER_ROUTER_GW
|
||||||
}]
|
}]
|
||||||
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
||||||
self._setup_delete_current_gw_port_deletes_fip_agent_gw_port(
|
self._setup_delete_current_gw_port_deletes_dvr_internal_ports(
|
||||||
port=port))
|
port=port))
|
||||||
self.assertTrue(d_csnat_port.called)
|
self.assertFalse(d_csnat_port.called)
|
||||||
self.assertFalse(d_agent_gw_port.called)
|
self.assertFalse(d_agent_gw_port.called)
|
||||||
self.assertFalse(del_fip.called)
|
self.assertFalse(del_fip.called)
|
||||||
d_csnat_port.assert_called_once_with(
|
|
||||||
mock.ANY, rtr)
|
|
||||||
|
|
||||||
def test_delete_current_gw_port_never_calls_delete_fipnamespace(self):
|
def test_delete_current_gw_port_never_calls_delete_fipnamespace(self):
|
||||||
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
||||||
self._setup_delete_current_gw_port_deletes_fip_agent_gw_port(
|
self._setup_delete_current_gw_port_deletes_dvr_internal_ports(
|
||||||
gw_port=False))
|
gw_port=False))
|
||||||
self.assertFalse(d_csnat_port.called)
|
self.assertFalse(d_csnat_port.called)
|
||||||
self.assertFalse(d_agent_gw_port.called)
|
self.assertFalse(d_agent_gw_port.called)
|
||||||
self.assertFalse(del_fip.called)
|
self.assertFalse(del_fip.called)
|
||||||
|
|
||||||
|
def test_delete_current_gw_port_deletes_csnat_port(self):
|
||||||
|
rtr, plugin, d_csnat_port, d_agent_gw_port, del_fip = (
|
||||||
|
self._setup_delete_current_gw_port_deletes_dvr_internal_ports(
|
||||||
|
new_network_id=None))
|
||||||
|
self.assertTrue(d_csnat_port.called)
|
||||||
|
self.assertTrue(d_agent_gw_port.called)
|
||||||
|
d_csnat_port.assert_called_once_with(mock.ANY, rtr)
|
||||||
|
d_agent_gw_port.assert_called_once_with(mock.ANY, None, 'ext_net_id')
|
||||||
|
del_fip.assert_called_once_with(mock.ANY, 'ext_net_id')
|
||||||
|
|
||||||
def _floatingip_on_port_test_setup(self, hostid):
|
def _floatingip_on_port_test_setup(self, hostid):
|
||||||
router = {'id': 'foo_router_id', 'distributed': True}
|
router = {'id': 'foo_router_id', 'distributed': True}
|
||||||
floatingip = {
|
floatingip = {
|
||||||
@ -461,6 +467,38 @@ class L3DvrTestCase(test_db_base_plugin_v2.NeutronDbPluginV2TestCase):
|
|||||||
fip, floatingip, router))
|
fip, floatingip, router))
|
||||||
self.assertFalse(create_fip.called)
|
self.assertFalse(create_fip.called)
|
||||||
|
|
||||||
|
def test_update_router_gw_info_external_network_change(self):
|
||||||
|
router_dict = {'name': 'test_router', 'admin_state_up': True,
|
||||||
|
'distributed': True}
|
||||||
|
router = self._create_router(router_dict)
|
||||||
|
with self.network() as net_ext_1,\
|
||||||
|
self.network() as net_ext_2,\
|
||||||
|
self.subnet() as subnet:
|
||||||
|
ext_net_1_id = net_ext_1['network']['id']
|
||||||
|
self.core_plugin.update_network(
|
||||||
|
self.ctx, ext_net_1_id,
|
||||||
|
{'network': {'router:external': True}})
|
||||||
|
self.mixin.update_router(
|
||||||
|
self.ctx, router['id'],
|
||||||
|
{'router': {'external_gateway_info':
|
||||||
|
{'network_id': ext_net_1_id}}})
|
||||||
|
self.mixin.add_router_interface(self.ctx, router['id'],
|
||||||
|
{'subnet_id': subnet['subnet']['id']})
|
||||||
|
|
||||||
|
ext_net_2_id = net_ext_2['network']['id']
|
||||||
|
self.core_plugin.update_network(
|
||||||
|
self.ctx, ext_net_2_id,
|
||||||
|
{'network': {'router:external': True}})
|
||||||
|
self.mixin.update_router(
|
||||||
|
self.ctx, router['id'],
|
||||||
|
{'router': {'external_gateway_info':
|
||||||
|
{'network_id': ext_net_2_id}}})
|
||||||
|
|
||||||
|
csnat_filters = {'device_owner': [const.DEVICE_OWNER_ROUTER_SNAT]}
|
||||||
|
csnat_ports = self.core_plugin.get_ports(
|
||||||
|
self.ctx, filters=csnat_filters)
|
||||||
|
self.assertEqual(1, len(csnat_ports))
|
||||||
|
|
||||||
def test_remove_router_interface_csnat_ports_removal(self):
|
def test_remove_router_interface_csnat_ports_removal(self):
|
||||||
router_dict = {'name': 'test_router', 'admin_state_up': True,
|
router_dict = {'name': 'test_router', 'admin_state_up': True,
|
||||||
'distributed': True}
|
'distributed': True}
|
||||||
|
Loading…
Reference in New Issue
Block a user