a178053854
Fixes bug 1234826 This fix adds a "nexus_l3_enable" configuration boolean for the Cisco Nexus plugin. When this config boolean is set to False (default), then the Nexus switches are only used for L2 switching/segmentation, and layer 3 functionality is deferred to the OVS subplugin / network control node. If this config boolean is set to True, layer 3 functionality, e.g. switch virtual interfaces, are supported on the Nexus switches. (Note that layer 3 functionality is not supported on all versions/models Nexus switches.) Some other things addressed with this fix: - The l3_port_check keyword argument which is optionally passed to the Cisco plugin's delete_port method was not being forwarded on to the OVS (sub) plugin. This keyword argument needs to be forwarded to OVS e.g. when the delete_port is being done in the context of a router interface delete (whereby l3_port_check==False). - UT test cases are added for new "nexus_l3_enable" config, which exercise router interface add/delete. - The Cisco test_network_plugin.py module is refactored/reorganized in order to cleanly add a new router interface test class. - The test_model_update_port_rollback test case was yielding a false positive result (device_owner was not being passed to self.port). Change-Id: I994b2b82769ea5e10e50dbe3a223d1518e99f714
138 lines
5.3 KiB
Python
138 lines
5.3 KiB
Python
# vim: tabstop=4 shiftwidth=4 softtabstop=4
|
|
|
|
# Copyright 2013 Cisco Systems, Inc.
|
|
#
|
|
# 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 oslo.config import cfg
|
|
|
|
from neutron.agent.common import config
|
|
|
|
|
|
cisco_plugins_opts = [
|
|
cfg.StrOpt('vswitch_plugin',
|
|
default='neutron.plugins.openvswitch.ovs_neutron_plugin.'
|
|
'OVSNeutronPluginV2',
|
|
help=_("Virtual Switch to use")),
|
|
cfg.StrOpt('nexus_plugin',
|
|
default='neutron.plugins.cisco.nexus.cisco_nexus_plugin_v2.'
|
|
'NexusPlugin',
|
|
help=_("Nexus Switch to use")),
|
|
]
|
|
|
|
cisco_opts = [
|
|
cfg.StrOpt('vlan_name_prefix', default='q-',
|
|
help=_("VLAN Name prefix")),
|
|
cfg.StrOpt('provider_vlan_name_prefix', default='p-',
|
|
help=_("VLAN Name prefix for provider vlans")),
|
|
cfg.BoolOpt('provider_vlan_auto_create', default=True,
|
|
help=_('Provider VLANs are automatically created as needed '
|
|
'on the Nexus switch')),
|
|
cfg.BoolOpt('provider_vlan_auto_trunk', default=True,
|
|
help=_('Provider VLANs are automatically trunked as needed '
|
|
'on the ports of the Nexus switch')),
|
|
cfg.BoolOpt('nexus_l3_enable', default=False,
|
|
help=_("Enable L3 support on the Nexus switches")),
|
|
cfg.BoolOpt('svi_round_robin', default=False,
|
|
help=_("Distribute SVI interfaces over all switches")),
|
|
cfg.StrOpt('model_class',
|
|
default='neutron.plugins.cisco.models.virt_phy_sw_v2.'
|
|
'VirtualPhysicalSwitchModelV2',
|
|
help=_("Model Class")),
|
|
cfg.StrOpt('nexus_driver',
|
|
default='neutron.plugins.cisco.test.nexus.'
|
|
'fake_nexus_driver.CiscoNEXUSFakeDriver',
|
|
help=_("Nexus Driver Name")),
|
|
]
|
|
|
|
cisco_n1k_opts = [
|
|
cfg.StrOpt('integration_bridge', default='br-int',
|
|
help=_("N1K Integration Bridge")),
|
|
cfg.BoolOpt('enable_tunneling', default=True,
|
|
help=_("N1K Enable Tunneling")),
|
|
cfg.StrOpt('tunnel_bridge', default='br-tun',
|
|
help=_("N1K Tunnel Bridge")),
|
|
cfg.StrOpt('local_ip', default='10.0.0.3',
|
|
help=_("N1K Local IP")),
|
|
cfg.StrOpt('tenant_network_type', default='local',
|
|
help=_("N1K Tenant Network Type")),
|
|
cfg.StrOpt('bridge_mappings', default='',
|
|
help=_("N1K Bridge Mappings")),
|
|
cfg.StrOpt('vxlan_id_ranges', default='5000:10000',
|
|
help=_("N1K VXLAN ID Ranges")),
|
|
cfg.StrOpt('network_vlan_ranges', default='vlan:1:4095',
|
|
help=_("N1K Network VLAN Ranges")),
|
|
cfg.StrOpt('default_network_profile', default='default_network_profile',
|
|
help=_("N1K default network profile")),
|
|
cfg.StrOpt('default_policy_profile', default='service_profile',
|
|
help=_("N1K default policy profile")),
|
|
cfg.StrOpt('network_node_policy_profile', default='dhcp_pp',
|
|
help=_("N1K policy profile for network node")),
|
|
cfg.StrOpt('poll_duration', default='10',
|
|
help=_("N1K Policy profile polling duration in seconds")),
|
|
]
|
|
|
|
cfg.CONF.register_opts(cisco_opts, "CISCO")
|
|
cfg.CONF.register_opts(cisco_n1k_opts, "CISCO_N1K")
|
|
cfg.CONF.register_opts(cisco_plugins_opts, "CISCO_PLUGINS")
|
|
config.register_root_helper(cfg.CONF)
|
|
|
|
# shortcuts
|
|
CONF = cfg.CONF
|
|
CISCO = cfg.CONF.CISCO
|
|
CISCO_N1K = cfg.CONF.CISCO_N1K
|
|
CISCO_PLUGINS = cfg.CONF.CISCO_PLUGINS
|
|
|
|
#
|
|
# device_dictionary - Contains all external device configuration.
|
|
#
|
|
# When populated the device dictionary format is:
|
|
# {('<device ID>', '<device ipaddr>', '<keyword>'): '<value>', ...}
|
|
#
|
|
# Example:
|
|
# {('NEXUS_SWITCH', '1.1.1.1', 'username'): 'admin',
|
|
# ('NEXUS_SWITCH', '1.1.1.1', 'password'): 'mySecretPassword',
|
|
# ('NEXUS_SWITCH', '1.1.1.1', 'compute1'): '1/1', ...}
|
|
#
|
|
device_dictionary = {}
|
|
|
|
|
|
class CiscoConfigOptions():
|
|
"""Cisco Configuration Options Class."""
|
|
|
|
def __init__(self):
|
|
self._create_device_dictionary()
|
|
|
|
def _create_device_dictionary(self):
|
|
"""
|
|
Create the device dictionary from the cisco_plugins.ini
|
|
device supported sections. Ex. NEXUS_SWITCH, N1KV.
|
|
"""
|
|
|
|
multi_parser = cfg.MultiConfigParser()
|
|
read_ok = multi_parser.read(CONF.config_file)
|
|
|
|
if len(read_ok) != len(CONF.config_file):
|
|
raise cfg.Error(_("Some config files were not parsed properly"))
|
|
|
|
for parsed_file in multi_parser.parsed:
|
|
for parsed_item in parsed_file.keys():
|
|
dev_id, sep, dev_ip = parsed_item.partition(':')
|
|
if dev_id.lower() in ['nexus_switch', 'n1kv']:
|
|
for dev_key, value in parsed_file[parsed_item].items():
|
|
device_dictionary[dev_id, dev_ip, dev_key] = value[0]
|
|
|
|
|
|
def get_device_dictionary():
|
|
return device_dictionary
|