blob: ca50c215b2fec75cd35aa4df9fb232a2d708a15d [file] [log] [blame]
# -*- coding: utf-8 -*-
'''
Management of Neutron resources
===============================
:depends: - neutronclient Python module
:configuration: See :py:mod:`salt.modules.neutron` for setup instructions.
.. code-block:: yaml
neutronng network present:
neutronng.network_present:
- name: Netone
- provider_physical_network: PHysnet1
- provider_network_type: vlan
'''
import logging
from functools import wraps
LOG = logging.getLogger(__name__)
def __virtual__():
'''
Only load if neutron module is present in __salt__
'''
return 'neutronng' if 'neutron.list_networks' in __salt__ else False
def _test_call(method):
(resource, functionality) = method.func_name.split('_')
if functionality == 'present':
functionality = 'updated'
else:
functionality = 'removed'
@wraps(method)
def check_for_testing(name, *args, **kwargs):
if __opts__.get('test', None):
return _no_change(name, resource, test=functionality)
return method(name, *args, **kwargs)
return check_for_testing
def _neutron_module_call(method, *args, **kwargs):
return __salt__['neutronng.{0}'.format(method)](*args, **kwargs)
def _get_tenant_id(tenant_name, *args, **kwargs):
try:
tenant_id = __salt__['keystoneng.tenant_get'](
name=tenant_name, **kwargs)[tenant_name]['id']
except:
tenant_id = None
LOG.debug('Cannot get the tenant id. User {0} is not an admin.'.format(
kwargs.get('connection_user')))
return tenant_id
def _auth(profile=None, endpoint_type=None):
'''
Set up neutron credentials
'''
if profile:
credentials = __salt__['config.option'](profile)
user = credentials['keystone.user']
password = credentials['keystone.password']
tenant = credentials['keystone.tenant']
auth_url = credentials['keystone.auth_url']
kwargs = {
'connection_user': user,
'connection_password': password,
'connection_tenant': tenant,
'connection_auth_url': auth_url,
'connection_endpoint_type': endpoint_type,
'profile': profile
}
return kwargs
@_test_call
def network_present(name=None,
network_id=None,
tenant=None,
provider_network_type=None,
provider_physical_network=None,
router_external=None,
admin_state_up=None,
shared=None,
provider_segmentation_id=None,
profile=None,
endpoint_type=None,
dns_domain=None):
'''
Ensure that the neutron network is present with the specified properties.
name
The name of the network to manage
'''
connection_args = _auth(profile, endpoint_type)
tenant_id = _get_tenant_id(tenant_name=tenant, **connection_args)
existing_networks = _neutron_module_call(
'list_networks', name=name, tenant_id=tenant_id,
**connection_args)['networks']
network_arguments = _get_non_null_args(
name=name,
provider_network_type=provider_network_type,
provider_physical_network=provider_physical_network,
router_external=router_external,
admin_state_up=admin_state_up,
shared=shared,
tenant_id=tenant_id,
provider_segmentation_id=provider_segmentation_id,
dns_domain=dns_domain)
if len(existing_networks) == 0:
network_arguments.update(connection_args)
res = _neutron_module_call(
'create_network', **network_arguments)['network']
if res.get('name') == name:
return _created(name, 'network', res['name'])
elif len(existing_networks) > 1:
LOG.error("More than one network found with the name: {0}".format(
name))
elif len(existing_networks) == 1:
existing_network = existing_networks[0]
LOG.info('CONNECTION STRINGS' + str(connection_args))
LOG.info('existing ' + str(existing_network))
LOG.info('new ' + str(network_arguments))
existing_network = dict((key.replace(':', '_', 1), value)
for key, value in
existing_network.iteritems())
# generate differential
diff = dict((key, value) for key, value in network_arguments.iteritems()
if existing_network.get(key, None) != value)
if diff:
# update the changes
network_arguments = diff.copy()
network_arguments.update(connection_args)
try:
LOG.debug('updating network {0} with changes {1}'.format(
name, str(diff)))
_neutron_module_call('update_network',
existing_network['id'],
**network_arguments)
changes_dict = _created(name, 'network', diff)
changes_dict['comment'] = '{1} {0} updated'.format(name, 'network')
return changes_dict
except:
LOG.error('Could not update network {0}'.format(name))
return _update_failed(name, 'network')
return _no_change(name, 'network')
return _create_failed(name, 'network')
@_test_call
def network_absent(name, network_id=None, profile=None, endpoint_type=None):
connection_args = _auth(profile, endpoint_type)
identifier = network_id or name
_neutron_module_call(
'delete_network', identifier, **connection_args)
return _absent(identifier, 'network')
@_test_call
def subnet_present(name,
network_name=None,
network_id=None,
tenant=None,
cidr=None,
ip_version=4,
enable_dhcp=True,
allocation_pools=None,
gateway_ip=None,
dns_nameservers=None,
host_routes=None,
profile=None,
endpoint_type=None):
'''
Ensure that the neutron subnet is present with the specified properties.
name
The name of the subnet to manage
'''
if network_name is None and network_id is None:
LOG.error("Network identificator name or uuid should be provided.")
return _create_failed(name, 'subnet')
connection_args = _auth(profile, endpoint_type)
tenant_id = _get_tenant_id(tenant_name=tenant, **connection_args)
existing_subnets = _neutron_module_call(
'list_subnets', tenant_id=tenant_id, name=name,
**connection_args)['subnets']
subnet_arguments = _get_non_null_args(
name=name,
cidr=cidr,
ip_version=ip_version,
enable_dhcp=enable_dhcp,
allocation_pools=allocation_pools,
gateway_ip=gateway_ip,
dns_nameservers=dns_nameservers,
host_routes=host_routes)
if network_id is None and network_name:
existing_networks = _neutron_module_call(
'list_networks', tenant_id=tenant_id, name=network_name,
**connection_args)['networks']
if len(existing_networks) == 0:
LOG.error("Can't find network with name: {0}".format(network_name))
elif len(existing_networks) == 1:
network_id = existing_networks[0]['id']
elif len(existing_networks) > 1:
LOG.error("Multiple networks with name: {0} found.".format(
network_name))
if network_id is None:
return _create_failed(name, 'subnet')
subnet_arguments['network_id'] = network_id
if len(existing_subnets) == 0:
subnet_arguments.update(connection_args)
res = _neutron_module_call('create_subnet', tenant_id=tenant_id,
**subnet_arguments)['subnet']
if res.get('name') == name:
return _created(name, 'subnet', res)
return _create_failed(name, 'subnet')
elif len(existing_subnets) == 1:
existing_subnet = existing_subnets[0]
# create differential
LOG.error('existing ' + str(existing_subnet))
LOG.error('new ' + str(subnet_arguments))
diff = dict((key, value) for key, value in subnet_arguments.iteritems()
if existing_subnet.get(key, None) != value)
if not diff:
return _no_change(name, 'subnet')
# update the changes
subnet_arguments = diff.copy()
subnet_arguments.update(connection_args)
try:
LOG.debug('updating subnet {0} with changes {1}'.format(
name, str(diff)))
_neutron_module_call('update_subnet',
existing_subnet['id'],
**subnet_arguments)
changes_dict = _created(name, 'subnet', diff)
changes_dict['comment'] = '{1} {0} updated'.format(name, 'subnet')
return changes_dict
except:
LOG.error('Could not update subnet {0}'.format(name))
return _update_failed(name, 'subnet')
elif len(existing_subnets) > 1:
LOG.error("Multiple subnets with name: {0} found".format(
name))
return _create_failed(name, 'network')
@_test_call
def subnet_absent(name, subnet_id=None, profile=None, endpoint_type=None):
connection_args = _auth(profile, endpoint_type)
identifier = subnet_id or name
_neutron_module_call(
'delete_subnet', identifier, **connection_args)
return _absent(name, 'subnet')
@_test_call
def router_present(name=None,
tenant=None,
gateway_network=None,
interfaces=None,
admin_state_up=True,
profile=None,
endpoint_type=None):
'''
Ensure that the neutron router is present with the specified properties.
name
The name of the subnet to manage
gateway_network
The network that would be the router's default gateway
interfaces
list of subnets the router attaches to
'''
connection_args = _auth(profile, endpoint_type)
tenant_name = tenant
try:
tenant_id = __salt__['keystoneng.tenant_get'](
name=tenant_name, **connection_args)[tenant_name]['id']
except:
tenant_id = None
LOG.debug('Cannot get the tenant id. User {0} is not an admin.'.format(
connection_args['connection_user']))
existing_router = _neutron_module_call(
'list_routers', name=name, **connection_args)
if not existing_router:
_neutron_module_call('create_router', name=name, tenant_id=tenant_id, admin_state_up=admin_state_up, **connection_args)
created_router = _neutron_module_call(
'list_routers', name=name, **connection_args)
if created_router:
router_id = created_router[name]['id']
network = _neutron_module_call(
'list_networks', name=gateway_network, **connection_args)["networks"]
#TODO test for more networks
gateway_network_id = network[0]['id']
_neutron_module_call('router_gateway_set',
router_id=router_id,
external_gateway=gateway_network_id,
**connection_args)
for interface in interfaces:
subnet = _neutron_module_call(
'list_subnets', name=interface, **connection_args)["subnets"]
subnet_id = subnet[0]['id']
_neutron_module_call('router_add_interface',
router_id=router_id,
subnet_id=subnet_id,
**connection_args)
return _created(name,
'router',
_neutron_module_call('list_routers',
name=name,
**connection_args))
return _create_failed(name, 'router')
router_id = existing_router[name]['id']
existing_router = existing_router[name]
diff = {}
if ( admin_state_up == True or admin_state_up == False ) and existing_router['admin_state_up'] != admin_state_up:
diff.update({'admin_state_up': admin_state_up})
if gateway_network:
network = _neutron_module_call(
'list_networks', name=gateway_network, **connection_args)["networks"]
gateway_network_id = network[0]['id']
if not existing_router['external_gateway_info'] and not existing_router['external_gateway_info'] == None:
if existing_router['external_gateway_info']['network_id'] != gateway_network_id:
diff.update({'external_gateway_info': {'network_id': gateway_network_id}})
elif not existing_router['external_gateway_info'] == None:
if not 'network_id' in existing_router['external_gateway_info'] or existing_router['external_gateway_info']['network_id'] != gateway_network_id:
diff.update({'external_gateway_info': {'network_id': gateway_network_id}})
if diff:
# update the changes
router_args = diff.copy()
router_args.update(connection_args)
try:
_neutron_module_call('update_router', existing_router['id'], **router_args)
changes_dict = _created(name, 'router', diff)
changes_dict['comment'] = 'Router {0} updated'.format(name)
return changes_dict
except:
LOG.exception('Router {0} could not be updated'.format(name))
return _update_failed(name, 'router')
return _no_change(name, 'router')
def floatingip_present(name=None,
tenant_name=None,
subnet=None,
tenant=None,
network=None,
port_id=None,
fip_exists=False,
profile=None,
endpoint_type=None):
'''
Ensure that the floating ip address is present for an instance
'''
instance_id = __salt__['novang.server_get'](name=name, tenant_name=tenant_name, profile=profile)
subnet_name = subnet
connection_args = _auth(profile, endpoint_type)
existing_subnet = _neutron_module_call(
'list_subnets', name=subnet_name, **connection_args)["subnets"]
subnet_id = existing_subnet[0]['id']
ret = {}
existing_ports = _neutron_module_call(
'list_ports', **connection_args)
existing_floatingips = _neutron_module_call(
'list_floatingips', **connection_args)
tenant = __salt__['keystoneng.tenant_get'](name=tenant_name, **connection_args)
tenant_id = tenant[tenant_name]['id']
existing_network = _neutron_module_call(
'list_networks', name=network, **connection_args)["networks"]
floating_network_id = existing_network[0]['id']
for key, value in existing_ports.iteritems():
try:
if value['fixed_ips'][0]['subnet_id'] == subnet_id and value['device_id'] == instance_id:
port_id=value['id']
except:
pass
for key, value in existing_floatingips.iteritems():
try:
if value['floating_network_id'] == floating_network_id and value['port_id'] == port_id and value['tenant_id'] == tenant_id:
fip_exists = True
break
except:
pass
if fip_exists == False:
for key, value in existing_ports.iteritems():
try:
if value['fixed_ips'][0]['subnet_id'] == subnet_id and value['device_id'] == instance_id:
ret = _neutron_module_call('create_floatingip', floating_network_id=floating_network_id, port_id=value['id'], tenant_id=tenant_id, **connection_args)
except:
pass
return _created('port', 'floatingip', ret)
else:
return _no_change('for instance {0}'.format(name), 'floatingip')
def security_group_present(name=None,
tenant=None,
description='',
rules=[],
profile=None,
endpoint_type=None):
'''
Ensure that the security group is present with the specified properties.
name
The name of the security group
description
The description of the security group
rules
list of rules to be added to the given security group
'''
# If the user is an admin, he's able to see the security groups from
# other tenants. In this case, we'll use the tenant id to get an existing
# security group.
connection_args = _auth(profile, endpoint_type)
tenant_name = tenant
try:
tenant_id = __salt__['keystoneng.tenant_get'](
name=tenant_name, **connection_args)[tenant_name]['id']
except:
tenant_id = None
LOG.debug('Cannot get the tenant id. User {0} is not an admin.'.format(
connection_args['connection_user']))
if tenant_id:
security_group = _neutron_module_call(
'list_security_groups', name=name, tenant_id=tenant_id,
**connection_args)
else:
security_group = _neutron_module_call(
'list_security_groups', name=name, **connection_args)
if not security_group:
# Create the security group as it doesn't exist already.
security_group_id = _neutron_module_call('create_security_group',
name=name,
description=description,
tenant_id=tenant_id,
**connection_args)
else:
security_group_id = security_group[name]['id']
# Set the missing rules attributes (in case the user didn't specify them
# in pillar) to some default values.
rules_attributes_defaults = {
'direction': 'ingress',
'ethertype': 'IPv4',
'protocol': 'TCP',
'port_range_min': None,
'port_range_max': None,
'remote_ip_prefix': None
}
for rule in rules:
for attribute in rules_attributes_defaults.keys():
if not rule.has_key(attribute):
rule[attribute] = rules_attributes_defaults[attribute]
# Remove all the duplicates rules given by the user in pillar.
unique_rules = []
for rule in rules:
if rule not in unique_rules:
unique_rules.append(rule)
# Get the existing security group rules.
existing_rules = _neutron_module_call(
'list_security_groups',
id=security_group_id,
**connection_args)[name]['security_group_rules']
new_rules = {}
for rule in unique_rules:
rule_found = False
for existing_rule in existing_rules:
attributes_match = True
# Compare the attributes of the existing security group rule with
# the attributes of the rule that we want to add.
for attribute in rules_attributes_defaults.keys():
existing_attribute = '' if not existing_rule[attribute] \
else str(existing_rule[attribute]).lower()
attribute = '' if not rule[attribute] \
else str(rule[attribute]).lower()
if existing_attribute != attribute:
attributes_match = False
break
if attributes_match:
rule_found = True
break
if rule_found:
# Skip adding the rule as it already exists.
continue
rule_index = len(new_rules) + 1
new_rules.update({'Rule {0}'.format(rule_index): rule})
_neutron_module_call('create_security_group_rule',
security_group_id=security_group_id,
direction=rule['direction'],
ethertype=rule['ethertype'],
protocol=rule['protocol'],
port_range_min=rule['port_range_min'],
port_range_max=rule['port_range_max'],
remote_ip_prefix=rule['remote_ip_prefix'],
tenant_id=tenant_id,
**connection_args)
if not security_group:
# The security group didn't exist. It was created and specified
# rules were added to it.
security_group = _neutron_module_call('list_security_groups',
id=security_group_id,
**connection_args)[name]
return _created(name, 'security_group', security_group)
if len(new_rules) == 0:
# Security group already exists and specified rules are already
# present.
return _no_change(name, 'security_group')
# Security group already exists, but the specified rules were added to it.
return _updated(name, 'security_group', {'New Rules': new_rules})
def port_present(network_name, profile=None, endpoint_type=None, name=None,
tenant=None, description='', fixed_ips=None, device_id=None,
device_owner=None, binding_host_id=None, admin_state_up=True,
mac_address=None, vnic_type=None, binding_profile=None,
security_groups=None, extra_dhcp_opt=None, qos_policy=None,
allowed_address_pair=None, dns_name=None):
"""
Ensure the port is present with specified parameters.
:param network_name: Name of the network to create port in
:param profile: Authentication profile
:param endpoint_type: Endpoint type
:param name: Name of this port
:param tenant: Tenant in which the port should be created, avaiable for
admin only.
:param description: Port description
:param fixed_ips: Desired IP and/or subnet for this port:
subnet_id=<name_or_id>,ip_address=<ip>.
:param device_id: Device ID of this port
:param device_owner: Device owner of this port
:param binding_host_id: he ID of the host where the port resides.
:param admin_state_up: Admin state of this port
:param mac_address: MAC address of this port
:param vnic_type: VNIC type for this port
:param binding_profile: Custom data to be passed as binding:profile
:param security_groups: Security group associated with the port
:param extra_dhcp_opt: Extra dhcp options to be assigned to this port:
opt_na me=<dhcp_option_name>,opt_value=<value>,
ip_version={4, 6}
:param qos_policy: ID or name of the QoS policy that shouldbe attached to
the resource
:param allowed_address_pair: ip_address=IP_ADDR|CIDR[,mac_address=MAC_ADDR]
Allowed address pair associated with the port.
"ip_address" parameter is required. IP address
or CIDR can be specified for "ip_address".
"mac_address" parameter is optional.
:param dns_name: Assign DNS name to the port (requires DNS integration
extension)
"""
connection_args = _auth(profile, endpoint_type)
tenant_id = _get_tenant_id(tenant_name=tenant, **connection_args)
network_id = None
port_exists = False
port_arguments = _get_non_null_args(
name=name, tenant_id=tenant_id, description=description,
fixed_ips=fixed_ips, device_id=device_id, device_owner=device_owner,
admin_state_up=admin_state_up,
mac_address=mac_address, vnic_type=vnic_type,
binding_profile=binding_profile,
extra_dhcp_opt=extra_dhcp_opt, qos_policy=qos_policy,
allowed_address_pair=allowed_address_pair, dns_name=dns_name)
if binding_host_id:
port_arguments['binding:host_id'] = binding_host_id
if security_groups:
sec_group_list = []
for sec_group_name in security_groups:
security_group = _neutron_module_call(
'list_security_groups', name=sec_group_name, **connection_args)
if security_group:
sec_group_list.append(security_group[sec_group_name]['id'])
port_arguments['security_groups'] = sec_group_list
existing_networks = _neutron_module_call(
'list_networks', tenant_id=tenant_id, name=network_name,
**connection_args)['networks']
if len(existing_networks) == 0:
LOG.error("Can't find network with name: {0}".format(network_name))
elif len(existing_networks) == 1:
network_id = existing_networks[0]['id']
elif len(existing_networks) > 1:
LOG.error("Multiple networks with name: {0} found.".format(network_name))
if network_id is None:
return _create_failed(name, 'port')
port_arguments['network_id'] = network_id
existing_ports = _neutron_module_call(
'list_ports', network_id=network_id, tenant_id=tenant_id,
**connection_args)
if name:
for key, value in existing_ports.iteritems():
try:
if value['name'] == name and value['tenant_id'] == tenant_id:
port_exists = True
break
except KeyError:
pass
if not port_exists:
port_arguments.update(connection_args)
res = _neutron_module_call('create_port', **port_arguments)['port']
if res['name'] == name:
return _created(name, 'port', res)
return _create_failed(name, 'port')
else:
return _no_change('for instance {0}'.format(name), 'port')
def _created(name, resource, resource_definition):
changes_dict = {'name': name,
'changes': resource_definition,
'result': True,
'comment': '{0} {1} created'.format(resource, name)}
return changes_dict
def _updated(name, resource, resource_definition):
changes_dict = {'name': name,
'changes': resource_definition,
'result': True,
'comment': '{0} {1} updated'.format(resource, name)}
return changes_dict
def _no_change(name, resource, test=False):
changes_dict = {'name': name,
'changes': {},
'result': True}
if test:
changes_dict['comment'] = \
'{0} {1} will be {2}'.format(resource, name, test)
else:
changes_dict['comment'] = \
'{0} {1} is in correct state'.format(resource, name)
return changes_dict
def _deleted(name, resource, resource_definition):
changes_dict = {'name': name,
'changes': {},
'comment': '{0} {1} removed'.format(resource, name),
'result': True}
return changes_dict
def _absent(name, resource):
changes_dict = {'name': name,
'changes': {},
'comment': '{0} {1} not present'.format(resource, name),
'result': True}
return changes_dict
def _delete_failed(name, resource):
changes_dict = {'name': name,
'changes': {},
'comment': '{0} {1} failed to delete'.format(resource,
name),
'result': False}
return changes_dict
def _create_failed(name, resource):
changes_dict = {'name': name,
'changes': {},
'comment': '{0} {1} failed to create'.format(resource,
name),
'result': False}
return changes_dict
def _update_failed(name, resource):
changes_dict = {'name': name,
'changes': {},
'comment': '{0} {1} failed to update'.format(resource,
name),
'result': False}
return changes_dict
def _get_non_null_args(**kwargs):
'''
Return those kwargs which are not null
'''
return dict((key, value,) for key, value in kwargs.iteritems()
if value is not None)