
Fixes bug #1185174 This reverts commit b8b2c4e, 2f13345, 362bd7f, 8da2fb7 and 956b873. We don't yet have the infrastructure in place to use latest oslo.config in the gate or even get the correct metadata for it in our packaging. The gory details are here: http://lists.openstack.org/pipermail/openstack-dev/2013-May/009586.html Change-Id: Idf89ca418df158e6f94279c9c5fa44f23073a9d7
191 lines
6.3 KiB
Python
191 lines
6.3 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
#
|
|
# Copyright 2013 VMware. All rights reserved.
|
|
#
|
|
# 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.
|
|
#
|
|
# @author: Salvatore Orlando, VMware
|
|
|
|
from abc import abstractmethod
|
|
|
|
from oslo.config import cfg
|
|
|
|
from quantum.api import extensions
|
|
from quantum.api.v2 import attributes
|
|
from quantum.api.v2 import base
|
|
from quantum import manager
|
|
from quantum import quota
|
|
|
|
|
|
RESOURCE_NAME = "network_gateway"
|
|
# Use dash for alias and collection name
|
|
EXT_ALIAS = RESOURCE_NAME.replace('_', '-')
|
|
COLLECTION_NAME = "%ss" % EXT_ALIAS
|
|
DEVICE_ID_ATTR = 'id'
|
|
IFACE_NAME_ATTR = 'interface_name'
|
|
|
|
# Attribute Map for Network Gateway Resource
|
|
# TODO(salvatore-orlando): add admin state as other quantum resources
|
|
RESOURCE_ATTRIBUTE_MAP = {
|
|
COLLECTION_NAME: {
|
|
'id': {'allow_post': False, 'allow_put': False,
|
|
'is_visible': True},
|
|
'name': {'allow_post': True, 'allow_put': True,
|
|
'validate': {'type:string': None},
|
|
'is_visible': True, 'default': ''},
|
|
'default': {'allow_post': False, 'allow_put': False,
|
|
'is_visible': True},
|
|
'devices': {'allow_post': True, 'allow_put': False,
|
|
'validate': {'type:device_list': None},
|
|
'is_visible': True},
|
|
'ports': {'allow_post': False, 'allow_put': False,
|
|
'default': [],
|
|
'is_visible': True},
|
|
'tenant_id': {'allow_post': True, 'allow_put': False,
|
|
'validate': {'type:string': None},
|
|
'required_by_policy': True,
|
|
'is_visible': True}
|
|
}
|
|
}
|
|
|
|
|
|
def _validate_device_list(data, valid_values=None):
|
|
"""Validate the list of service definitions."""
|
|
if not data:
|
|
# Devices must be provided
|
|
msg = _("Cannot create a gateway with an empty device list")
|
|
return msg
|
|
try:
|
|
for device in data:
|
|
key_specs = {DEVICE_ID_ATTR:
|
|
{'type:regex': attributes.UUID_PATTERN,
|
|
'required': True},
|
|
IFACE_NAME_ATTR:
|
|
{'type:string': None,
|
|
'required': False}}
|
|
err_msg = attributes._validate_dict(
|
|
device, key_specs=key_specs)
|
|
if err_msg:
|
|
return err_msg
|
|
unexpected_keys = [key for key in device if key not in key_specs]
|
|
if unexpected_keys:
|
|
err_msg = ("Unexpected keys found in device description:%s",
|
|
",".join(unexpected_keys))
|
|
return err_msg
|
|
except TypeError:
|
|
return (_("%s: provided data are not iterable") %
|
|
_validate_device_list.__name__)
|
|
|
|
nw_gw_quota_opts = [
|
|
cfg.IntOpt('quota_network_gateway',
|
|
default=5,
|
|
help=_('number of network gateways allowed per tenant, '
|
|
'-1 for unlimited'))
|
|
]
|
|
|
|
cfg.CONF.register_opts(nw_gw_quota_opts, 'QUOTAS')
|
|
|
|
attributes.validators['type:device_list'] = _validate_device_list
|
|
|
|
|
|
class Nvp_networkgw(object):
|
|
"""API extension for Layer-2 Gateway support.
|
|
|
|
The Layer-2 gateway feature allows for connecting quantum networks
|
|
with external networks at the layer-2 level. No assumption is made on
|
|
the location of the external network, which might not even be directly
|
|
reachable from the hosts where the VMs are deployed.
|
|
|
|
This is achieved by instantiating 'network gateways', and then connecting
|
|
Quantum network to them.
|
|
"""
|
|
|
|
@classmethod
|
|
def get_name(cls):
|
|
return "Quantum-NVP Network Gateway"
|
|
|
|
@classmethod
|
|
def get_alias(cls):
|
|
return EXT_ALIAS
|
|
|
|
@classmethod
|
|
def get_description(cls):
|
|
return "Connects Quantum networks with external networks at layer 2"
|
|
|
|
@classmethod
|
|
def get_namespace(cls):
|
|
return "http://docs.openstack.org/ext/quantum/network-gateway/api/v1.0"
|
|
|
|
@classmethod
|
|
def get_updated(cls):
|
|
return "2012-11-30T10:00:00-00:00"
|
|
|
|
@classmethod
|
|
def get_resources(cls):
|
|
"""Returns Ext Resources."""
|
|
plugin = manager.QuantumManager.get_plugin()
|
|
params = RESOURCE_ATTRIBUTE_MAP.get(COLLECTION_NAME, dict())
|
|
|
|
member_actions = {'connect_network': 'PUT',
|
|
'disconnect_network': 'PUT'}
|
|
|
|
# register quotas for network gateways
|
|
quota.QUOTAS.register_resource_by_name(RESOURCE_NAME)
|
|
collection_name = COLLECTION_NAME.replace('_', '-')
|
|
controller = base.create_resource(collection_name,
|
|
RESOURCE_NAME,
|
|
plugin, params,
|
|
member_actions=member_actions)
|
|
return [extensions.ResourceExtension(COLLECTION_NAME,
|
|
controller,
|
|
member_actions=member_actions)]
|
|
|
|
def get_extended_resources(self, version):
|
|
if version == "2.0":
|
|
return RESOURCE_ATTRIBUTE_MAP
|
|
else:
|
|
return {}
|
|
|
|
|
|
class NetworkGatewayPluginBase(object):
|
|
|
|
@abstractmethod
|
|
def create_network_gateway(self, context, network_gateway):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def update_network_gateway(self, context, id, network_gateway):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def get_network_gateway(self, context, id, fields=None):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def delete_network_gateway(self, context, id):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def get_network_gateways(self, context, filters=None, fields=None):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def connect_network(self, context, network_gateway_id,
|
|
network_mapping_info):
|
|
pass
|
|
|
|
@abstractmethod
|
|
def disconnect_network(self, context, network_gateway_id,
|
|
network_mapping_info):
|
|
pass
|