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:
parent
efa4316f64
commit
34d3e21a40
@ -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'
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
@ -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):
|
||||||
|
@ -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):
|
||||||
|
@ -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
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -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='')
|
||||||
]
|
]
|
||||||
|
@ -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
|
||||||
|
@ -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)
|
||||||
|
@ -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]
|
||||||
|
@ -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:
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
|
||||||
|
@ -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())
|
||||||
|
@ -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()}],
|
||||||
|
@ -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'}]}
|
||||||
|
Loading…
Reference in New Issue
Block a user