Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

L4l7 #22

Open
wants to merge 23 commits into
base: master
Choose a base branch
from
Open

L4l7 #22

Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
fffe3ce
Basic working L4L7Cluster, L4L7Device implementations.
Sep 23, 2015
f4c7832
Add optional args to L4L7Cluster.
Sep 23, 2015
4c8d2ea
Add CCred, CCredSecret, CIf, CMgmt, and RsMDevAtt creation for L4-L7 …
Sep 25, 2015
5d8c76d
Add YAML file for L4L7Cluster.
Sep 25, 2015
6f7f5a2
Modifications to L4L7Cluster.py.
Oct 6, 2015
e2d679e
Working LDevVip, CDev and RsCIfPathAtt objects with dynamicallyCreate…
Oct 6, 2015
5e74571
Add createL4L7LogicalInterface.py.
Oct 7, 2015
2022ae0
Add createL4L7ConcreteInterface.py.
Oct 7, 2015
b0aabf3
Add L4-L7 devices logical interface support to dynamicallyCreateAppli…
Oct 7, 2015
d8d6cdf
Small change to createL4L7LogicalInterface.py.
Oct 8, 2015
88775b8
Document L4-L7 section of dynamicallyCreateApplication.yaml. Add plac…
Oct 8, 2015
aca1542
Add virtual cluster and virtual device support.
Oct 9, 2015
d771f72
Simplify YAML configuration for L4-L7 devices.
Oct 9, 2015
b539575
Add VMM domain profile source relation support for clusters.
Oct 9, 2015
8c795bf
Add VMM domain controller profile source relation support for devices.
Oct 9, 2015
67fdfc0
Add recursive L4-L7 services folder and parameter parsing.
Oct 9, 2015
8c50ee3
Add support for L4-L7 device folders/parameters to dynamicallyCreateA…
Oct 9, 2015
d1901bd
Add support for source relation to physical domain profile (RsALDevTo…
Oct 19, 2015
9ee16e2
update l4l7 device folder parsing function
Oct 19, 2015
d3ab179
Add logical device physical domain profile relation to YAML file, doc…
Oct 19, 2015
c675169
Change formatting in createL4L7LogicalInterface.py
Oct 20, 2015
e1818d6
Refactor dynamicallyCreateApplication.py to work when missing keys (s…
Oct 20, 2015
c842a4e
Update createL4L7Device folder parsing to be in-line with createL4L7C…
Feb 4, 2016
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
154 changes: 154 additions & 0 deletions configuration-python/generic_code/apicPython/createL4L7Cluster.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,154 @@

from cobra.model.vns import LDevVip, RsMDevAtt, CCred, CCredSecret, CMgmt, RsALDevToDomP, RsALDevToPhysDomP, DevFolder, DevParam
from createMo import *
import getpass
import sys

DEFAULT_CONTEXT_AWARENESS = 'single-Context'
DEFAULT_DEVICE_TYPE = 'PHYSICAL'
DEFAULT_FUNCTION_TYPE = 'GoTo'

def input_key_args():
return input_raw_input('Device Cluster Name', required=True)

def input_optional_args():
args = {
'contextAware': input_options('L4-L7 Device Cluster - Context Awareness', default=DEFAULT_CONTEXT_AWARENESS, options=['single-Context', 'multi-Context']),
'devtype': input_options('L4-L7 Device Cluster - Device Type', default=DEFAULT_DEVICE_TYPE, options=['PHYSICAL', 'VIRTUAL']),
'funcType': input_options('L4-L7 Device Cluster - Function Type', default=DEFAULT_FUNCTION_TYPE, options=['GoTo','GoThrough']),
}
return args

def create_l4l7_cluster(fv_tenant, name, **args):
"""Create L4L7 Cluster"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['contextAware', 'devtype', 'funcType']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}

vns_ldevvip = LDevVip(fv_tenant, name, **kwargs)
if 'device_package' in args:
vns_rsmdevatt = add_metadata_source_relation(vns_ldevvip, optional_args=args)
if 'cluster_username' in args:
vns_ccred = add_concrete_device_access_credentials(vns_ldevvip, optional_args=args)
if 'cluster_password' in args:
vns_ccredsecret = add_concrete_device_access_credentials_secret(vns_ldevvip, optional_args=args)
if 'cluster_ip' in args or 'cluster_port' in args:
vns_cmgmt = add_management_interface(vns_ldevvip, optional_args=args)
if 'vmm_provider' in args and 'vmm_domain' in args:
vns_rsaldevtodomp = add_source_relation_to_vmm_domain_profile(vns_ldevvip, optional_args=args)
if 'physical_domain' in args:
vns_rsaldevtophysdomp = add_source_relation_to_physical_domain_profile(vns_ldevvip, optional_args=args)
if 'device_folders' in args:
add_l4l7_device_folders(vns_ldevvip, args['device_folders'])
return vns_ldevvip

def add_metadata_source_relation(cluster_mo, **args):
"""vnsRsMDevAtt: "A source relation to the metadata definitions for a service device type. Functions as a pointer to the device package.
e.g: uni/infra/mDev-{device_package_vendor}-{device_package_model}-{device_package_version}
"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
tdn = 'uni/infra/mDev-{device_package}'.format(**args)
return RsMDevAtt(cluster_mo, tDn=tdn)

def add_concrete_device_access_credentials(cluster_mo, **args):
"""The concrete device access credentials in the L4-L7 device cluster. The concrete device access credentials normally include a password that is not displayed and is stored in encrypted form."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
return CCred(cluster_mo, name='username', value=args['cluster_username'])

def add_concrete_device_access_credentials_secret(cluster_mo, **args):
"""The secret for the concrete device access credentials in the L4-L7 device cluster. The concrete device access credentials normally include a password that is not displayed and is stored in encrypted form."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
return CCredSecret(cluster_mo, name='password', value=args['cluster_password'])

def add_management_interface(cluster_mo, **args):
"""The management interface is used to manage a concrete device in the L4-L7 device cluster. The management interface is identified by a host address and port number."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['cluster_ip', 'cluster_port']
key_map = {'cluster_ip': 'host', 'cluster_port': 'port'}
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
kwargs = {key_map[k]: v for k, v in kwargs.items()}
return CMgmt(cluster_mo, name='devMgmt', **kwargs)

def add_source_relation_to_vmm_domain_profile(cluster_mo, **args):
"""A source relation to the VMM domain profile."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['vmm_provider', 'vmm_domain']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
return RsALDevToDomP(cluster_mo, tDn='uni/vmmp-{vmm_provider}/dom-{vmm_domain}'.format(**kwargs))

def add_source_relation_to_physical_domain_profile(cluster_mo, **args):
"""A source relation to a physical domain profile."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['physical_domain']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
return RsALDevToPhysDomP(cluster_mo, tDn='uni/phys-{physical_domain}'.format(**kwargs))

def add_l4l7_device_folders(parent_mo, folder_list):
for folder in folder_list:
add_l4l7_device_folder(parent_mo, **folder)

def add_l4l7_device_folder(parent_mo, **args):
"""Recursively add device folders and parameters to parent mo.

@param parent_mo: The parent MO of the top level folder is the CDev, but the parent MO of a subfolder is its parent folder.
"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
folder_required_keys = ['name', 'key']
param_required_keys = ['name', 'key', 'value']

# parse folders
if all(k in args.keys() for k in folder_required_keys):
vns_devfolder = DevFolder(parent_mo, **{k: v for k,v in args.items() if k in folder_required_keys and v})

# parse params
if 'device_params' in args.keys(): # This folder contains device params
for param in args['device_params']:
if all(k in param.keys() for k in param_required_keys):
DevParam(vns_devfolder, **param)

# parse subfolders
if 'device_folders' in args.keys():
for folder in args['device_folders']:
add_l4l7_device_folder(vns_devfolder, **folder)
else:
raise Exception('Invalid L4-L7 device folder configuration. Missing required keys "{0}": {1}'.format(folder_required_keys, repr(args)))

class CreateL4L7Cluster(CreateMo):
def __init__(self):
self.description = 'Create an L4-L7 device cluster'
self.tenant_required = True
self.contract = None
super(CreateL4L7Cluster, self).__init__()

def set_cli_mode(self):
super(CreateL4L7Cluster, self).set_cli_mode()
self.parser_cli.add_argument('name', help='Cluster Name')
self.parser_cli.add_argument('-d', '--device_package', help='Device package, e.g "Cisco-FirePOWER-1.0"', metavar='VENDOR-MODEL-VERSION')
self.parser_cli.add_argument('-f', '--function_type', choices=['GoTo','GoThrough'], dest='funcType', help='A GoTo device has a specific destination, depending on the package. A GoThrough device is a transparent device.')
self.parser_cli.add_argument('-t', '--device_type', choices=['PHYSICAL', 'VIRTUAL'], dest='devtype', help='Specifies whether the device cluster has PHYSICAL appliances or VIRTUAL appliances.')
self.parser_cli.add_argument('-u1', '--username', dest='cluster_username', help='Username for the L4-L7 cluster.')
self.parser_cli.add_argument('-u2', '--password', dest='cluster_password', help='Password for the L4-L7 cluster.')
self.parser_cli.add_argument('-i', '--ip', dest='cluster_ip', help='IP Address of the L4-L7 cluster host.')
self.parser_cli.add_argument('-p', '--port', dest='cluster_port', help='Port of the L4-L7 cluster host.')
self.parser_cli.add_argument('-x', '--context_aware', choices=['single-Context', 'multi-Context'], dest='contextAware',
help='The context-awareness of the Device Cluster. Single means that the device cluster cannot be shared across multiple tenants of a given type that are hosted on the provider network. Multiple means that the device cluster can be shared across multiple tenants of a given type that you are hosting on this provider network. ')

def read_key_args(self):
self.tenant = self.args.pop('tenant')
self.name = self.args.pop('name')

def wizard_mode_input_args(self):
self.args['name'] = input_key_args()
if not self.delete:
self.args['optional_args'] = input_optional_args()

def delete_mo(self):
self.check_if_mo_exist('uni/tn-'+self.tenant+'/lDevVip-', self.name, LDevVip, description='LDevVip')
super(CreateL4L7Cluster, self).delete_mo()

def main_function(self):
fv_tenant = self.check_if_tenant_exist()
vns_ldevvip = create_l4l7_cluster(fv_tenant, self.name, optional_args=self.optional_args)

if __name__ == '__main__':
mo = CreateL4L7Cluster()
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@

host: 10.10.10.10
user: admin
password: password

tenant: my_tenant
name: my_cluster
context_aware: single-Context
device_type: PHYSICAL
function_type: GoThrough
cluster_username: admin
cluster_password: cluster_password
cluster_ip: 10.10.10.11
cluster_port: 443
device_package_vendor: Cisco
device_package_model: FirePOWER
device_package_version: 1.0
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@

from cobra.model.vns import CDev, CIf, RsCIfPathAtt
from createMo import *

def input_key_args():
return input_raw_input('Concrete Interface Name', required=True)

def input_optional_args():
args = {}
# TODO
return args

def create_l4l7_concrete_interface(parent_mo, name, **args):
"""Create L4L7 Concrete Interface"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['name', 'vnicName']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
vns_cif = CIf(parent_mo, name, **kwargs)
if 'path' in args:
vns_rscifpathatt = add_source_relation_to_path_endpoint(vns_cif, args['path'])
return vns_cif

def add_source_relation_to_path_endpoint(concrete_interface_mo, path):
"""A source relation to a path endpoint. e.g: 'topology/pod-1/paths-1001/pathep-[eth1/10]' """
return RsCIfPathAtt(concrete_interface_mo, tDn=path)

class CreateL4L7ConcreteInterface(CreateMo):
def __init__(self):
self.description = 'Create an L4-L7 concrete device'
self.tenant_required = True
self.contract = None
super(CreateL4L7ConcreteInterface, self).__init__()
119 changes: 119 additions & 0 deletions configuration-python/generic_code/apicPython/createL4L7Device.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,119 @@

from cobra.model.vns import CDev, LDevVip, CCred, CCredSecret, CMgmt, RsCDevToCtrlrP, DevFolder, DevParam
from createMo import *

def input_key_args():
return input_raw_input('Concrete Device Name', required=True)

def input_optional_args():
args = {}
# TODO
return args

def create_l4l7_device(parent_mo, name, **args):
"""Create L4L7 Device"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['name', 'vmName', 'vcenterName', 'devCtxLbl']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
vns_cdev = CDev(parent_mo, name, **kwargs)
if 'device_username' in args:
vns_ccred = add_concrete_device_access_credentials(vns_cdev, optional_args=args)
if 'device_password' in args:
vns_ccredsecret = add_concrete_device_access_credentials_secret(vns_cdev, optional_args=args)
if 'device_ip' in args or 'device_port' in args:
vns_cmgmt = add_management_interface(vns_cdev, optional_args=args)
if 'vmm_provider' in args and 'vmm_domain' in args and 'vmm_controller' in args:
vns_rscdevtoctrlrp = add_source_relation_to_vmm_domain_controller_profile(vns_cdev, optional_args=args)
if 'device_folders' in args:
add_l4l7_device_folders(vns_cdev, args['device_folders'])
return vns_cdev

def add_concrete_device_access_credentials(device_mo, **args):
"""The concrete device access credentials in the L4-L7 device cluster. The concrete device access credentials normally include a password that is not displayed and is stored in encrypted form."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
return CCred(device_mo, name='username', value=args['device_username'])

def add_concrete_device_access_credentials_secret(device_mo, **args):
"""The secret for the concrete device access credentials in the L4-L7 device cluster. The concrete device access credentials normally include a password that is not displayed and is stored in encrypted form."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
return CCredSecret(device_mo, name='password', value=args['device_password'])

def add_management_interface(device_mo, **args):
"""The management interface is used to manage a concrete device in the L4-L7 device cluster. The management interface is identified by a host address and port number."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['device_ip', 'device_port']
key_map = {'device_ip': 'host', 'device_port': 'port'}
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
kwargs = {key_map[k]: v for k, v in kwargs.items()}
return CMgmt(device_mo, name='devMgmt', **kwargs)

def add_source_relation_to_vmm_domain_controller_profile(device_mo, **args):
"""Source relation to the vmm domain controller profile for validation."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['vmm_provider','vmm_domain','vmm_controller']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
return RsCDevToCtrlrP(device_mo, tDn='uni/vmmp-{vmm_provider}/dom-{vmm_domain}/ctrlr-{vmm_controller}'.format(**kwargs))

def add_l4l7_device_folders(parent_mo, folder_list):
for folder in folder_list:
add_l4l7_device_folder(parent_mo, **folder)

def add_l4l7_device_folder(parent_mo, **args):
"""Recursively add device folders and parameters to parent mo.

@param parent_mo: The parent MO of the top level folder is the CDev, but the parent MO of a subfolder is its parent folder.
"""
args = args['optional_args'] if 'optional_args' in args.keys() else args
folder_required_keys = ['name', 'key']
param_required_keys = ['name', 'key', 'value']

# parse folders
if all(k in args.keys() for k in folder_required_keys):
vns_devfolder = DevFolder(parent_mo, **{k: v for k,v in args.items() if k in folder_required_keys and v})

# parse params
if 'device_params' in args.keys(): # This folder contains device params
for param in args['device_params']:
if all(k in param.keys() for k in param_required_keys):
DevParam(vns_devfolder, **param)

# parse subfolders
if 'device_folders' in args.keys():
for folder in args['device_folders']:
add_l4l7_device_folder(vns_devfolder, **folder)
else:
raise Exception('Invalid L4-L7 device folder configuration. Missing required keys "{0}": {1}'.format(folder_required_keys, repr(args)))

class CreateL4L7Device(CreateMo):
def __init__(self):
self.description = 'Create an L4-L7 concrete device'
self.tenant_required = True
self.contract = None
super(CreateL4L7Device, self).__init__()

def set_cli_mode(self):
super(CreateL4L7Device, self).set_cli_mode()
self.parser_cli.add_argument('cluster_name', help='Name of the parent L4-L7 Device Cluster')
self.parser_cli.add_argument('name', help='Device Name')

def read_key_args(self):
self.tenant = self.args.pop('tenant')
self.cluster_name = self.args.pop('cluster_name')
self.name = self.args.pop('name')

def wizard_mode_input_args(self):
self.args['name'] = input_key_args()
if not self.delete:
self.args['optional_args'] = input_optional_args()

def delete_mo(self):
self.check_if_mo_exist('uni/tn-{tenant}/lDevVip-{cluster_name}/cDev-'.format(**self.__dict__), self.name, CDev, description='CDev')
super(CreateL4L7Device, self).delete_mo()

def main_function(self):
# Query a tenant
parent_mo = self.check_if_mo_exist('uni/tn-{tenant}/lDevVip-'.format(**self.__dict__), self.cluster_name, LDevVip, description='LDevVip')
vns_cdev = create_l4l7_device(parent_mo, self.name, optional_args=self.optional_args)

if __name__ == '__main__':
mo = CreateL4L7Device()
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@

from cobra.model.vns import CDev, LIf, RsMetaIf, RsCIfAtt, RsCIfAttN
from createMo import *

def input_key_args():
return input_raw_input('Concrete Interface Name', required=True)

def input_optional_args():
args = {}
# TODO
return args

def create_l4l7_logical_interface(parent_mo, name, **args):
"""The logical interface is associated with a set of concrete interfaces from the L4-L7 device cluster. This is used to define the connection between a service graph and device interfaces."""
args = args['optional_args'] if 'optional_args' in args.keys() else args
valid_keys = ['encap']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
vns_lif = LIf(parent_mo, name, **kwargs)
if 'label' in args:
vns_rsmetaif = add_source_relation_to_interface_label(vns_lif, device_package=args['device_package'], label=args['label'])
if 'concrete_interface':
vns_rscifatt = add_association_to_concrete_interface(vns_lif, tenant=args['tenant'], cluster=args['cluster'], device=args['device'], cifname=args['cifname'])
return vns_lif

def add_source_relation_to_interface_label(logical_interface_mo, **args):
"""A source relation to an interface label. e.g: tDn='uni/infra/mDev-Cisco-FirePOWER-1.0/mIfLbl-external' """
valid_keys = ['device_package', 'label']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
return RsMetaIf(logical_interface_mo, tDn='uni/infra/mDev-{device_package}/mIfLbl-{label}'.format(**kwargs))

def add_association_to_concrete_interface(logical_interface_mo, **args):
"""Association to a set of concrete interfaces from the device in the cluster."""
valid_keys = ['tenant', 'cluster', 'device', 'cifname']
kwargs = {k: v for k, v in args.items() if (k in valid_keys and v)}
return RsCIfAtt(logical_interface_mo, tDn='uni/tn-{tenant}/lDevVip-{cluster}/cDev-{device}/cIf-[{cifname}]'.format(**kwargs))

class CreateL4L7ConcreteInterface(CreateMo):
def __init__(self):
self.description = 'Create an L4-L7 concrete device'
self.tenant_required = True
self.contract = None
super(CreateL4L7ConcreteInterface, self).__init__()
Loading