NSXv3: Fix tap-flow-create to use floating IP of destination port

This patch changes the workflow for L3 SPAN such that the destination
port must have a L3 reachable IP address i.e. a floating IP so that
L3 SPAN can work. Unit tests will be added in a separate patch.

Change-Id: Ie0c2aed71988cbbd9cba60bb1fcb1ec30d0b4bc4
This commit is contained in:
Abhishek Raut 2016-08-19 21:57:57 -07:00 committed by Gary Kotton
parent 288e242c99
commit 58dc54f52a
3 changed files with 27 additions and 6 deletions

View File

@ -2593,6 +2593,18 @@ class NsxV3Plugin(agentschedulers_db.AZDhcpAgentSchedulerDbMixin,
if floatingip['floatingip']['port_id'] if floatingip['floatingip']['port_id']
else const.FLOATINGIP_STATUS_DOWN)) else const.FLOATINGIP_STATUS_DOWN))
def get_floatingips(self, context, filters=None, fields=None,
sorts=None, limit=None, marker=None,
page_reverse=False):
#NOTE(abhiraut): Although this method does not do anything fancy
# we have defined it here so that TaaS driver
# can retrieve floating IP for a particular port ID
# using filters.
return super(NsxV3Plugin, self).get_floatingips(
context, filters=filters, fields=fields,
sorts=sorts, limit=limit, marker=marker,
page_reverse=page_reverse)
def create_floatingip(self, context, floatingip): def create_floatingip(self, context, floatingip):
new_fip = self._create_floating_ip_wrapper(context, floatingip) new_fip = self._create_floating_ip_wrapper(context, floatingip)
router_id = new_fip['router_id'] router_id = new_fip['router_id']

View File

@ -195,22 +195,28 @@ class NsxV3Driver(base_driver.TaasBaseDriver,
tags): tags):
"""Create a PortMirroring SwitchingProfile for L3SPAN.""" """Create a PortMirroring SwitchingProfile for L3SPAN."""
tf = context.tap_flow tf = context.tap_flow
dest_port = self._get_port_details(context._plugin_context, # Verify whether destination port is L3 reachable. i.e. destination
dest_port_id) # port has a floating IP address.
fips = self._nsx_plugin.get_floatingips(
context._plugin_context, filters={'port_id': dest_port_id})
if not fips:
msg = (_("Destination port %s must have a floating IP for "
"L3 SPAN") % dest_port_id)
raise nsx_exc.NsxTaaSDriverException(msg=msg)
destinations = [] destinations = []
# Retrieve destination port's IP addresses and add it to the list # Retrieve destination port's IP addresses and add it to the list
# since the backend expects a list of IP addresses. # since the backend expects a list of IP addresses.
for fixed_ip in dest_port['fixed_ips']: for fip in fips:
# NOTE(abhiraut): nsx-v3 doesn't seem to handle ipv6 addresses # NOTE(abhiraut): nsx-v3 doesn't seem to handle ipv6 addresses
# currently so for now we remove them here and do not pass # currently so for now we remove them here and do not pass
# them to the backend which would raise an error. # them to the backend which would raise an error.
if netaddr.IPNetwork(fixed_ip['ip_address']).version == 6: if netaddr.IPNetwork(fip['floating_ip_address']).version == 6:
LOG.warning(_LW("Skipping IPv6 address %(ip)s for L3SPAN " LOG.warning(_LW("Skipping IPv6 address %(ip)s for L3SPAN "
"tap flow: %(tap_flow)s"), "tap flow: %(tap_flow)s"),
{'tap_flow': tf['id'], {'tap_flow': tf['id'],
'ip': fixed_ip['ip_address']}) 'ip': fip['floating_ip_address']})
continue continue
destinations.append(fixed_ip['ip_address']) destinations.append(fip['floating_ip_address'])
# Create a switch profile in the backend. # Create a switch profile in the backend.
try: try:
port_mirror_profile = (self._nsx_plugin._switching_profiles. port_mirror_profile = (self._nsx_plugin._switching_profiles.

View File

@ -45,6 +45,9 @@ class TestNsxV3TaaSDriver(test_taas_db.TaaSDbTestCase,
return_value=mock.MagicMock()).start() return_value=mock.MagicMock()).start()
self.taas_plugin = taas_plugin.TaasPlugin() self.taas_plugin = taas_plugin.TaasPlugin()
self.core_plugin = importutils.import_object(NSX_V3_PLUGIN_CLASS) self.core_plugin = importutils.import_object(NSX_V3_PLUGIN_CLASS)
mock.patch(
'vmware_nsx.plugins.nsx_v3.plugin.NsxV3Plugin.get_floatingips',
return_value=([{'floating_ip_address': '172.10.10.10'}])).start()
self.ctx = context.get_admin_context() self.ctx = context.get_admin_context()
def test_validate_tap_flow_same_network_same_port_fail(self): def test_validate_tap_flow_same_network_same_port_fail(self):