heat/heat/engine/resources/openstack/neutron/subnet.py

303 lines
11 KiB
Python

#
# 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_utils import netutils
from heat.common import exception
from heat.common.i18n import _
from heat.engine import attributes
from heat.engine import constraints
from heat.engine import properties
from heat.engine.resources.openstack.neutron import neutron
from heat.engine import support
class Subnet(neutron.NeutronResource):
PROPERTIES = (
NETWORK_ID, NETWORK, CIDR, VALUE_SPECS, NAME, IP_VERSION,
DNS_NAMESERVERS, GATEWAY_IP, ENABLE_DHCP, ALLOCATION_POOLS,
TENANT_ID, HOST_ROUTES, IPV6_RA_MODE, IPV6_ADDRESS_MODE,
) = (
'network_id', 'network', 'cidr', 'value_specs', 'name', 'ip_version',
'dns_nameservers', 'gateway_ip', 'enable_dhcp', 'allocation_pools',
'tenant_id', 'host_routes', 'ipv6_ra_mode', 'ipv6_address_mode',
)
_ALLOCATION_POOL_KEYS = (
ALLOCATION_POOL_START, ALLOCATION_POOL_END,
) = (
'start', 'end',
)
_HOST_ROUTES_KEYS = (
ROUTE_DESTINATION, ROUTE_NEXTHOP,
) = (
'destination', 'nexthop',
)
_IPV6_DHCP_MODES = (
DHCPV6_STATEFUL, DHCPV6_STATELESS, SLAAC,
) = (
'dhcpv6-stateful', 'dhcpv6-stateless', 'slaac',
)
ATTRIBUTES = (
NAME_ATTR, NETWORK_ID_ATTR, TENANT_ID_ATTR, ALLOCATION_POOLS_ATTR,
GATEWAY_IP_ATTR, HOST_ROUTES_ATTR, IP_VERSION_ATTR, CIDR_ATTR,
DNS_NAMESERVERS_ATTR, ENABLE_DHCP_ATTR, SHOW,
) = (
'name', 'network_id', 'tenant_id', 'allocation_pools',
'gateway_ip', 'host_routes', 'ip_version', 'cidr',
'dns_nameservers', 'enable_dhcp', 'show',
)
properties_schema = {
NETWORK_ID: properties.Schema(
properties.Schema.STRING,
support_status=support.SupportStatus(
status=support.DEPRECATED,
message=_('Use property %s.') % NETWORK,
version='2014.2'),
constraints=[
constraints.CustomConstraint('neutron.network')
],
),
NETWORK: properties.Schema(
properties.Schema.STRING,
_('The ID of the attached network.'),
constraints=[
constraints.CustomConstraint('neutron.network')
],
support_status=support.SupportStatus(version='2014.2')
),
CIDR: properties.Schema(
properties.Schema.STRING,
_('The CIDR.'),
required=True,
constraints=[
constraints.CustomConstraint('net_cidr')
]
),
VALUE_SPECS: properties.Schema(
properties.Schema.MAP,
_('Extra parameters to include in the creation request.'),
default={},
update_allowed=True
),
NAME: properties.Schema(
properties.Schema.STRING,
_('The name of the subnet.'),
update_allowed=True
),
IP_VERSION: properties.Schema(
properties.Schema.INTEGER,
_('The IP version, which is 4 or 6.'),
default=4,
constraints=[
constraints.AllowedValues([4, 6]),
]
),
DNS_NAMESERVERS: properties.Schema(
properties.Schema.LIST,
_('A specified set of DNS name servers to be used.'),
default=[],
update_allowed=True
),
GATEWAY_IP: properties.Schema(
properties.Schema.STRING,
_('The gateway IP address. Set to any of [ null | ~ | "" ] to '
'create the subnet without a gateway. If omitted, the first IP '
'address within the subnet is assigned to the gateway.'),
update_allowed=True
),
ENABLE_DHCP: properties.Schema(
properties.Schema.BOOLEAN,
_('Set to true if DHCP is enabled and false if DHCP is disabled.'),
default=True,
update_allowed=True
),
ALLOCATION_POOLS: properties.Schema(
properties.Schema.LIST,
_('The start and end addresses for the allocation pools.'),
schema=properties.Schema(
properties.Schema.MAP,
schema={
ALLOCATION_POOL_START: properties.Schema(
properties.Schema.STRING,
required=True,
constraints=[
constraints.CustomConstraint('ip_addr')
]
),
ALLOCATION_POOL_END: properties.Schema(
properties.Schema.STRING,
required=True,
constraints=[
constraints.CustomConstraint('ip_addr')
]
),
},
)
),
TENANT_ID: properties.Schema(
properties.Schema.STRING,
_('The ID of the tenant who owns the network. Only administrative'
' users can specify a tenant ID other than their own.')
),
HOST_ROUTES: properties.Schema(
properties.Schema.LIST,
schema=properties.Schema(
properties.Schema.MAP,
schema={
ROUTE_DESTINATION: properties.Schema(
properties.Schema.STRING,
required=True
),
ROUTE_NEXTHOP: properties.Schema(
properties.Schema.STRING,
required=True,
constraints=[
constraints.CustomConstraint('ip_addr')
]
),
},
),
update_allowed=True
),
IPV6_RA_MODE: properties.Schema(
properties.Schema.STRING,
_('IPv6 RA (Router Advertisement) mode.'
' dhcpv6-stateful, dhcpv6-stateless, or slaac.'),
constraints=[
constraints.AllowedValues([DHCPV6_STATEFUL, DHCPV6_STATELESS,
SLAAC]),
],
support_status=support.SupportStatus(version='2015.1')
),
IPV6_ADDRESS_MODE: properties.Schema(
properties.Schema.STRING,
_('IPv6 address mode.'
' dhcpv6-stateful, dhcpv6-stateless, or slaac.'),
constraints=[
constraints.AllowedValues([DHCPV6_STATEFUL, DHCPV6_STATELESS,
SLAAC]),
],
support_status=support.SupportStatus(version='2015.1')
),
}
attributes_schema = {
NAME_ATTR: attributes.Schema(
_("Friendly name of the subnet.")
),
NETWORK_ID_ATTR: attributes.Schema(
_("Parent network of the subnet.")
),
TENANT_ID_ATTR: attributes.Schema(
_("Tenant owning the subnet.")
),
ALLOCATION_POOLS_ATTR: attributes.Schema(
_("Ip allocation pools and their ranges.")
),
GATEWAY_IP_ATTR: attributes.Schema(
_("Ip of the subnet's gateway.")
),
HOST_ROUTES_ATTR: attributes.Schema(
_("Additional routes for this subnet.")
),
IP_VERSION_ATTR: attributes.Schema(
_("Ip version for the subnet.")
),
CIDR_ATTR: attributes.Schema(
_("CIDR block notation for this subnet.")
),
DNS_NAMESERVERS_ATTR: attributes.Schema(
_("List of dns nameservers.")
),
ENABLE_DHCP_ATTR: attributes.Schema(
_("'true' if DHCP is enabled for this subnet; 'false' otherwise.")
),
SHOW: attributes.Schema(
_("All attributes.")
),
}
@classmethod
def _null_gateway_ip(cls, props):
if cls.GATEWAY_IP not in props:
return
# Specifying null in the gateway_ip will result in
# a property containing an empty string.
# A null gateway_ip has special meaning in the API
# so this needs to be set back to None.
# See bug https://bugs.launchpad.net/heat/+bug/1226666
if props.get(cls.GATEWAY_IP) == '':
props[cls.GATEWAY_IP] = None
def validate(self):
super(Subnet, self).validate()
self._validate_depr_property_required(self.properties,
self.NETWORK, self.NETWORK_ID)
ra_mode = self.properties[self.IPV6_RA_MODE]
address_mode = self.properties[self.IPV6_ADDRESS_MODE]
if (self.properties[self.IP_VERSION] == 4) and (
ra_mode or address_mode):
msg = _('ipv6_ra_mode and ipv6_address_mode are not supported '
'for ipv4.')
raise exception.StackValidationFailed(message=msg)
if ra_mode and address_mode and (ra_mode != address_mode):
msg = _('When both ipv6_ra_mode and ipv6_address_mode are set, '
'they must be equal.')
raise exception.StackValidationFailed(message=msg)
gateway_ip = self.properties.get(self.GATEWAY_IP)
if (gateway_ip and gateway_ip not in ['~', ''] and
not netutils.is_valid_ip(gateway_ip)):
msg = (_('Gateway IP address "%(gateway)" is in '
'invalid format.'), gateway_ip)
raise exception.StackValidationFailed(message=msg)
def handle_create(self):
props = self.prepare_properties(
self.properties,
self.physical_resource_name())
self.client_plugin().resolve_network(props, self.NETWORK, 'network_id')
self._null_gateway_ip(props)
subnet = self.neutron().create_subnet({'subnet': props})['subnet']
self.resource_id_set(subnet['id'])
def handle_delete(self):
client = self.neutron()
try:
client.delete_subnet(self.resource_id)
except Exception as ex:
self.client_plugin().ignore_not_found(ex)
else:
return True
def _show_resource(self):
return self.neutron().show_subnet(self.resource_id)['subnet']
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
props = self.prepare_update_properties(json_snippet)
self.neutron().update_subnet(
self.resource_id, {'subnet': props})
def resource_mapping():
return {
'OS::Neutron::Subnet': Subnet,
}