heat/heat/engine/resources/neutron/router.py

320 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 heat.common import exception
from heat.engine import clients
from heat.engine import properties
from heat.engine.resources.neutron import neutron
from heat.engine.resources.neutron import subnet
from heat.engine import support
from heat.openstack.common import log as logging
if clients.neutronclient is not None:
from neutronclient.common.exceptions import NeutronClientException
from neutronclient.neutron import v2_0 as neutronV20
logger = logging.getLogger(__name__)
class Router(neutron.NeutronResource):
PROPERTIES = (
NAME, EXTERNAL_GATEWAY, VALUE_SPECS, ADMIN_STATE_UP,
AGENT_ID,
) = (
'name', 'external_gateway_info', 'value_specs', 'admin_state_up',
'agent_id',
)
_EXTERNAL_GATEWAY_KEYS = (
EXTERNAL_GATEWAY_NETWORK, EXTERNAL_GATEWAY_ENABLE_SNAT,
) = (
'network', 'enable_snat',
)
properties_schema = {
NAME: properties.Schema(
properties.Schema.STRING,
_('The name of the router.'),
update_allowed=True
),
EXTERNAL_GATEWAY: properties.Schema(
properties.Schema.MAP,
_('External network gateway configuration for a router.'),
schema={
EXTERNAL_GATEWAY_NETWORK: properties.Schema(
properties.Schema.STRING,
_('ID or name of the external network for the gateway.'),
required=True,
update_allowed=True
),
EXTERNAL_GATEWAY_ENABLE_SNAT: properties.Schema(
properties.Schema.BOOLEAN,
_('Enables Source NAT on the router gateway.'),
default=True,
update_allowed=True
),
},
update_allowed=True
),
VALUE_SPECS: properties.Schema(
properties.Schema.MAP,
_('Extra parameters to include in the creation request.'),
default={},
update_allowed=True
),
ADMIN_STATE_UP: properties.Schema(
properties.Schema.BOOLEAN,
_('The administrative state of the router.'),
default=True,
update_allowed=True
),
AGENT_ID: properties.Schema(
properties.Schema.STRING,
_('ID of the L3 agent. NOTE: The default policy setting in '
'Neutron restricts usage of this property to administrative '
'users only.'),
update_allowed=True
),
}
attributes_schema = {
"status": _("The status of the router."),
"external_gateway_info": _("Gateway network for the router."),
"name": _("Friendly name of the router."),
"admin_state_up": _("Administrative state of the router."),
"tenant_id": _("Tenant owning the router."),
"show": _("All attributes."),
}
update_allowed_keys = ('Properties',)
def add_dependencies(self, deps):
super(Router, self).add_dependencies(deps)
external_gw = self.properties.get(self.EXTERNAL_GATEWAY)
if external_gw:
external_gw_net = external_gw.get(self.EXTERNAL_GATEWAY_NETWORK)
for res in self.stack.itervalues():
if res.has_interface('OS::Neutron::Subnet'):
subnet_net = res.properties.get(subnet.Subnet.NETWORK_ID)
if subnet_net == external_gw_net:
deps += (self, res)
def prepare_properties(self, properties, name):
props = super(Router, self).prepare_properties(properties, name)
gateway = props.get(self.EXTERNAL_GATEWAY)
if gateway:
gateway['network_id'] = neutronV20.find_resourceid_by_name_or_id(
self.neutron(),
'network',
gateway.pop(self.EXTERNAL_GATEWAY_NETWORK))
return props
def handle_create(self):
props = self.prepare_properties(
self.properties,
self.physical_resource_name())
agent_id = props.pop(self.AGENT_ID, None)
router = self.neutron().create_router({'router': props})['router']
self.resource_id_set(router['id'])
if agent_id:
self._replace_agent(agent_id)
def _show_resource(self):
return self.neutron().show_router(
self.resource_id)['router']
def check_create_complete(self, *args):
attributes = self._show_resource()
return self.is_built(attributes)
def handle_delete(self):
client = self.neutron()
try:
client.delete_router(self.resource_id)
except NeutronClientException as ex:
self._handle_not_found_exception(ex)
else:
return self._delete_task()
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
props = self.prepare_update_properties(json_snippet)
agent_id = props.pop(self.AGENT_ID, None)
if self.AGENT_ID in prop_diff:
self._replace_agent(agent_id)
del prop_diff[self.AGENT_ID]
if len(prop_diff) > 0:
self.neutron().update_router(
self.resource_id, {'router': props})
def _replace_agent(self, agent_id=None):
ret = self.neutron().list_l3_agent_hosting_routers(
self.resource_id)
for agent in ret['agents']:
self.neutron().remove_router_from_l3_agent(
agent['id'], self.resource_id)
if agent_id:
self.neutron().add_router_to_l3_agent(
agent_id, {'router_id': self.resource_id})
class RouterInterface(neutron.NeutronResource):
PROPERTIES = (
ROUTER_ID, SUBNET_ID, PORT_ID,
) = (
'router_id', 'subnet_id', 'port_id',
)
properties_schema = {
ROUTER_ID: properties.Schema(
properties.Schema.STRING,
_('The router id.'),
required=True
),
SUBNET_ID: properties.Schema(
properties.Schema.STRING,
_('The subnet id, either subnet_id or port_id should be '
'specified.')
),
PORT_ID: properties.Schema(
properties.Schema.STRING,
_('The port id, either subnet_id or port_id should be specified.')
),
}
def validate(self):
'''
Validate any of the provided params
'''
super(RouterInterface, self).validate()
subnet_id = self.properties.get(self.SUBNET_ID)
port_id = self.properties.get(self.PORT_ID)
if subnet_id and port_id:
raise exception.ResourcePropertyConflict(self.SUBNET_ID,
self.PORT_ID)
if not subnet_id and not port_id:
msg = 'Either subnet_id or port_id must be specified.'
raise exception.StackValidationFailed(message=msg)
def handle_create(self):
router_id = self.properties.get(self.ROUTER_ID)
key = self.SUBNET_ID
value = self.properties.get(key)
if not value:
key = self.PORT_ID
value = self.properties.get(key)
self.neutron().add_interface_router(
router_id,
{key: value})
self.resource_id_set('%s:%s=%s' % (router_id, key, value))
def handle_delete(self):
if not self.resource_id:
return
client = self.neutron()
tokens = self.resource_id.replace('=', ':').split(':')
if len(tokens) == 2: # compatible with old data
tokens.insert(1, 'subnet_id')
(router_id, key, value) = tokens
try:
client.remove_interface_router(
router_id,
{key: value})
except NeutronClientException as ex:
self._handle_not_found_exception(ex)
class RouterGateway(neutron.NeutronResource):
support_status = support.SupportStatus(
support.DEPRECATED,
_('RouterGateway resource is deprecated and should not be used. '
'Instead use the `external_gateway_info` property in the router '
'resource to set up the gateway.')
)
PROPERTIES = (
ROUTER_ID, NETWORK_ID,
) = (
'router_id', 'network_id',
)
properties_schema = {
ROUTER_ID: properties.Schema(
properties.Schema.STRING,
_('ID of the router.'),
required=True
),
NETWORK_ID: properties.Schema(
properties.Schema.STRING,
_('ID of the external network for the gateway.'),
required=True
),
}
def add_dependencies(self, deps):
super(RouterGateway, self).add_dependencies(deps)
for resource in self.stack.itervalues():
# depend on any RouterInterface in this template with the same
# router_id as this router_id
if (resource.has_interface('OS::Neutron::RouterInterface') and
resource.properties.get(RouterInterface.ROUTER_ID) ==
self.properties.get(self.ROUTER_ID)):
deps += (self, resource)
# depend on any subnet in this template with the same network_id
# as this network_id, as the gateway implicitly creates a port
# on that subnet
elif (resource.has_interface('OS::Neutron::Subnet') and
resource.properties.get(subnet.Subnet.NETWORK_ID) ==
self.properties.get(self.NETWORK_ID)):
deps += (self, resource)
def handle_create(self):
router_id = self.properties.get(self.ROUTER_ID)
network_id = neutronV20.find_resourceid_by_name_or_id(
self.neutron(),
'network',
self.properties.get(self.NETWORK_ID))
self.neutron().add_gateway_router(
router_id,
{'network_id': network_id})
self.resource_id_set('%s:%s' % (router_id, network_id))
def handle_delete(self):
if not self.resource_id:
return
client = self.neutron()
(router_id, network_id) = self.resource_id.split(':')
try:
client.remove_gateway_router(router_id)
except NeutronClientException as ex:
self._handle_not_found_exception(ex)
def resource_mapping():
if clients.neutronclient is None:
return {}
return {
'OS::Neutron::Router': Router,
'OS::Neutron::RouterInterface': RouterInterface,
'OS::Neutron::RouterGateway': RouterGateway,
}