Implement L3 support in Metaplugin

Added flavor:route property for router
Rename flavor:id to flavor:network
Fixes bug 1038778

Change-Id: Ia358b4f03c1b96ade2d1b7323298b117b2cbe52a
This commit is contained in:
Nachi Ueno 2012-08-20 05:00:14 +00:00
parent efa4316f64
commit 34d3e21a40
14 changed files with 282 additions and 76 deletions

View File

@ -16,9 +16,11 @@ reconnect_interval = 2
## This is list of flavor:quantum_plugins ## This is list of flavor:quantum_plugins
# extension method is used in the order of this list # extension method is used in the order of this list
plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2' plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2'
l3_plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2'
# Default value of flavor # Default value of flavor
default_flavor = 'openvswitch' default_flavor = 'openvswitch'
default_l3_flavor = 'openvswitch'
# supported extentions # supported extentions
supported_extension_aliases = 'providernet' supported_extension_aliases = 'providernet'

View File

@ -219,7 +219,8 @@ class L3NATAgent(object):
for p in new_ports: for p in new_ports:
self._set_subnet_info(p) self._set_subnet_info(p)
ri.internal_ports.append(p) ri.internal_ports.append(p)
self.internal_network_added(ri, ex_gw_port, p['id'], self.internal_network_added(ri, ex_gw_port,
p['network_id'], p['id'],
p['ip_cidr'], p['mac_address']) p['ip_cidr'], p['mac_address'])
for p in old_ports: for p in old_ports:
@ -305,7 +306,8 @@ class L3NATAgent(object):
if not ip_lib.device_exists(interface_name, if not ip_lib.device_exists(interface_name,
root_helper=self.conf.root_helper, root_helper=self.conf.root_helper,
namespace=ri.ns_name()): namespace=ri.ns_name()):
self.driver.plug(None, ex_gw_port['id'], interface_name, self.driver.plug(ex_gw_port['network_id'],
ex_gw_port['id'], interface_name,
ex_gw_port['mac_address'], ex_gw_port['mac_address'],
bridge=self.conf.external_network_bridge, bridge=self.conf.external_network_bridge,
namespace=ri.ns_name(), namespace=ri.ns_name(),
@ -367,13 +369,13 @@ class L3NATAgent(object):
rules.extend(self.internal_network_nat_rules(ex_gw_ip, cidr)) rules.extend(self.internal_network_nat_rules(ex_gw_ip, cidr))
return rules return rules
def internal_network_added(self, ri, ex_gw_port, port_id, def internal_network_added(self, ri, ex_gw_port, network_id, port_id,
internal_cidr, mac_address): internal_cidr, mac_address):
interface_name = self.get_internal_device_name(port_id) interface_name = self.get_internal_device_name(port_id)
if not ip_lib.device_exists(interface_name, if not ip_lib.device_exists(interface_name,
root_helper=self.conf.root_helper, root_helper=self.conf.root_helper,
namespace=ri.ns_name()): namespace=ri.ns_name()):
self.driver.plug(None, port_id, interface_name, mac_address, self.driver.plug(network_id, port_id, interface_name, mac_address,
namespace=ri.ns_name(), namespace=ri.ns_name(),
prefix=INTERNAL_DEV_PREFIX) prefix=INTERNAL_DEV_PREFIX)

View File

@ -24,6 +24,7 @@ from quantum.agent.linux import ip_lib
from quantum.agent.linux import ovs_lib from quantum.agent.linux import ovs_lib
from quantum.agent.linux import utils from quantum.agent.linux import utils
from quantum.common import exceptions from quantum.common import exceptions
from quantum.extensions.flavor import (FLAVOR_NETWORK)
from quantum.openstack.common import cfg from quantum.openstack.common import cfg
from quantum.openstack.common import importutils from quantum.openstack.common import importutils
@ -244,7 +245,7 @@ class MetaInterfaceDriver(LinuxInterfaceDriver):
def _get_driver_by_network_id(self, network_id): def _get_driver_by_network_id(self, network_id):
network = self.quantum.show_network(network_id) network = self.quantum.show_network(network_id)
flavor = network['network']['flavor:id'] flavor = network['network'][FLAVOR_NETWORK]
return self.flavor_driver_map[flavor] return self.flavor_driver_map[flavor]
def _get_driver_by_device_name(self, device_name, namespace=None): def _get_driver_by_device_name(self, device_name, namespace=None):

View File

@ -21,9 +21,18 @@ from quantum.api.v2 import attributes
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
FLAVOR_NETWORK = 'flavor:network'
FLAVOR_ROUTER = 'flavor:router'
FLAVOR_ATTRIBUTE = { FLAVOR_ATTRIBUTE = {
'networks': { 'networks': {
'flavor:id': {'allow_post': True, FLAVOR_NETWORK: {'allow_post': True,
'allow_put': False,
'is_visible': True,
'default': attributes.ATTR_NOT_SPECIFIED}
},
'routers': {
FLAVOR_ROUTER: {'allow_post': True,
'allow_put': False, 'allow_put': False,
'is_visible': True, 'is_visible': True,
'default': attributes.ATTR_NOT_SPECIFIED} 'default': attributes.ATTR_NOT_SPECIFIED}
@ -34,7 +43,7 @@ FLAVOR_ATTRIBUTE = {
class Flavor(object): class Flavor(object):
@classmethod @classmethod
def get_name(cls): def get_name(cls):
return "Flavor for each network" return "Flavor support for network and router"
@classmethod @classmethod
def get_alias(cls): def get_alias(cls):

View File

@ -1,8 +1,9 @@
# -- Background # -- Background
This plugin support multiple plugin at same time. This plugin is for L3 connectivility This plugin supports multiple plugin at same time. This plugin is for L3 connectivility
between networks which are realized by different plugins.This plugin add new attribute 'flavor:id'. between networks which are realized by different plugins.This plugin adds new attributes 'flavor:network' and 'flavor:router".
flavor:id correspond to specific plugin ( flavor-plugin mapping could be configureable by plugin_list config. flavor:network corresponds to specific l2 plugin ( flavor-plugin mapping could be configureable by plugin_list config.
flavor:router corresponds to specific l3 plugin ( flavor-plugin mapping could be configureable by l3_plugin_list config. Note that Metaplugin can provide l3 functionaliteis for l2 plugin which didn't support l3 extension yet.
This plugin also support extensions. We can map extension to plugin by using extension_map config. This plugin also support extensions. We can map extension to plugin by using extension_map config.
[DATABASE] [DATABASE]
@ -23,9 +24,13 @@ reconnect_interval = 2
## This is list of flavor:quantum_plugins ## This is list of flavor:quantum_plugins
# extension method is used in the order of this list # extension method is used in the order of this list
plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2' plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2'
# plugin for l3
l3_plugin_list= 'openvswitch:quantum.plugins.openvswitch.ovs_quantum_plugin.OVSQuantumPluginV2,linuxbridge:quantum.plugins.linuxbridge.lb_quantum_plugin.LinuxBridgePluginV2'
# Default value of flavor # Default value of flavor
default_flavor = 'openvswitch' default_flavor = 'openvswitch'
# Default value for l3
default_l3_flavor = 'openvswitch'
# supported extentions # supported extentions
supported_extension_aliases = 'providernet' supported_extension_aliases = 'providernet'
@ -42,6 +47,14 @@ meta_flavor_driver_mappings = openvswitch:quantum.agent.linux.interface.OVSInter
# interface driver for MetaPlugin # interface driver for MetaPlugin
interface_driver = quantum.agent.linux.interface.MetaInterfaceDriver interface_driver = quantum.agent.linux.interface.MetaInterfaceDriver
[Proxy]
auth_url = http://10.0.0.1:35357/v2.0
auth_region = RegionOne
admin_tenant_name = service
admin_user = quantum
admin_password = password
# -- Agent # -- Agent
Agents for Metaplugin are in quantum/plugins/metaplugin/agent Agents for Metaplugin are in quantum/plugins/metaplugin/agent
linuxbridge_quantum_agent and ovs_quantum_agent is available. linuxbridge_quantum_agent and ovs_quantum_agent is available.
@ -70,12 +83,10 @@ Example flavor configration for ProxyPluginV2
meta_flavor_driver_mappings = "openvswitch:quantum.agent.linux.interface.OVSInterfaceDriver,proxy:quantum.plugins.metaplugin.proxy_quantum_plugin.ProxyPluginV2" meta_flavor_driver_mappings = "openvswitch:quantum.agent.linux.interface.OVSInterfaceDriver,proxy:quantum.plugins.metaplugin.proxy_quantum_plugin.ProxyPluginV2"
[Proxy] - Limited L3 support
auth_url = http://10.0.0.1:35357/v2.0 In folsom version, l3 is an extension. There is no way to extend exntension attributes.
auth_region = RegionOne so you can set flavor:router value but you can't get flavor:router value in API output.
admin_tenant_name = service L3 agent dont's support flavor:router.
admin_user = quantum
admin_password = password

View File

@ -26,7 +26,9 @@ database_opts = [
meta_plugin_opts = [ meta_plugin_opts = [
cfg.StrOpt('plugin_list', default=''), cfg.StrOpt('plugin_list', default=''),
cfg.StrOpt('l3_plugin_list', default=''),
cfg.StrOpt('default_flavor', default=''), cfg.StrOpt('default_flavor', default=''),
cfg.StrOpt('default_l3_flavor', default=''),
cfg.StrOpt('supported_extension_aliases', default=''), cfg.StrOpt('supported_extension_aliases', default=''),
cfg.StrOpt('extension_map', default='') cfg.StrOpt('extension_map', default='')
] ]

View File

@ -24,7 +24,7 @@ from quantum.plugins.metaplugin import meta_models_v2
def get_flavor_by_network(net_id): def get_flavor_by_network(net_id):
session = db.get_session() session = db.get_session()
try: try:
binding = (session.query(meta_models_v2.Flavor). binding = (session.query(meta_models_v2.NetworkFlavor).
filter_by(network_id=net_id). filter_by(network_id=net_id).
one()) one())
except exc.NoResultFound: except exc.NoResultFound:
@ -32,9 +32,28 @@ def get_flavor_by_network(net_id):
return binding.flavor return binding.flavor
def add_flavor_binding(flavor, net_id): def add_network_flavor_binding(flavor, net_id):
session = db.get_session() session = db.get_session()
binding = meta_models_v2.Flavor(flavor=flavor, network_id=net_id) binding = meta_models_v2.NetworkFlavor(flavor=flavor, network_id=net_id)
session.add(binding)
session.flush()
return binding
def get_flavor_by_router(router_id):
session = db.get_session()
try:
binding = (session.query(meta_models_v2.RouterFlavor).
filter_by(router_id=router_id).
one())
except exc.NoResultFound:
return None
return binding.flavor
def add_router_flavor_binding(flavor, router_id):
session = db.get_session()
binding = meta_models_v2.RouterFlavor(flavor=flavor, router_id=router_id)
session.add(binding) session.add(binding)
session.flush() session.flush()
return binding return binding

View File

@ -21,7 +21,7 @@ from sqlalchemy import Column, String
from quantum.db import models_v2 from quantum.db import models_v2
class Flavor(models_v2.model_base.BASEV2): class NetworkFlavor(models_v2.model_base.BASEV2):
"""Represents a binding of network_id to flavor.""" """Represents a binding of network_id to flavor."""
flavor = Column(String(255)) flavor = Column(String(255))
network_id = sa.Column(sa.String(36), sa.ForeignKey('networks.id', network_id = sa.Column(sa.String(36), sa.ForeignKey('networks.id',
@ -29,4 +29,15 @@ class Flavor(models_v2.model_base.BASEV2):
primary_key=True) primary_key=True)
def __repr__(self): def __repr__(self):
return "<Flavor(%s,%s)>" % (self.flavor, self.network_id) return "<NetworkFlavor(%s,%s)>" % (self.flavor, self.network_id)
class RouterFlavor(models_v2.model_base.BASEV2):
"""Represents a binding of router_id to flavor."""
flavor = Column(String(255))
router_id = sa.Column(sa.String(36), sa.ForeignKey('routers.id',
ondelete="CASCADE"),
primary_key=True)
def __repr__(self):
return "<RouterFlavor(%s,%s)>" % (self.flavor, self.router_id)

View File

@ -17,24 +17,37 @@
import logging import logging
from quantum.common import exceptions as exc
from quantum.api.v2 import attributes from quantum.api.v2 import attributes
from quantum.common import exceptions as exc
from quantum.common.utils import find_config_file from quantum.common.utils import find_config_file
from quantum.db import api as db from quantum.db import api as db
from quantum.db import db_base_plugin_v2 from quantum.db import db_base_plugin_v2
from quantum.db import l3_db
from quantum.db import models_v2 from quantum.db import models_v2
from quantum.extensions.flavor import (FLAVOR_NETWORK, FLAVOR_ROUTER)
from quantum.openstack.common import cfg from quantum.openstack.common import cfg
from quantum.openstack.common import importutils from quantum.openstack.common import importutils
from quantum.plugins.metaplugin.common import config from quantum.plugins.metaplugin.common import config
from quantum.plugins.metaplugin import meta_db_v2 from quantum.plugins.metaplugin import meta_db_v2
from quantum.plugins.metaplugin.meta_models_v2 import Flavor from quantum.plugins.metaplugin.meta_models_v2 import (NetworkFlavor,
RouterFlavor)
from quantum import policy from quantum import policy
LOG = logging.getLogger("metaplugin") LOG = logging.getLogger("metaplugin")
class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2): # Metaplugin Exceptions
class FlavorNotFound(exc.NotFound):
message = _("Flavor %(flavor)s could not be found")
class FaildToAddFlavorBinding(exc.QuantumException):
message = _("Failed to add flavor binding")
class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2,
l3_db.L3_NAT_db_mixin):
def __init__(self, configfile=None): def __init__(self, configfile=None):
LOG.debug("Start initializing metaplugin") LOG.debug("Start initializing metaplugin")
options = {"sql_connection": cfg.CONF.DATABASE.sql_connection} options = {"sql_connection": cfg.CONF.DATABASE.sql_connection}
@ -45,7 +58,7 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
options.update({"reconnect_interval": reconnect_interval}) options.update({"reconnect_interval": reconnect_interval})
self.supported_extension_aliases = \ self.supported_extension_aliases = \
cfg.CONF.META.supported_extension_aliases.split(',') cfg.CONF.META.supported_extension_aliases.split(',')
self.supported_extension_aliases.append('flavor') self.supported_extension_aliases += ['flavor', 'os-quantum-router']
# Ignore config option overapping # Ignore config option overapping
def _is_opt_registered(opts, opt): def _is_opt_registered(opts, opt):
@ -69,7 +82,30 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
# Needed to clear _ENGINE for each plugin # Needed to clear _ENGINE for each plugin
db._ENGINE = None db._ENGINE = None
self.l3_plugins = {}
l3_plugin_list = [plugin_set.split(':')
for plugin_set
in cfg.CONF.META.l3_plugin_list.split(',')]
for flavor, plugin_provider in l3_plugin_list:
if flavor in self.plugins:
self.l3_plugins[flavor] = self.plugins[flavor]
else:
# For l3 only plugin
self.l3_plugins[flavor] = self._load_plugin(plugin_provider)
db._ENGINE = None
self.default_flavor = cfg.CONF.META.default_flavor
if not self.default_flavor in self.plugins:
raise exc.Invalid('default_flavor %s is not plugin list' %
self.default_flavor)
self.default_l3_flavor = cfg.CONF.META.default_l3_flavor
if not self.default_l3_flavor in self.l3_plugins:
raise exc.Invalid('default_l3_flavor %s is not plugin list' %
self.default_l3_flavor)
db.configure_db(options) db.configure_db(options)
self.extension_map = {} self.extension_map = {}
if not cfg.CONF.META.extension_map == '': if not cfg.CONF.META.extension_map == '':
extension_list = [method_set.split(':') extension_list = [method_set.split(':')
@ -80,28 +116,21 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
self.default_flavor = cfg.CONF.META.default_flavor self.default_flavor = cfg.CONF.META.default_flavor
if not self.default_flavor in self.plugins:
raise exc.Invalid('default_flavor %s is not plugin list' %
self.default_flavor)
def _load_plugin(self, plugin_provider): def _load_plugin(self, plugin_provider):
LOG.debug("Plugin location:%s", plugin_provider) LOG.debug("Plugin location:%s", plugin_provider)
# If the plugin can't be found let them know gracefully
try:
LOG.info("Loading Plugin: %s" % plugin_provider)
plugin_klass = importutils.import_class(plugin_provider) plugin_klass = importutils.import_class(plugin_provider)
except exc.ClassNotFound:
LOG.exception("Error loading plugin")
raise Exception("Plugin not found. You can install a "
"plugin with: pip install <plugin-name>\n"
"Example: pip install quantum-sample-plugin")
return plugin_klass() return plugin_klass()
def _get_plugin(self, flavor): def _get_plugin(self, flavor):
if not flavor in self.plugins: if not flavor in self.plugins:
raise Exception("Plugin for flavor %s not found." % flavor) raise FlavorNotFound(flavor=flavor)
return self.plugins[flavor] return self.plugins[flavor]
def _get_l3_plugin(self, flavor):
if not flavor in self.l3_plugins:
raise FlavorNotFound(flavor=flavor)
return self.l3_plugins[flavor]
def __getattr__(self, key): def __getattr__(self, key):
# At first, try to pickup extension command from extension_map # At first, try to pickup extension command from extension_map
@ -111,8 +140,7 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
if plugin and hasattr(plugin, key): if plugin and hasattr(plugin, key):
return getattr(plugin, key) return getattr(plugin, key)
# Second, try to match extension method in order of pluign list # Second, try to match extension method in order of plugin list
for flavor, plugin in self.plugins.items(): for flavor, plugin in self.plugins.items():
if hasattr(plugin, key): if hasattr(plugin, key):
return getattr(plugin, key) return getattr(plugin, key)
@ -121,22 +149,23 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
raise AttributeError raise AttributeError
def _extend_network_dict(self, context, network): def _extend_network_dict(self, context, network):
network['flavor:id'] = self._get_flavor_by_network_id(network['id']) flavor = self._get_flavor_by_network_id(network['id'])
network[FLAVOR_NETWORK] = flavor
def create_network(self, context, network): def create_network(self, context, network):
n = network['network'] n = network['network']
flavor = n.get('flavor:id') flavor = n.get(FLAVOR_NETWORK)
if not str(flavor) in self.plugins: if not str(flavor) in self.plugins:
flavor = self.default_flavor flavor = self.default_flavor
plugin = self._get_plugin(flavor) plugin = self._get_plugin(flavor)
net = plugin.create_network(context, network) net = plugin.create_network(context, network)
LOG.debug("Created network: %s with flavor %s " % (net['id'], flavor)) LOG.debug("Created network: %s with flavor %s " % (net['id'], flavor))
try: try:
meta_db_v2.add_flavor_binding(flavor, str(net['id'])) meta_db_v2.add_network_flavor_binding(flavor, str(net['id']))
except Exception as e: except:
LOG.error('failed to add flavor bindings') LOG.exception('failed to add flavor bindings')
plugin.delete_network(context, net['id']) plugin.delete_network(context, net['id'])
raise Exception('Failed to create network') raise FaildToAddFlavorBinding()
LOG.debug("Created network: %s" % net['id']) LOG.debug("Created network: %s" % net['id'])
self._extend_network_dict(context, net) self._extend_network_dict(context, net)
@ -151,19 +180,20 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
flavor = meta_db_v2.get_flavor_by_network(id) flavor = meta_db_v2.get_flavor_by_network(id)
plugin = self._get_plugin(flavor) plugin = self._get_plugin(flavor)
net = plugin.get_network(context, id, fields) net = plugin.get_network(context, id, fields)
if not fields or 'flavor:id' in fields: if not fields or FLAVOR_NETWORK in fields:
self._extend_network_dict(context, net) self._extend_network_dict(context, net)
return net return net
def get_networks_with_flavor(self, context, filters=None, def get_networks_with_flavor(self, context, filters=None,
fields=None): fields=None):
collection = self._model_query(context, models_v2.Network) collection = self._model_query(context, models_v2.Network)
collection = collection.join(Flavor, model = NetworkFlavor
models_v2.Network.id == Flavor.network_id) collection = collection.join(model,
models_v2.Network.id == model.network_id)
if filters: if filters:
for key, value in filters.iteritems(): for key, value in filters.iteritems():
if key == 'flavor:id': if key == FLAVOR_NETWORK:
column = Flavor.flavor column = NetworkFlavor.flavor
else: else:
column = getattr(models_v2.Network, key, None) column = getattr(models_v2.Network, key, None)
if column: if column:
@ -178,6 +208,9 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
def _get_flavor_by_network_id(self, network_id): def _get_flavor_by_network_id(self, network_id):
return meta_db_v2.get_flavor_by_network(network_id) return meta_db_v2.get_flavor_by_network(network_id)
def _get_flavor_by_router_id(self, router_id):
return meta_db_v2.get_flavor_by_router(router_id)
def _get_plugin_by_network_id(self, network_id): def _get_plugin_by_network_id(self, network_id):
flavor = self._get_flavor_by_network_id(network_id) flavor = self._get_flavor_by_network_id(network_id)
return self._get_plugin(flavor) return self._get_plugin(flavor)
@ -194,7 +227,10 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
plugin = self._get_plugin_by_network_id(port_in_db['network_id']) plugin = self._get_plugin_by_network_id(port_in_db['network_id'])
return plugin.update_port(context, id, port) return plugin.update_port(context, id, port)
def delete_port(self, context, id): def delete_port(self, context, id, l3_port_check=True):
if l3_port_check:
self.prevent_l3_port_deletion(context, id)
self.disassociate_floatingips(context, id)
port_in_db = self.get_port(context, id) port_in_db = self.get_port(context, id)
plugin = self._get_plugin_by_network_id(port_in_db['network_id']) plugin = self._get_plugin_by_network_id(port_in_db['network_id'])
return plugin.delete_port(context, id) return plugin.delete_port(context, id)
@ -215,3 +251,68 @@ class MetaPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
s = self.get_subnet(context, id) s = self.get_subnet(context, id)
plugin = self._get_plugin_by_network_id(s['network_id']) plugin = self._get_plugin_by_network_id(s['network_id'])
return plugin.delete_subnet(context, id) return plugin.delete_subnet(context, id)
def _extend_router_dict(self, context, router):
flavor = self._get_flavor_by_router_id(router['id'])
router[FLAVOR_ROUTER] = flavor
def create_router(self, context, router):
r = router['router']
flavor = r.get(FLAVOR_ROUTER)
if not str(flavor) in self.l3_plugins:
flavor = self.default_l3_flavor
plugin = self._get_l3_plugin(flavor)
r_in_db = plugin.create_router(context, router)
LOG.debug("Created router: %s with flavor %s " % (r_in_db['id'],
flavor))
try:
meta_db_v2.add_router_flavor_binding(flavor, str(r_in_db['id']))
except:
LOG.exception('failed to add flavor bindings')
plugin.delete_router(context, r_in_db['id'])
raise FaildToAddFlavorBinding()
LOG.debug("Created router: %s" % r_in_db['id'])
self._extend_router_dict(context, r_in_db)
return r_in_db
def update_router(self, context, id, router):
flavor = meta_db_v2.get_flavor_by_router(id)
plugin = self._get_l3_plugin(flavor)
return plugin.update_router(context, id, router)
def delete_router(self, context, id):
flavor = meta_db_v2.get_flavor_by_router(id)
plugin = self._get_l3_plugin(flavor)
return plugin.delete_router(context, id)
def get_router(self, context, id, fields=None):
flavor = meta_db_v2.get_flavor_by_router(id)
plugin = self._get_l3_plugin(flavor)
router = plugin.get_router(context, id, fields)
if not fields or FLAVOR_ROUTER in fields:
self._extend_router_dict(context, router)
return router
def get_routers_with_flavor(self, context, filters=None,
fields=None):
collection = self._model_query(context, l3_db.Router)
r_model = RouterFlavor
collection = collection.join(r_model,
l3_db.Router.id == r_model.router_id)
if filters:
for key, value in filters.iteritems():
if key == FLAVOR_ROUTER:
column = RouterFlavor.flavor
else:
column = getattr(l3_db.Router, key, None)
if column:
collection = collection.filter(column.in_(value))
return [self._make_router_dict(c, fields) for c in collection.all()]
def get_routers(self, context, filters=None, fields=None):
routers = self.get_routers_with_flavor(context, filters,
None)
return [self.get_router(context, router['id'],
fields)
for router in routers]

View File

@ -19,6 +19,7 @@ import logging
from quantum.db import api as db from quantum.db import api as db
from quantum.db import db_base_plugin_v2 from quantum.db import db_base_plugin_v2
from quantum.db import l3_db
from quantum.db import models_v2 from quantum.db import models_v2
from quantum.openstack.common import cfg from quantum.openstack.common import cfg
from quantumclient.common import exceptions from quantumclient.common import exceptions
@ -28,7 +29,8 @@ from quantumclient.v2_0 import client
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
class ProxyPluginV2(db_base_plugin_v2.QuantumDbPluginV2): class ProxyPluginV2(db_base_plugin_v2.QuantumDbPluginV2,
l3_db.L3_NAT_db_mixin):
def __init__(self, configfile=None): def __init__(self, configfile=None):
options = {"sql_connection": cfg.CONF.DATABASE.sql_connection} options = {"sql_connection": cfg.CONF.DATABASE.sql_connection}
options.update({'base': models_v2.model_base.BASEV2}) options.update({'base': models_v2.model_base.BASEV2})
@ -126,7 +128,11 @@ class ProxyPluginV2(db_base_plugin_v2.QuantumDbPluginV2):
LOG.error("update port failed: %e" % e) LOG.error("update port failed: %e" % e)
return port_in_db return port_in_db
def delete_port(self, context, id): def delete_port(self, context, id, l3_port_check=True):
if l3_port_check:
self.prevent_l3_port_deletion(context, id)
self.disassociate_floatingips(context, id)
try: try:
self._get_client().delete_port(id) self._get_client().delete_port(id)
except exceptions.PortNotFoundClient: except exceptions.PortNotFoundClient:

View File

@ -18,10 +18,12 @@ from quantum.common import exceptions as q_exc
from quantum.common.utils import find_config_file from quantum.common.utils import find_config_file
from quantum.db import api as db from quantum.db import api as db
from quantum.db import db_base_plugin_v2 from quantum.db import db_base_plugin_v2
from quantum.db import l3_db
from quantum.db import models_v2 from quantum.db import models_v2
class Fake1(db_base_plugin_v2.QuantumDbPluginV2): class Fake1(db_base_plugin_v2.QuantumDbPluginV2,
l3_db.L3_NAT_db_mixin):
def fake_func(self): def fake_func(self):
return 'fake1' return 'fake1'
@ -45,7 +47,7 @@ class Fake1(db_base_plugin_v2.QuantumDbPluginV2):
port = super(Fake1, self).update_port(context, id, port) port = super(Fake1, self).update_port(context, id, port)
return port return port
def delete_port(self, context, id): def delete_port(self, context, id, l3_port_check=True):
return super(Fake1, self).delete_port(context, id) return super(Fake1, self).delete_port(context, id)

View File

@ -16,17 +16,19 @@
# under the License. # under the License.
import os import os
import mox
import mock
import uuid import uuid
import unittest
import mock
import mox
import stubout import stubout
import unittest2 as unittest
from quantum.common import config from quantum.common import config
from quantum.common.exceptions import NotImplementedError from quantum.common.exceptions import NotImplementedError
from quantum.db import api as db from quantum.db import api as db
from quantum.db import models_v2 from quantum.db import models_v2
from quantum.extensions.flavor import (FLAVOR_NETWORK, FLAVOR_ROUTER)
from quantum.extensions import l3
from quantum.openstack.common import cfg from quantum.openstack.common import cfg
from quantum.plugins.metaplugin.meta_quantum_plugin import MetaPluginV2 from quantum.plugins.metaplugin.meta_quantum_plugin import MetaPluginV2
from quantum.plugins.metaplugin.proxy_quantum_plugin import ProxyPluginV2 from quantum.plugins.metaplugin.proxy_quantum_plugin import ProxyPluginV2
@ -39,9 +41,12 @@ ETCDIR = os.path.join(ROOTDIR, 'etc')
META_PATH = "quantum.plugins.metaplugin" META_PATH = "quantum.plugins.metaplugin"
FAKE_PATH = "quantum.tests.unit.metaplugin" FAKE_PATH = "quantum.tests.unit.metaplugin"
PROXY_PATH = "%s.proxy_quantum_plugin.ProxyPluginV2" % META_PATH PROXY_PATH = "%s.proxy_quantum_plugin.ProxyPluginV2" % META_PATH
PLUGIN_LIST = \ PLUGIN_LIST = """
'fake1:%s.fake_plugin.Fake1,fake2:%s.fake_plugin.Fake2,proxy:%s' % \ fake1:%s.fake_plugin.Fake1,fake2:%s.fake_plugin.Fake2,proxy:%s
(FAKE_PATH, FAKE_PATH, PROXY_PATH) """.strip() % (FAKE_PATH, FAKE_PATH, PROXY_PATH)
L3_PLUGIN_LIST = """
fake1:%s.fake_plugin.Fake1,fake2:%s.fake_plugin.Fake2
""".strip() % (FAKE_PATH, FAKE_PATH)
def etcdir(*p): def etcdir(*p):
@ -75,7 +80,9 @@ class MetaQuantumPluginV2Test(unittest.TestCase):
cfg.CONF.set_override('admin_password', 'password', 'PROXY') cfg.CONF.set_override('admin_password', 'password', 'PROXY')
cfg.CONF.set_override('admin_tenant_name', 'service', 'PROXY') cfg.CONF.set_override('admin_tenant_name', 'service', 'PROXY')
cfg.CONF.set_override('plugin_list', PLUGIN_LIST, 'META') cfg.CONF.set_override('plugin_list', PLUGIN_LIST, 'META')
cfg.CONF.set_override('l3_plugin_list', L3_PLUGIN_LIST, 'META')
cfg.CONF.set_override('default_flavor', 'fake2', 'META') cfg.CONF.set_override('default_flavor', 'fake2', 'META')
cfg.CONF.set_override('default_l3_flavor', 'fake1', 'META')
cfg.CONF.set_override('base_mac', "12:34:56:78:90:ab") cfg.CONF.set_override('base_mac', "12:34:56:78:90:ab")
#TODO(nati) remove this after subnet quota change is merged #TODO(nati) remove this after subnet quota change is merged
cfg.CONF.max_dns_nameservers = 10 cfg.CONF.max_dns_nameservers = 10
@ -105,7 +112,7 @@ class MetaQuantumPluginV2Test(unittest.TestCase):
'admin_state_up': True, 'admin_state_up': True,
'shared': False, 'shared': False,
'tenant_id': self.fake_tenant_id, 'tenant_id': self.fake_tenant_id,
'flavor:id': flavor}} FLAVOR_NETWORK: flavor}}
return data return data
def _fake_port(self, net_id): def _fake_port(self, net_id):
@ -134,18 +141,25 @@ class MetaQuantumPluginV2Test(unittest.TestCase):
'enable_dhcp': True, 'enable_dhcp': True,
'ip_version': 4}} 'ip_version': 4}}
def _fake_router(self, flavor):
data = {'router': {'name': flavor, 'admin_state_up': True,
'tenant_id': self.fake_tenant_id,
FLAVOR_ROUTER: flavor,
'external_gateway_info': None}}
return data
def test_create_delete_network(self): def test_create_delete_network(self):
network1 = self._fake_network('fake1') network1 = self._fake_network('fake1')
ret1 = self.plugin.create_network(self.context, network1) ret1 = self.plugin.create_network(self.context, network1)
self.assertEqual('fake1', ret1['flavor:id']) self.assertEqual('fake1', ret1[FLAVOR_NETWORK])
network2 = self._fake_network('fake2') network2 = self._fake_network('fake2')
ret2 = self.plugin.create_network(self.context, network2) ret2 = self.plugin.create_network(self.context, network2)
self.assertEqual('fake2', ret2['flavor:id']) self.assertEqual('fake2', ret2[FLAVOR_NETWORK])
network3 = self._fake_network('proxy') network3 = self._fake_network('proxy')
ret3 = self.plugin.create_network(self.context, network3) ret3 = self.plugin.create_network(self.context, network3)
self.assertEqual('proxy', ret3['flavor:id']) self.assertEqual('proxy', ret3[FLAVOR_NETWORK])
db_ret1 = self.plugin.get_network(self.context, ret1['id']) db_ret1 = self.plugin.get_network(self.context, ret1['id'])
self.assertEqual('fake1', db_ret1['name']) self.assertEqual('fake1', db_ret1['name'])
@ -160,7 +174,7 @@ class MetaQuantumPluginV2Test(unittest.TestCase):
self.assertEqual(3, len(db_ret4)) self.assertEqual(3, len(db_ret4))
db_ret5 = self.plugin.get_networks(self.context, db_ret5 = self.plugin.get_networks(self.context,
{'flavor:id': ['fake1']}) {FLAVOR_NETWORK: ['fake1']})
self.assertEqual(1, len(db_ret5)) self.assertEqual(1, len(db_ret5))
self.assertEqual('fake1', db_ret5[0]['name']) self.assertEqual('fake1', db_ret5[0]['name'])
self.plugin.delete_network(self.context, ret1['id']) self.plugin.delete_network(self.context, ret1['id'])
@ -268,6 +282,26 @@ class MetaQuantumPluginV2Test(unittest.TestCase):
self.plugin.delete_network(self.context, network_ret2['id']) self.plugin.delete_network(self.context, network_ret2['id'])
self.plugin.delete_network(self.context, network_ret3['id']) self.plugin.delete_network(self.context, network_ret3['id'])
def test_create_delete_router(self):
router1 = self._fake_router('fake1')
router_ret1 = self.plugin.create_router(self.context, router1)
router2 = self._fake_router('fake2')
router_ret2 = self.plugin.create_router(self.context, router2)
self.assertEqual('fake1', router_ret1[FLAVOR_ROUTER])
self.assertEqual('fake2', router_ret2[FLAVOR_ROUTER])
router_in_db1 = self.plugin.get_router(self.context, router_ret1['id'])
router_in_db2 = self.plugin.get_router(self.context, router_ret2['id'])
self.assertEqual('fake1', router_in_db1[FLAVOR_ROUTER])
self.assertEqual('fake2', router_in_db2[FLAVOR_ROUTER])
self.plugin.delete_router(self.context, router_ret1['id'])
self.plugin.delete_router(self.context, router_ret2['id'])
with self.assertRaises(l3.RouterNotFound):
self.plugin.get_router(self.context, router_ret1['id'])
def test_extension_method(self): def test_extension_method(self):
self.assertEqual('fake1', self.plugin.fake_func()) self.assertEqual('fake1', self.plugin.fake_func())
self.assertEqual('fake2', self.plugin.fake_func2()) self.assertEqual('fake2', self.plugin.fake_func2())

View File

@ -86,6 +86,7 @@ class TestBasicRouterOperations(unittest.TestCase):
def _test_internal_network_action(self, action): def _test_internal_network_action(self, action):
port_id = _uuid() port_id = _uuid()
router_id = _uuid() router_id = _uuid()
network_id = _uuid()
ri = l3_agent.RouterInfo(router_id, self.conf.root_helper) ri = l3_agent.RouterInfo(router_id, self.conf.root_helper)
agent = l3_agent.L3NATAgent(self.conf) agent = l3_agent.L3NATAgent(self.conf)
interface_name = agent.get_internal_device_name(port_id) interface_name = agent.get_internal_device_name(port_id)
@ -95,7 +96,8 @@ class TestBasicRouterOperations(unittest.TestCase):
if action == 'add': if action == 'add':
self.device_exists.return_value = False self.device_exists.return_value = False
agent.internal_network_added(ri, ex_gw_port, port_id, cidr, mac) agent.internal_network_added(ri, ex_gw_port, network_id,
port_id, cidr, mac)
self.assertEquals(self.mock_driver.plug.call_count, 1) self.assertEquals(self.mock_driver.plug.call_count, 1)
self.assertEquals(self.mock_driver.init_l3.call_count, 1) self.assertEquals(self.mock_driver.init_l3.call_count, 1)
elif action == 'remove': elif action == 'remove':
@ -120,6 +122,7 @@ class TestBasicRouterOperations(unittest.TestCase):
'subnet_id': _uuid()}], 'subnet_id': _uuid()}],
'subnet': {'gateway_ip': '20.0.0.1'}, 'subnet': {'gateway_ip': '20.0.0.1'},
'id': _uuid(), 'id': _uuid(),
'network_id': _uuid(),
'mac_address': 'ca:fe:de:ad:be:ef', 'mac_address': 'ca:fe:de:ad:be:ef',
'ip_cidr': '20.0.0.30/24'} 'ip_cidr': '20.0.0.30/24'}
@ -180,9 +183,11 @@ class TestBasicRouterOperations(unittest.TestCase):
# return data so that state is built up # return data so that state is built up
ex_gw_port = {'id': _uuid(), ex_gw_port = {'id': _uuid(),
'network_id': _uuid(),
'fixed_ips': [{'ip_address': '19.4.4.4', 'fixed_ips': [{'ip_address': '19.4.4.4',
'subnet_id': _uuid()}]} 'subnet_id': _uuid()}]}
internal_port = {'id': _uuid(), internal_port = {'id': _uuid(),
'network_id': _uuid(),
'admin_state_up': True, 'admin_state_up': True,
'fixed_ips': [{'ip_address': '35.4.4.4', 'fixed_ips': [{'ip_address': '35.4.4.4',
'subnet_id': _uuid()}], 'subnet_id': _uuid()}],

View File

@ -23,8 +23,9 @@ from quantum.agent.common import config
from quantum.agent.linux import interface from quantum.agent.linux import interface
from quantum.agent.linux import ip_lib from quantum.agent.linux import ip_lib
from quantum.agent.linux import utils from quantum.agent.linux import utils
from quantum.openstack.common import cfg
from quantum.agent.dhcp_agent import DeviceManager from quantum.agent.dhcp_agent import DeviceManager
from quantum.extensions.flavor import (FLAVOR_NETWORK)
from quantum.openstack.common import cfg
class BaseChild(interface.LinuxInterfaceDriver): class BaseChild(interface.LinuxInterfaceDriver):
@ -345,7 +346,7 @@ class TestMetaInterfaceDriver(TestBase):
self.client_inst = mock.Mock() self.client_inst = mock.Mock()
client_cls.return_value = self.client_inst client_cls.return_value = self.client_inst
fake_network = {'network': {'flavor:id': 'fake1'}} fake_network = {'network': {FLAVOR_NETWORK: 'fake1'}}
fake_port = {'ports': fake_port = {'ports':
[{'mac_address': [{'mac_address':
'aa:bb:cc:dd:ee:ffa', 'network_id': 'test'}]} 'aa:bb:cc:dd:ee:ffa', 'network_id': 'test'}]}