[server side] Expose port forwardings in FIP API

This patch introduces a new API extension named 'extend-fip-port-forwarding'
for exposing 'port_forwardings' field in floatingip responses.

Partially-Implements: blueprint port-forwarding
Change-Id: I9016abb6eb650c86c570a0ee78ee12361f4632e4
Partial-Bug: #1491317
This commit is contained in:
ZhaoBo 2018-07-03 22:47:04 +08:00
parent d33a8b6d05
commit d00a1558a5
3 changed files with 166 additions and 1 deletions
neutron
extensions
services/portforwarding
tests/unit/extensions

@ -0,0 +1,19 @@
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
from neutron_lib.api.definitions import expose_port_forwarding_in_fip as apiref
from neutron_lib.api import extensions
class Expose_port_forwarding_in_fip(extensions.APIExtensionDescriptor):
api_definition = apiref

@ -18,6 +18,7 @@ import functools
import netaddr
from neutron_lib.api.definitions import floating_ip_port_forwarding as apidef
from neutron_lib.api.definitions import l3
from neutron_lib.callbacks import registry
from neutron_lib import constants as lib_consts
from neutron_lib.db import utils as db_utils
@ -64,7 +65,8 @@ class PortForwardingPlugin(fip_pf.PortForwardingPluginBase):
This class implements a Port Forwarding plugin.
"""
supported_extension_aliases = ['floating-ip-port-forwarding']
supported_extension_aliases = ['floating-ip-port-forwarding',
'expose-port-forwarding-in-fip']
__native_pagination_support = True
__native_sorting_support = True
@ -75,6 +77,26 @@ class PortForwardingPlugin(fip_pf.PortForwardingPluginBase):
self.l3_plugin = directory.get_plugin(constants.L3)
self.core_plugin = directory.get_plugin()
@staticmethod
@resource_extend.extends([l3.FLOATINGIPS])
def _extend_floatingip_dict(result_dict, db):
fields = [apidef.INTERNAL_IP_ADDRESS, apidef.PROTOCOL,
apidef.INTERNAL_PORT, apidef.EXTERNAL_PORT]
result_dict[apidef.COLLECTION_NAME] = []
if db.port_forwardings:
port_forwarding_result = []
for port_forwarding in db.port_forwardings:
pf_dict = pf.PortForwarding.modify_fields_from_db(
port_forwarding)
for key in list(pf_dict.keys()):
if key not in fields:
pf_dict.pop(key)
elif key == apidef.INTERNAL_IP_ADDRESS:
pf_dict[key] = str(pf_dict[key])
port_forwarding_result.append(pf_dict)
result_dict[apidef.COLLECTION_NAME] = port_forwarding_result
return result_dict
def _get_internal_ip_subnet(self, request_ip, fixed_ips):
request_ip = netaddr.IPNetwork(request_ip)
for fixed_ip in fixed_ips:

@ -0,0 +1,124 @@
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import mock
from neutron_lib.api.definitions import external_net as extnet_apidef
from neutron_lib.api.definitions import floating_ip_port_forwarding as apidef
from neutron_lib import context
from neutron.extensions import floating_ip_port_forwarding as pf_ext
from neutron.extensions import l3
from neutron.services.portforwarding import pf_plugin
from neutron.tests.unit.db import test_db_base_plugin_v2
from neutron.tests.unit.extensions import test_l3
PLUGIN_NAME = 'port_forwarding'
L3_PLUGIN = 'neutron.tests.unit.extensions.test_l3.TestL3NatServicePlugin'
CORE_PLUGIN = 'neutron.tests.unit.extensions.test_l3.TestNoL3NatPlugin'
class ExtendFipPortForwardingExtensionManager(object):
def get_resources(self):
return (l3.L3.get_resources() +
pf_ext.Floating_ip_port_forwarding.get_resources())
def get_actions(self):
return []
def get_request_extensions(self):
return []
class TestExtendFipPortForwardingExtension(
test_db_base_plugin_v2.NeutronDbPluginV2TestCase,
test_l3.L3NatTestCaseMixin):
def setUp(self):
mock.patch('neutron.api.rpc.handlers.resources_rpc.'
'ResourcesPushRpcApi').start()
svc_plugins = {'l3_plugin_name': L3_PLUGIN,
'port_forwarding_plugin_name': PLUGIN_NAME}
ext_mgr = ExtendFipPortForwardingExtensionManager()
super(TestExtendFipPortForwardingExtension, self).setUp(
plugin=CORE_PLUGIN, ext_mgr=ext_mgr, service_plugins=svc_plugins)
self.pf_plugin = pf_plugin.PortForwardingPlugin()
def test_get_fip_after_port_forwarding_create(self):
port_forwarding = {
apidef.RESOURCE_NAME:
{apidef.EXTERNAL_PORT: 2225,
apidef.INTERNAL_PORT: 25,
apidef.INTERNAL_PORT_ID: None,
apidef.PROTOCOL: "tcp",
apidef.INTERNAL_IP_ADDRESS: None}}
ctx = context.get_admin_context()
kwargs = {'arg_list': (extnet_apidef.EXTERNAL,),
extnet_apidef.EXTERNAL: True}
with self.network(**kwargs) as extnet, self.network() as innet:
with self.subnet(network=extnet, cidr='200.0.0.0/22'),\
self.subnet(network=innet, cidr='10.0.0.0/24') as insub,\
self.router() as router:
fip = self._make_floatingip(self.fmt, extnet['network']['id'])
# check the floatingip response contains port_forwarding field
self.assertIn(apidef.COLLECTION_NAME, fip['floatingip'])
self._add_external_gateway_to_router(router['router']['id'],
extnet['network']['id'])
self._router_interface_action('add', router['router']['id'],
insub['subnet']['id'], None)
def _get_expected(ref):
want_fields = [apidef.INTERNAL_IP_ADDRESS, apidef.PROTOCOL,
apidef.INTERNAL_PORT, apidef.EXTERNAL_PORT]
expect = {
key: value
for key, value in ref[apidef.RESOURCE_NAME].items()
if key in want_fields}
return expect
with self.port(subnet=insub) as port1,\
self.port(subnet=insub) as port2:
update_dict1 = {
apidef.INTERNAL_PORT_ID: port1['port']['id'],
apidef.INTERNAL_IP_ADDRESS:
port1['port']['fixed_ips'][0]['ip_address']}
port_forwarding[apidef.RESOURCE_NAME].update(update_dict1)
self.pf_plugin.create_floatingip_port_forwarding(
ctx, fip['floatingip']['id'], port_forwarding)
body = self._show('floatingips', fip['floatingip']['id'])
self.assertEqual(
1, len(body['floatingip'][apidef.COLLECTION_NAME]))
expect_result1 = _get_expected(port_forwarding)
self.assertEqual(
expect_result1,
body['floatingip'][apidef.COLLECTION_NAME][0])
update_dict2 = {
apidef.EXTERNAL_PORT: 2226,
apidef.INTERNAL_PORT_ID: port2['port']['id'],
apidef.INTERNAL_IP_ADDRESS:
port2['port']['fixed_ips'][0]['ip_address']}
port_forwarding[apidef.RESOURCE_NAME].update(update_dict2)
self.pf_plugin.create_floatingip_port_forwarding(
ctx, fip['floatingip']['id'], port_forwarding)
body = self._show('floatingips', fip['floatingip']['id'])
self.assertEqual(
2, len(body['floatingip'][apidef.COLLECTION_NAME]))
expect_result2 = _get_expected(port_forwarding)
expect = [expect_result1, expect_result2]
self.assertEqual(
expect, body['floatingip'][apidef.COLLECTION_NAME])