group-based-policy/gbpservice/neutron/plugins/ml2plus/drivers/apic_aim/extension_driver.py

183 lines
8.2 KiB
Python

# Copyright (c) 2016 Cisco Systems Inc.
# 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.
from neutron.api import extensions
from neutron.db import address_scope_db
from neutron import manager as n_manager
from neutron_lib import exceptions as n_exc
from oslo_log import log
from oslo_utils import excutils
from aim.api import resource as aim_res
from aim import exceptions as aim_exc
from gbpservice._i18n import _LE
from gbpservice._i18n import _LI
from gbpservice.neutron import extensions as extensions_pkg
from gbpservice.neutron.extensions import cisco_apic
from gbpservice.neutron.plugins.ml2plus import driver_api as api_plus
from gbpservice.neutron.plugins.ml2plus.drivers.apic_aim import (
extension_db as extn_db)
LOG = log.getLogger(__name__)
class ApicExtensionDriver(api_plus.ExtensionDriver,
extn_db.ExtensionDbMixin):
def __init__(self):
LOG.info(_LI("APIC AIM ED __init__"))
self._mechanism_driver = None
def initialize(self):
LOG.info(_LI("APIC AIM ED initializing"))
extensions.append_api_extensions_path(extensions_pkg.__path__)
@property
def _md(self):
if not self._mechanism_driver:
# REVISIT(rkukura): It might be safer to search the MDs by
# class rather than index by name, or to use a class
# variable to find the instance.
plugin = n_manager.NeutronManager.get_plugin()
mech_mgr = plugin.mechanism_manager
self._mechanism_driver = mech_mgr.mech_drivers['apic_aim'].obj
return self._mechanism_driver
@property
def extension_alias(self):
return "cisco-apic"
def extend_network_dict(self, session, base_model, result):
try:
self._md.extend_network_dict(session, base_model, result)
res_dict = self.get_network_extn_db(session, result['id'])
if cisco_apic.EXTERNAL_NETWORK in res_dict:
result.setdefault(cisco_apic.DIST_NAMES, {})[
cisco_apic.EXTERNAL_NETWORK] = res_dict.pop(
cisco_apic.EXTERNAL_NETWORK)
result.update(res_dict)
except Exception:
with excutils.save_and_reraise_exception():
LOG.exception(_LE("APIC AIM extend_network_dict failed"))
def process_create_network(self, plugin_context, data, result):
if (data.get(cisco_apic.DIST_NAMES) and
data[cisco_apic.DIST_NAMES].get(cisco_apic.EXTERNAL_NETWORK)):
dn = data[cisco_apic.DIST_NAMES][cisco_apic.EXTERNAL_NETWORK]
try:
aim_res.ExternalNetwork.from_dn(dn)
except aim_exc.InvalidDNForAciResource:
raise n_exc.InvalidInput(
error_message=('%s is not valid ExternalNetwork DN' % dn))
res_dict = {cisco_apic.EXTERNAL_NETWORK: dn,
cisco_apic.NAT_TYPE:
data.get(cisco_apic.NAT_TYPE, 'distributed'),
cisco_apic.EXTERNAL_CIDRS:
data.get(cisco_apic.EXTERNAL_CIDRS, ['0.0.0.0/0'])}
self.set_network_extn_db(plugin_context.session, result['id'],
res_dict)
result.setdefault(cisco_apic.DIST_NAMES, {})[
cisco_apic.EXTERNAL_NETWORK] = res_dict.pop(
cisco_apic.EXTERNAL_NETWORK)
result.update(res_dict)
def process_update_network(self, plugin_context, data, result):
# only CIDRs can be updated
if not cisco_apic.EXTERNAL_CIDRS in data:
return
if result.get(cisco_apic.DIST_NAMES, {}).get(
cisco_apic.EXTERNAL_NETWORK):
res_dict = {cisco_apic.EXTERNAL_CIDRS:
data[cisco_apic.EXTERNAL_CIDRS]}
self.set_network_extn_db(plugin_context.session, result['id'],
res_dict)
result.update(res_dict)
def extend_subnet_dict(self, session, base_model, result):
try:
self._md.extend_subnet_dict(session, base_model, result)
res_dict = self.get_subnet_extn_db(session, result['id'])
result[cisco_apic.SNAT_HOST_POOL] = (
res_dict.get(cisco_apic.SNAT_HOST_POOL, False))
except Exception:
with excutils.save_and_reraise_exception():
LOG.exception(_LE("APIC AIM extend_subnet_dict failed"))
def process_create_subnet(self, plugin_context, data, result):
res_dict = {cisco_apic.SNAT_HOST_POOL:
data.get(cisco_apic.SNAT_HOST_POOL, False)}
self.set_subnet_extn_db(plugin_context.session, result['id'],
res_dict)
result.update(res_dict)
def process_update_subnet(self, plugin_context, data, result):
if not cisco_apic.SNAT_HOST_POOL in data:
return
res_dict = {cisco_apic.SNAT_HOST_POOL: data[cisco_apic.SNAT_HOST_POOL]}
self.set_subnet_extn_db(plugin_context.session, result['id'],
res_dict)
result.update(res_dict)
def extend_address_scope_dict(self, session, base_model, result):
try:
self._md.extend_address_scope_dict(session, base_model, result)
res_dict = self.get_address_scope_extn_db(session, result['id'])
if cisco_apic.VRF in res_dict:
result.setdefault(cisco_apic.DIST_NAMES, {})[
cisco_apic.VRF] = res_dict.pop(cisco_apic.VRF)
except Exception:
with excutils.save_and_reraise_exception():
LOG.exception(_LE("APIC AIM extend_address_scope_dict failed"))
def process_create_address_scope(self, plugin_context, data, result):
if (data.get(cisco_apic.DIST_NAMES) and
data[cisco_apic.DIST_NAMES].get(cisco_apic.VRF)):
dn = data[cisco_apic.DIST_NAMES][cisco_apic.VRF]
try:
vrf = aim_res.VRF.from_dn(dn)
except aim_exc.InvalidDNForAciResource:
raise n_exc.InvalidInput(
error_message=('%s is not valid VRF DN' % dn))
session = plugin_context.session
# Check if there is another address scope of the same
# address family mapping to same VRF.
#
# Case 1: Another address-scope with pre-existing VRF
scopes = self.get_address_scopes_by_vrf_dn(session, dn)
for scope in scopes:
if scope.ip_version == data['ip_version']:
raise n_exc.InvalidInput(
error_message=('VRF %s is already in use by '
'address-scope %s' % (dn, scope)))
# Case 2: Another address-scope with orchestrated VRF
#
# REVISIT: We don't filter by the project ID because the
# mapping of these to AIM Tenant names is not necessarily
# reversible. Consider persisting the APIC VRF identities.
scope_id = self._md.name_mapper.reverse_address_scope(
session, vrf.name, enforce=False)
if scope_id:
scope = (session.query(address_scope_db.AddressScope)
.filter_by(id=scope_id)
.first())
if scope and scope.ip_version == data['ip_version']:
raise n_exc.InvalidInput(
error_message=('VRF %s is already in use by '
'address-scope %s' % (dn, scope)))
self.set_address_scope_extn_db(session, result['id'],
{cisco_apic.VRF: dn})
result.setdefault(cisco_apic.DIST_NAMES, {})[cisco_apic.VRF] = dn