Introduce status for floating IPs
Add a new 'status' attribute to the floating IP resource. Extend the plugin RPC interface for allowing status updates from agents, and implement support for operational status in the L3 agent. The default behaviour for all the plugins different from neutron.services.l3_router.l3_router_plugin is to set the status of a floating IP to ACTIVE upon creation. Implements blueprint fip-op-status Change-Id: Ib0176fbcde95b0db5dfdabd4fc297d1d29c0d604
This commit is contained in:
parent
cdeace03d6
commit
0608b455ec
@ -37,6 +37,7 @@ from neutron.openstack.common import lockutils
|
|||||||
from neutron.openstack.common import log as logging
|
from neutron.openstack.common import log as logging
|
||||||
from neutron.openstack.common import loopingcall
|
from neutron.openstack.common import loopingcall
|
||||||
from neutron.openstack.common import periodic_task
|
from neutron.openstack.common import periodic_task
|
||||||
|
from neutron.openstack.common import processutils
|
||||||
from neutron.openstack.common.rpc import common as rpc_common
|
from neutron.openstack.common.rpc import common as rpc_common
|
||||||
from neutron.openstack.common.rpc import proxy
|
from neutron.openstack.common.rpc import proxy
|
||||||
from neutron.openstack.common import service
|
from neutron.openstack.common import service
|
||||||
@ -56,6 +57,7 @@ class L3PluginApi(proxy.RpcProxy):
|
|||||||
|
|
||||||
API version history:
|
API version history:
|
||||||
1.0 - Initial version.
|
1.0 - Initial version.
|
||||||
|
1.1 - Floating IP operational status updates
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -85,6 +87,15 @@ class L3PluginApi(proxy.RpcProxy):
|
|||||||
host=self.host),
|
host=self.host),
|
||||||
topic=self.topic)
|
topic=self.topic)
|
||||||
|
|
||||||
|
def update_floatingip_statuses(self, context, router_id, fip_statuses):
|
||||||
|
"""Call the plugin update floating IPs's operational status."""
|
||||||
|
return self.call(context,
|
||||||
|
self.make_msg('update_floatingip_statuses',
|
||||||
|
router_id=router_id,
|
||||||
|
fip_statuses=fip_statuses),
|
||||||
|
topic=self.topic,
|
||||||
|
version='1.1')
|
||||||
|
|
||||||
|
|
||||||
class RouterInfo(object):
|
class RouterInfo(object):
|
||||||
|
|
||||||
@ -94,6 +105,7 @@ class RouterInfo(object):
|
|||||||
self._snat_enabled = None
|
self._snat_enabled = None
|
||||||
self._snat_action = None
|
self._snat_action = None
|
||||||
self.internal_ports = []
|
self.internal_ports = []
|
||||||
|
self.floating_ips = set()
|
||||||
self.root_helper = root_helper
|
self.root_helper = root_helper
|
||||||
self.use_namespaces = use_namespaces
|
self.use_namespaces = use_namespaces
|
||||||
# Invoke the setter for establishing initial SNAT action
|
# Invoke the setter for establishing initial SNAT action
|
||||||
@ -434,21 +446,41 @@ class L3NATAgent(firewall_l3_agent.FWaaSL3AgentRpcCallback, manager.Manager):
|
|||||||
self.external_gateway_removed(ri, ri.ex_gw_port,
|
self.external_gateway_removed(ri, ri.ex_gw_port,
|
||||||
interface_name, internal_cidrs)
|
interface_name, internal_cidrs)
|
||||||
|
|
||||||
|
# Process static routes for router
|
||||||
|
self.routes_updated(ri)
|
||||||
# Process SNAT rules for external gateway
|
# Process SNAT rules for external gateway
|
||||||
ri.perform_snat_action(self._handle_router_snat_rules,
|
ri.perform_snat_action(self._handle_router_snat_rules,
|
||||||
internal_cidrs, interface_name)
|
internal_cidrs, interface_name)
|
||||||
|
|
||||||
# Process SNAT/DNAT rules for floating IPs
|
# Process SNAT/DNAT rules for floating IPs
|
||||||
if ex_gw_port:
|
fip_statuses = {}
|
||||||
self.process_router_floating_ip_nat_rules(ri)
|
try:
|
||||||
|
if ex_gw_port:
|
||||||
|
existing_floating_ips = ri.floating_ips
|
||||||
|
self.process_router_floating_ip_nat_rules(ri)
|
||||||
|
ri.iptables_manager.defer_apply_off()
|
||||||
|
# Once NAT rules for floating IPs are safely in place
|
||||||
|
# configure their addresses on the external gateway port
|
||||||
|
fip_statuses = self.process_router_floating_ip_addresses(
|
||||||
|
ri, ex_gw_port)
|
||||||
|
except Exception:
|
||||||
|
# TODO(salv-orlando): Less broad catching
|
||||||
|
# All floating IPs must be put in error state
|
||||||
|
for fip in ri.router.get(l3_constants.FLOATINGIP_KEY, []):
|
||||||
|
fip_statuses[fip] = l3_constants.FLOATINGIP_STATUS_ERROR
|
||||||
|
|
||||||
ri.ex_gw_port = ex_gw_port
|
|
||||||
self.routes_updated(ri)
|
|
||||||
ri.iptables_manager.defer_apply_off()
|
|
||||||
# Once NAT rules for floating IPs are safely in place
|
|
||||||
# configure their addresses on the external gateway port
|
|
||||||
if ex_gw_port:
|
if ex_gw_port:
|
||||||
self.process_router_floating_ip_addresses(ri, ex_gw_port)
|
# Identify floating IPs which were disabled
|
||||||
|
ri.floating_ips = set(fip_statuses.keys())
|
||||||
|
for fip_id in existing_floating_ips - ri.floating_ips:
|
||||||
|
fip_statuses[fip_id] = l3_constants.FLOATINGIP_STATUS_DOWN
|
||||||
|
# Update floating IP status on the neutron server
|
||||||
|
self.plugin_rpc.update_floatingip_statuses(
|
||||||
|
self.context, ri.router_id, fip_statuses)
|
||||||
|
|
||||||
|
# Update ex_gw_port and enable_snat on the router info cache
|
||||||
|
ri.ex_gw_port = ex_gw_port
|
||||||
|
ri.enable_snat = ri.router.get('enable_snat')
|
||||||
|
|
||||||
def _handle_router_snat_rules(self, ri, ex_gw_port, internal_cidrs,
|
def _handle_router_snat_rules(self, ri, ex_gw_port, internal_cidrs,
|
||||||
interface_name, action):
|
interface_name, action):
|
||||||
@ -497,6 +529,7 @@ class L3NATAgent(firewall_l3_agent.FWaaSL3AgentRpcCallback, manager.Manager):
|
|||||||
Ensures addresses for existing floating IPs and cleans up
|
Ensures addresses for existing floating IPs and cleans up
|
||||||
those that should not longer be configured.
|
those that should not longer be configured.
|
||||||
"""
|
"""
|
||||||
|
fip_statuses = {}
|
||||||
interface_name = self.get_external_device_name(ex_gw_port['id'])
|
interface_name = self.get_external_device_name(ex_gw_port['id'])
|
||||||
device = ip_lib.IPDevice(interface_name, self.root_helper,
|
device = ip_lib.IPDevice(interface_name, self.root_helper,
|
||||||
namespace=ri.ns_name())
|
namespace=ri.ns_name())
|
||||||
@ -512,14 +545,30 @@ class L3NATAgent(firewall_l3_agent.FWaaSL3AgentRpcCallback, manager.Manager):
|
|||||||
|
|
||||||
if ip_cidr not in existing_cidrs:
|
if ip_cidr not in existing_cidrs:
|
||||||
net = netaddr.IPNetwork(ip_cidr)
|
net = netaddr.IPNetwork(ip_cidr)
|
||||||
device.addr.add(net.version, ip_cidr, str(net.broadcast))
|
try:
|
||||||
self._send_gratuitous_arp_packet(ri, interface_name, fip_ip)
|
device.addr.add(net.version, ip_cidr, str(net.broadcast))
|
||||||
|
except (processutils.UnknownArgumentError,
|
||||||
|
processutils.ProcessExecutionError):
|
||||||
|
# any exception occurred here should cause the floating IP
|
||||||
|
# to be set in error state
|
||||||
|
fip_statuses[fip['id']] = (
|
||||||
|
l3_constants.FLOATINGIP_STATUS_ERROR)
|
||||||
|
LOG.warn(_("Unable to configure IP address for "
|
||||||
|
"floating IP: %s"), fip['id'])
|
||||||
|
continue
|
||||||
|
# As GARP is processed in a distinct thread the call below
|
||||||
|
# won't raise an exception to be handled.
|
||||||
|
self._send_gratuitous_arp_packet(
|
||||||
|
ri, interface_name, fip_ip)
|
||||||
|
fip_statuses[fip['id']] = (
|
||||||
|
l3_constants.FLOATINGIP_STATUS_ACTIVE)
|
||||||
|
|
||||||
# Clean up addresses that no longer belong on the gateway interface.
|
# Clean up addresses that no longer belong on the gateway interface.
|
||||||
for ip_cidr in existing_cidrs - new_cidrs:
|
for ip_cidr in existing_cidrs - new_cidrs:
|
||||||
if ip_cidr.endswith(FLOATING_IP_CIDR_SUFFIX):
|
if ip_cidr.endswith(FLOATING_IP_CIDR_SUFFIX):
|
||||||
net = netaddr.IPNetwork(ip_cidr)
|
net = netaddr.IPNetwork(ip_cidr)
|
||||||
device.addr.delete(net.version, ip_cidr)
|
device.addr.delete(net.version, ip_cidr)
|
||||||
|
return fip_statuses
|
||||||
|
|
||||||
def _get_ex_gw_port(self, ri):
|
def _get_ex_gw_port(self, ri):
|
||||||
return ri.router.get('gw_port')
|
return ri.router.get('gw_port')
|
||||||
|
@ -13,6 +13,8 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
|
|
||||||
|
# TODO(salv-orlando): Verify if a single set of operational
|
||||||
|
# status constants is achievable
|
||||||
NET_STATUS_ACTIVE = 'ACTIVE'
|
NET_STATUS_ACTIVE = 'ACTIVE'
|
||||||
NET_STATUS_BUILD = 'BUILD'
|
NET_STATUS_BUILD = 'BUILD'
|
||||||
NET_STATUS_DOWN = 'DOWN'
|
NET_STATUS_DOWN = 'DOWN'
|
||||||
@ -23,6 +25,10 @@ PORT_STATUS_BUILD = 'BUILD'
|
|||||||
PORT_STATUS_DOWN = 'DOWN'
|
PORT_STATUS_DOWN = 'DOWN'
|
||||||
PORT_STATUS_ERROR = 'ERROR'
|
PORT_STATUS_ERROR = 'ERROR'
|
||||||
|
|
||||||
|
FLOATINGIP_STATUS_ACTIVE = 'ACTIVE'
|
||||||
|
FLOATINGIP_STATUS_DOWN = 'DOWN'
|
||||||
|
FLOATINGIP_STATUS_ERROR = 'ERROR'
|
||||||
|
|
||||||
DEVICE_OWNER_ROUTER_INTF = "network:router_interface"
|
DEVICE_OWNER_ROUTER_INTF = "network:router_interface"
|
||||||
DEVICE_OWNER_ROUTER_GW = "network:router_gateway"
|
DEVICE_OWNER_ROUTER_GW = "network:router_gateway"
|
||||||
DEVICE_OWNER_FLOATINGIP = "network:floatingip"
|
DEVICE_OWNER_FLOATINGIP = "network:floatingip"
|
||||||
|
@ -68,6 +68,11 @@ class FloatingIP(model_base.BASEV2, models_v2.HasId, models_v2.HasTenant):
|
|||||||
fixed_port_id = sa.Column(sa.String(36), sa.ForeignKey('ports.id'))
|
fixed_port_id = sa.Column(sa.String(36), sa.ForeignKey('ports.id'))
|
||||||
fixed_ip_address = sa.Column(sa.String(64))
|
fixed_ip_address = sa.Column(sa.String(64))
|
||||||
router_id = sa.Column(sa.String(36), sa.ForeignKey('routers.id'))
|
router_id = sa.Column(sa.String(36), sa.ForeignKey('routers.id'))
|
||||||
|
# Additional attribute for keeping track of the router where the floating
|
||||||
|
# ip was associated in order to be able to ensure consistency even if an
|
||||||
|
# aysnchronous backend is unavailable when the floating IP is disassociated
|
||||||
|
last_known_router_id = sa.Column(sa.String(36))
|
||||||
|
status = sa.Column(sa.String(16))
|
||||||
|
|
||||||
|
|
||||||
class L3_NAT_db_mixin(l3.RouterPluginBase):
|
class L3_NAT_db_mixin(l3.RouterPluginBase):
|
||||||
@ -450,7 +455,8 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
|
|||||||
'floating_network_id': floatingip['floating_network_id'],
|
'floating_network_id': floatingip['floating_network_id'],
|
||||||
'router_id': floatingip['router_id'],
|
'router_id': floatingip['router_id'],
|
||||||
'port_id': floatingip['fixed_port_id'],
|
'port_id': floatingip['fixed_port_id'],
|
||||||
'fixed_ip_address': floatingip['fixed_ip_address']}
|
'fixed_ip_address': floatingip['fixed_ip_address'],
|
||||||
|
'status': floatingip['status']}
|
||||||
return self._fields(res, fields)
|
return self._fields(res, fields)
|
||||||
|
|
||||||
def _get_router_for_floatingip(self, context, internal_port,
|
def _get_router_for_floatingip(self, context, internal_port,
|
||||||
@ -564,6 +570,7 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
|
|||||||
return (fip['port_id'], internal_ip_address, router_id)
|
return (fip['port_id'], internal_ip_address, router_id)
|
||||||
|
|
||||||
def _update_fip_assoc(self, context, fip, floatingip_db, external_port):
|
def _update_fip_assoc(self, context, fip, floatingip_db, external_port):
|
||||||
|
previous_router_id = floatingip_db.router_id
|
||||||
port_id = internal_ip_address = router_id = None
|
port_id = internal_ip_address = router_id = None
|
||||||
if (('fixed_ip_address' in fip and fip['fixed_ip_address']) and
|
if (('fixed_ip_address' in fip and fip['fixed_ip_address']) and
|
||||||
not ('port_id' in fip and fip['port_id'])):
|
not ('port_id' in fip and fip['port_id'])):
|
||||||
@ -590,9 +597,12 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
|
|||||||
pass
|
pass
|
||||||
floatingip_db.update({'fixed_ip_address': internal_ip_address,
|
floatingip_db.update({'fixed_ip_address': internal_ip_address,
|
||||||
'fixed_port_id': port_id,
|
'fixed_port_id': port_id,
|
||||||
'router_id': router_id})
|
'router_id': router_id,
|
||||||
|
'last_known_router_id': previous_router_id})
|
||||||
|
|
||||||
def create_floatingip(self, context, floatingip):
|
def create_floatingip(
|
||||||
|
self, context, floatingip,
|
||||||
|
initial_status=l3_constants.FLOATINGIP_STATUS_ACTIVE):
|
||||||
fip = floatingip['floatingip']
|
fip = floatingip['floatingip']
|
||||||
tenant_id = self._get_tenant_id_for_create(context, fip)
|
tenant_id = self._get_tenant_id_for_create(context, fip)
|
||||||
fip_id = uuidutils.generate_uuid()
|
fip_id = uuidutils.generate_uuid()
|
||||||
@ -625,6 +635,7 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
|
|||||||
floatingip_db = FloatingIP(
|
floatingip_db = FloatingIP(
|
||||||
id=fip_id,
|
id=fip_id,
|
||||||
tenant_id=tenant_id,
|
tenant_id=tenant_id,
|
||||||
|
status=initial_status,
|
||||||
floating_network_id=fip['floating_network_id'],
|
floating_network_id=fip['floating_network_id'],
|
||||||
floating_ip_address=floating_ip_address,
|
floating_ip_address=floating_ip_address,
|
||||||
floating_port_id=external_port['id'])
|
floating_port_id=external_port['id'])
|
||||||
@ -664,6 +675,13 @@ class L3_NAT_db_mixin(l3.RouterPluginBase):
|
|||||||
context, router_ids, 'update_floatingip')
|
context, router_ids, 'update_floatingip')
|
||||||
return self._make_floatingip_dict(floatingip_db)
|
return self._make_floatingip_dict(floatingip_db)
|
||||||
|
|
||||||
|
def update_floatingip_status(self, context, floatingip_id, status):
|
||||||
|
"""Update operational status for floating IP in neutron DB."""
|
||||||
|
# TODO(salv-orlando): Optimize avoiding fetch before update
|
||||||
|
with context.session.begin(subtransactions=True):
|
||||||
|
floatingip_db = self._get_floatingip(context, floatingip_id)
|
||||||
|
floatingip_db['status'] = status
|
||||||
|
|
||||||
def delete_floatingip(self, context, id):
|
def delete_floatingip(self, context, id):
|
||||||
floatingip = self._get_floatingip(context, id)
|
floatingip = self._get_floatingip(context, id)
|
||||||
router_id = floatingip['router_id']
|
router_id = floatingip['router_id']
|
||||||
|
@ -94,3 +94,30 @@ class L3RpcCallbackMixin(object):
|
|||||||
LOG.debug(_("External network ID returned to l3 agent: %s"),
|
LOG.debug(_("External network ID returned to l3 agent: %s"),
|
||||||
net_id)
|
net_id)
|
||||||
return net_id
|
return net_id
|
||||||
|
|
||||||
|
def update_floatingip_statuses(self, context, router_id, fip_statuses):
|
||||||
|
"""Update operational status for a floating IP."""
|
||||||
|
l3_plugin = manager.NeutronManager.get_service_plugins()[
|
||||||
|
plugin_constants.L3_ROUTER_NAT]
|
||||||
|
with context.session.begin(subtransactions=True):
|
||||||
|
for (floatingip_id, status) in fip_statuses.iteritems():
|
||||||
|
LOG.debug(_("New status for floating IP %(floatingip_id)s: "
|
||||||
|
"%(status)s"), {'floatingip_id': floatingip_id,
|
||||||
|
'status': status})
|
||||||
|
l3_plugin.update_floatingip_status(context,
|
||||||
|
floatingip_id,
|
||||||
|
status)
|
||||||
|
# Find all floating IPs known to have been the given router
|
||||||
|
# for which an update was not received. Set them DOWN mercilessly
|
||||||
|
# This situation might occur for some asynchronous backends if
|
||||||
|
# notifications were missed
|
||||||
|
known_router_fips = l3_plugin.get_floatingips(
|
||||||
|
context, {'last_known_router_id': [router_id]})
|
||||||
|
# Consider only floating ips which were disassociated in the API
|
||||||
|
# FIXME(salv-orlando): Filtering in code should be avoided.
|
||||||
|
# the plugin should offer a way to specify a null filter
|
||||||
|
fips_to_disable = (fip['id'] for fip in known_router_fips
|
||||||
|
if not fip['router_id'])
|
||||||
|
for fip_id in fips_to_disable:
|
||||||
|
l3_plugin.update_floatingip_status(
|
||||||
|
context, fip_id, constants.FLOATINGIP_STATUS_DOWN)
|
||||||
|
@ -0,0 +1,59 @@
|
|||||||
|
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
||||||
|
#
|
||||||
|
# Copyright 2014 OpenStack Foundation
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
#
|
||||||
|
|
||||||
|
"""floatingip_status
|
||||||
|
|
||||||
|
Revision ID: 2eeaf963a447
|
||||||
|
Revises: f44ab9871cd6
|
||||||
|
Create Date: 2014-01-14 11:58:13.754747
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
# revision identifiers, used by Alembic.
|
||||||
|
revision = '2eeaf963a447'
|
||||||
|
down_revision = 'f44ab9871cd6'
|
||||||
|
|
||||||
|
# Change to ['*'] if this migration applies to all plugins
|
||||||
|
|
||||||
|
migration_for_plugins = [
|
||||||
|
'*'
|
||||||
|
]
|
||||||
|
|
||||||
|
from alembic import op
|
||||||
|
import sqlalchemy as sa
|
||||||
|
|
||||||
|
from neutron.db import migration
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade(active_plugins=None, options=None):
|
||||||
|
if not migration.should_run(active_plugins, migration_for_plugins):
|
||||||
|
return
|
||||||
|
op.add_column('floatingips',
|
||||||
|
sa.Column('last_known_router_id',
|
||||||
|
sa.String(length=36),
|
||||||
|
nullable=True))
|
||||||
|
op.add_column('floatingips',
|
||||||
|
sa.Column('status',
|
||||||
|
sa.String(length=16),
|
||||||
|
nullable=True))
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade(active_plugins=None, options=None):
|
||||||
|
if not migration.should_run(active_plugins, migration_for_plugins):
|
||||||
|
return
|
||||||
|
op.drop_column('floatingips', 'last_known_router_id')
|
||||||
|
op.drop_column('floatingips', 'status')
|
@ -125,7 +125,9 @@ RESOURCE_ATTRIBUTE_MAP = {
|
|||||||
'tenant_id': {'allow_post': True, 'allow_put': False,
|
'tenant_id': {'allow_post': True, 'allow_put': False,
|
||||||
'required_by_policy': True,
|
'required_by_policy': True,
|
||||||
'validate': {'type:string': None},
|
'validate': {'type:string': None},
|
||||||
'is_visible': True}
|
'is_visible': True},
|
||||||
|
'status': {'allow_post': False, 'allow_put': False,
|
||||||
|
'is_visible': True},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,8 +37,7 @@ from neutron.plugins.common import constants
|
|||||||
|
|
||||||
class L3RouterPluginRpcCallbacks(l3_rpc_base.L3RpcCallbackMixin):
|
class L3RouterPluginRpcCallbacks(l3_rpc_base.L3RpcCallbackMixin):
|
||||||
|
|
||||||
# Set RPC API version to 1.0 by default.
|
RPC_API_VERSION = '1.1'
|
||||||
RPC_API_VERSION = '1.0'
|
|
||||||
|
|
||||||
def create_rpc_dispatcher(self):
|
def create_rpc_dispatcher(self):
|
||||||
"""Get the rpc dispatcher for this manager.
|
"""Get the rpc dispatcher for this manager.
|
||||||
@ -91,3 +90,18 @@ class L3RouterPlugin(db_base_plugin_v2.CommonDbMixin,
|
|||||||
return ("L3 Router Service Plugin for basic L3 forwarding"
|
return ("L3 Router Service Plugin for basic L3 forwarding"
|
||||||
" between (L2) Neutron networks and access to external"
|
" between (L2) Neutron networks and access to external"
|
||||||
" networks via a NAT gateway.")
|
" networks via a NAT gateway.")
|
||||||
|
|
||||||
|
def create_floatingip(self, context, floatingip):
|
||||||
|
"""Create floating IP.
|
||||||
|
|
||||||
|
:param context: Neutron request context
|
||||||
|
:param floatingip: data fo the floating IP being created
|
||||||
|
:returns: A floating IP object on success
|
||||||
|
|
||||||
|
AS the l3 router plugin aysnchrounously creates floating IPs
|
||||||
|
leveraging tehe l3 agent, the initial status fro the floating
|
||||||
|
IP object will be DOWN.
|
||||||
|
"""
|
||||||
|
return super(L3RouterPlugin, self).create_floatingip(
|
||||||
|
context, floatingip,
|
||||||
|
initial_status=q_const.FLOATINGIP_STATUS_DOWN)
|
||||||
|
@ -25,6 +25,7 @@ from neutron.agent.linux import interface
|
|||||||
from neutron.common import config as base_config
|
from neutron.common import config as base_config
|
||||||
from neutron.common import constants as l3_constants
|
from neutron.common import constants as l3_constants
|
||||||
from neutron.common import exceptions as n_exc
|
from neutron.common import exceptions as n_exc
|
||||||
|
from neutron.openstack.common import processutils
|
||||||
from neutron.openstack.common import uuidutils
|
from neutron.openstack.common import uuidutils
|
||||||
from neutron.tests import base
|
from neutron.tests import base
|
||||||
|
|
||||||
@ -382,11 +383,14 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
|
|
||||||
def test_process_router(self):
|
def test_process_router(self):
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
fake_fip_id = 'fake_fip_id'
|
||||||
agent.process_router_floating_ip_addresses = mock.Mock()
|
agent.process_router_floating_ip_addresses = mock.Mock()
|
||||||
agent.process_router_floating_ip_nat_rules = mock.Mock()
|
agent.process_router_floating_ip_nat_rules = mock.Mock()
|
||||||
|
agent.process_router_floating_ip_addresses.return_value = {
|
||||||
|
fake_fip_id: 'ACTIVE'}
|
||||||
router = self._prepare_router_data()
|
router = self._prepare_router_data()
|
||||||
fake_floatingips1 = {'floatingips': [
|
fake_floatingips1 = {'floatingips': [
|
||||||
{'id': _uuid(),
|
{'id': fake_fip_id,
|
||||||
'floating_ip_address': '8.8.8.8',
|
'floating_ip_address': '8.8.8.8',
|
||||||
'fixed_ip_address': '7.7.7.7',
|
'fixed_ip_address': '7.7.7.7',
|
||||||
'port_id': _uuid()}]}
|
'port_id': _uuid()}]}
|
||||||
@ -433,8 +437,9 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
|
|
||||||
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
||||||
def test_process_router_floating_ip_addresses_add(self, IPDevice):
|
def test_process_router_floating_ip_addresses_add(self, IPDevice):
|
||||||
|
fip_id = _uuid()
|
||||||
fip = {
|
fip = {
|
||||||
'id': _uuid(), 'port_id': _uuid(),
|
'id': fip_id, 'port_id': _uuid(),
|
||||||
'floating_ip_address': '15.1.2.3',
|
'floating_ip_address': '15.1.2.3',
|
||||||
'fixed_ip_address': '192.168.0.1'
|
'fixed_ip_address': '192.168.0.1'
|
||||||
}
|
}
|
||||||
@ -447,8 +452,10 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
|
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
|
||||||
agent.process_router_floating_ip_addresses(ri, {'id': _uuid()})
|
fip_statuses = agent.process_router_floating_ip_addresses(
|
||||||
|
ri, {'id': _uuid()})
|
||||||
|
self.assertEqual({fip_id: l3_constants.FLOATINGIP_STATUS_ACTIVE},
|
||||||
|
fip_statuses)
|
||||||
device.addr.add.assert_called_once_with(4, '15.1.2.3/32', '15.1.2.3')
|
device.addr.add.assert_called_once_with(4, '15.1.2.3/32', '15.1.2.3')
|
||||||
|
|
||||||
def test_process_router_floating_ip_nat_rules_add(self):
|
def test_process_router_floating_ip_nat_rules_add(self):
|
||||||
@ -481,8 +488,9 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
|
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
|
||||||
agent.process_router_floating_ip_addresses(ri, {'id': _uuid()})
|
fip_statuses = agent.process_router_floating_ip_addresses(
|
||||||
|
ri, {'id': _uuid()})
|
||||||
|
self.assertEqual({}, fip_statuses)
|
||||||
device.addr.delete.assert_called_once_with(4, '15.1.2.3/32')
|
device.addr.delete.assert_called_once_with(4, '15.1.2.3/32')
|
||||||
|
|
||||||
def test_process_router_floating_ip_nat_rules_remove(self):
|
def test_process_router_floating_ip_nat_rules_remove(self):
|
||||||
@ -494,13 +502,14 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
agent.process_router_floating_ip_nat_rules(ri)
|
agent.process_router_floating_ip_nat_rules(ri)
|
||||||
|
|
||||||
nat = ri.iptables_manager.ipv4['nat']
|
nat = ri.iptables_manager.ipv4['nat']
|
||||||
nat = ri.iptables_manager.ipv4['nat']
|
nat = ri.iptables_manager.ipv4['nat`']
|
||||||
nat.clear_rules_by_tag.assert_called_once_with('floating_ip')
|
nat.clear_rules_by_tag.assert_called_once_with('floating_ip')
|
||||||
|
|
||||||
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
||||||
def test_process_router_floating_ip_addresses_remap(self, IPDevice):
|
def test_process_router_floating_ip_addresses_remap(self, IPDevice):
|
||||||
|
fip_id = _uuid()
|
||||||
fip = {
|
fip = {
|
||||||
'id': _uuid(), 'port_id': _uuid(),
|
'id': fip_id, 'port_id': _uuid(),
|
||||||
'floating_ip_address': '15.1.2.3',
|
'floating_ip_address': '15.1.2.3',
|
||||||
'fixed_ip_address': '192.168.0.2'
|
'fixed_ip_address': '192.168.0.2'
|
||||||
}
|
}
|
||||||
@ -513,31 +522,55 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
|
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
|
||||||
agent.process_router_floating_ip_addresses(ri, {'id': _uuid()})
|
fip_statuses = agent.process_router_floating_ip_addresses(
|
||||||
|
ri, {'id': _uuid()})
|
||||||
|
self.assertEqual({fip_id: l3_constants.FLOATINGIP_STATUS_ACTIVE},
|
||||||
|
fip_statuses)
|
||||||
|
|
||||||
self.assertFalse(device.addr.add.called)
|
self.assertFalse(device.addr.add.called)
|
||||||
self.assertFalse(device.addr.delete.called)
|
self.assertFalse(device.addr.delete.called)
|
||||||
|
|
||||||
def test_process_router_floating_ip_nat_rules_remap(self):
|
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
||||||
|
def test_process_router_with_disabled_floating_ip(self, IPDevice):
|
||||||
|
fip_id = _uuid()
|
||||||
fip = {
|
fip = {
|
||||||
'id': _uuid(), 'port_id': _uuid(),
|
'id': fip_id, 'port_id': _uuid(),
|
||||||
'floating_ip_address': '15.1.2.3',
|
'floating_ip_address': '15.1.2.3',
|
||||||
'fixed_ip_address': '192.168.0.2'
|
'fixed_ip_address': '192.168.0.2'
|
||||||
}
|
}
|
||||||
|
|
||||||
ri = mock.MagicMock()
|
ri = mock.MagicMock()
|
||||||
|
ri.floating_ips = [fip]
|
||||||
|
ri.router.get.return_value = []
|
||||||
|
|
||||||
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
|
||||||
|
fip_statuses = agent.process_router_floating_ip_addresses(
|
||||||
|
ri, {'id': _uuid()})
|
||||||
|
|
||||||
|
self.assertIsNone(fip_statuses.get(fip_id))
|
||||||
|
|
||||||
|
@mock.patch('neutron.agent.linux.ip_lib.IPDevice')
|
||||||
|
def test_process_router_floating_ip_with_device_add_error(self, IPDevice):
|
||||||
|
IPDevice.return_value = device = mock.Mock()
|
||||||
|
device.addr.add.side_effect = processutils.ProcessExecutionError
|
||||||
|
device.addr.list.return_value = []
|
||||||
|
fip_id = _uuid()
|
||||||
|
fip = {
|
||||||
|
'id': fip_id, 'port_id': _uuid(),
|
||||||
|
'floating_ip_address': '15.1.2.3',
|
||||||
|
'fixed_ip_address': '192.168.0.2'
|
||||||
|
}
|
||||||
|
ri = mock.MagicMock()
|
||||||
ri.router.get.return_value = [fip]
|
ri.router.get.return_value = [fip]
|
||||||
|
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
|
||||||
agent.process_router_floating_ip_nat_rules(ri)
|
fip_statuses = agent.process_router_floating_ip_addresses(
|
||||||
|
ri, {'id': _uuid()})
|
||||||
|
|
||||||
nat = ri.iptables_manager.ipv4['nat']
|
self.assertEqual({fip_id: l3_constants.FLOATINGIP_STATUS_ERROR},
|
||||||
nat.clear_rules_by_tag.assert_called_once_with('floating_ip')
|
fip_statuses)
|
||||||
rules = agent.floating_forward_rules('15.1.2.3', '192.168.0.2')
|
|
||||||
for chain, rule in rules:
|
|
||||||
nat.add_rule.assert_any_call(chain, rule, tag='floating_ip')
|
|
||||||
|
|
||||||
def test_process_router_snat_disabled(self):
|
def test_process_router_snat_disabled(self):
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
@ -686,6 +719,36 @@ class TestBasicRouterOperations(base.BaseTestCase):
|
|||||||
self.assertNotIn(
|
self.assertNotIn(
|
||||||
router[l3_constants.INTERFACE_KEY][0], ri.internal_ports)
|
router[l3_constants.INTERFACE_KEY][0], ri.internal_ports)
|
||||||
|
|
||||||
|
def test_process_router_floatingip_disabled(self):
|
||||||
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
|
with mock.patch.object(
|
||||||
|
agent.plugin_rpc,
|
||||||
|
'update_floatingip_statuses') as mock_update_fip_status:
|
||||||
|
fip_id = _uuid()
|
||||||
|
router = self._prepare_router_data(num_internal_ports=1)
|
||||||
|
router[l3_constants.FLOATINGIP_KEY] = [
|
||||||
|
{'id': fip_id,
|
||||||
|
'floating_ip_address': '8.8.8.8',
|
||||||
|
'fixed_ip_address': '7.7.7.7',
|
||||||
|
'port_id': router[l3_constants.INTERFACE_KEY][0]['id']}]
|
||||||
|
|
||||||
|
ri = l3_agent.RouterInfo(router['id'], self.conf.root_helper,
|
||||||
|
self.conf.use_namespaces, router=router)
|
||||||
|
agent.process_router(ri)
|
||||||
|
# Assess the call for putting the floating IP up was performed
|
||||||
|
mock_update_fip_status.assert_called_once_with(
|
||||||
|
mock.ANY, ri.router_id,
|
||||||
|
{fip_id: l3_constants.FLOATINGIP_STATUS_ACTIVE})
|
||||||
|
mock_update_fip_status.reset_mock()
|
||||||
|
# Process the router again, this time without floating IPs
|
||||||
|
router[l3_constants.FLOATINGIP_KEY] = []
|
||||||
|
ri.router = router
|
||||||
|
agent.process_router(ri)
|
||||||
|
# Assess the call for putting the floating IP up was performed
|
||||||
|
mock_update_fip_status.assert_called_once_with(
|
||||||
|
mock.ANY, ri.router_id,
|
||||||
|
{fip_id: l3_constants.FLOATINGIP_STATUS_DOWN})
|
||||||
|
|
||||||
def test_handle_router_snat_rules_add_back_jump(self):
|
def test_handle_router_snat_rules_add_back_jump(self):
|
||||||
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
agent = l3_agent.L3NATAgent(HOSTNAME, self.conf)
|
||||||
ri = mock.MagicMock()
|
ri = mock.MagicMock()
|
||||||
|
@ -1195,7 +1195,8 @@ class L3NatTestCaseBase(L3NatTestCaseMixin):
|
|||||||
private_sub['subnet']['id'],
|
private_sub['subnet']['id'],
|
||||||
None)
|
None)
|
||||||
|
|
||||||
def test_floatingip_update(self):
|
def test_floatingip_update(
|
||||||
|
self, expected_status=l3_constants.FLOATINGIP_STATUS_ACTIVE):
|
||||||
with self.port() as p:
|
with self.port() as p:
|
||||||
private_sub = {'subnet': {'id':
|
private_sub = {'subnet': {'id':
|
||||||
p['port']['fixed_ips'][0]['subnet_id']}}
|
p['port']['fixed_ips'][0]['subnet_id']}}
|
||||||
@ -1203,6 +1204,7 @@ class L3NatTestCaseBase(L3NatTestCaseMixin):
|
|||||||
body = self._show('floatingips', fip['floatingip']['id'])
|
body = self._show('floatingips', fip['floatingip']['id'])
|
||||||
self.assertIsNone(body['floatingip']['port_id'])
|
self.assertIsNone(body['floatingip']['port_id'])
|
||||||
self.assertIsNone(body['floatingip']['fixed_ip_address'])
|
self.assertIsNone(body['floatingip']['fixed_ip_address'])
|
||||||
|
self.assertEqual(body['floatingip']['status'], expected_status)
|
||||||
|
|
||||||
port_id = p['port']['id']
|
port_id = p['port']['id']
|
||||||
ip_address = p['port']['fixed_ips'][0]['ip_address']
|
ip_address = p['port']['fixed_ips'][0]['ip_address']
|
||||||
|
Loading…
Reference in New Issue
Block a user