¡@

Home 

OpenStack Study: cisco_csr_rest_client.py

OpenStack Index

**** CubicPower OpenStack Study ****

# Copyright 2014 Cisco Systems, Inc. All rights reserved.

#

# 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.

#

# @author: Paul Michali, Cisco Systems, Inc.

import time

import netaddr

import requests

from requests import exceptions as r_exc

from neutron.openstack.common import jsonutils

from neutron.openstack.common import log as logging

TIMEOUT = 20.0

LOG = logging.getLogger(__name__)

HEADER_CONTENT_TYPE_JSON = {'content-type': 'application/json'}

URL_BASE = 'https://%(host)s/api/v1/%(resource)s'

**** CubicPower OpenStack Study ****

def make_route_id(cidr, interface):

    """Build ID that will be used to identify route for later deletion."""

    net = netaddr.IPNetwork(cidr)

    return '%(network)s_%(prefix)s_%(interface)s' % {

        'network': net.network,

        'prefix': net.prefixlen,

        'interface': interface}

**** CubicPower OpenStack Study ****

class CsrRestClient(object):

"""REST CsrRestClient for accessing the Cisco Cloud Services Router."""

**** CubicPower OpenStack Study ****

    def __init__(self, host, tunnel_ip, username, password, timeout=None):

        self.host = host

        self.tunnel_ip = tunnel_ip

        self.auth = (username, password)

        self.token = None

        self.status = requests.codes.OK

        self.timeout = timeout

        self.max_tries = 5

        self.session = requests.Session()

**** CubicPower OpenStack Study ****

    def _response_info_for(self, response, method):

        """Return contents or location from response.

        For a POST or GET with a 200 response, the response content

        is returned.

        For a POST with a 201 response, return the header's location,

        which contains the identifier for the created resource.

        If there is an error, return the response content, so that

        it can be used in error processing ('error-code', 'error-message',

        and 'detail' fields).

        """

        if method in ('POST', 'GET') and self.status == requests.codes.OK:

            LOG.debug(_('RESPONSE: %s'), response.json())

            return response.json()

        if method == 'POST' and self.status == requests.codes.CREATED:

            return response.headers.get('location', '')

        if self.status >= requests.codes.BAD_REQUEST and response.content:

            if 'error-code' in response.content:

                content = jsonutils.loads(response.content)

                LOG.debug("Error response content %s", content)

                return content

**** CubicPower OpenStack Study ****

    def _request(self, method, url, **kwargs):

        """Perform REST request and save response info."""

        try:

            LOG.debug(_("%(method)s: Request for %(resource)s payload: "

                        "%(payload)s"),

                      {'method': method.upper(), 'resource': url,

                       'payload': kwargs.get('data')})

            start_time = time.time()

            response = self.session.request(method, url, verify=False,

                                            timeout=self.timeout, **kwargs)

            LOG.debug(_("%(method)s Took %(time).2f seconds to process"),

                      {'method': method.upper(),

                       'time': time.time() - start_time})

        except (r_exc.Timeout, r_exc.SSLError) as te:

            # Should never see SSLError, unless requests package is old (<2.0)

            timeout_val = 0.0 if self.timeout is None else self.timeout

            LOG.warning(_("%(method)s: Request timeout%(ssl)s "

                          "(%(timeout).3f sec) for CSR(%(host)s)"),

                        {'method': method,

                         'timeout': timeout_val,

                         'ssl': '(SSLError)'

                         if isinstance(te, r_exc.SSLError) else '',

                         'host': self.host})

            self.status = requests.codes.REQUEST_TIMEOUT

        except r_exc.ConnectionError:

            LOG.exception(_("%(method)s: Unable to connect to CSR(%(host)s)"),

                          {'method': method, 'host': self.host})

            self.status = requests.codes.NOT_FOUND

        except Exception as e:

            LOG.error(_("%(method)s: Unexpected error for CSR (%(host)s): "

                        "%(error)s"),

                      {'method': method, 'host': self.host, 'error': e})

            self.status = requests.codes.INTERNAL_SERVER_ERROR

        else:

            self.status = response.status_code

            LOG.debug(_("%(method)s: Completed [%(status)s]"),

                      {'method': method, 'status': self.status})

            return self._response_info_for(response, method)

**** CubicPower OpenStack Study ****

    def authenticate(self):

        """Obtain a token to use for subsequent CSR REST requests.

        This is called when there is no token yet, or if the token has expired

        and attempts to use it resulted in an UNAUTHORIZED REST response.

        """

        url = URL_BASE % {'host': self.host, 'resource': 'auth/token-services'}

        headers = {'Content-Length': '0',

                   'Accept': 'application/json'}

        headers.update(HEADER_CONTENT_TYPE_JSON)

        LOG.debug(_("%(auth)s with CSR %(host)s"),

                  {'auth': 'Authenticating' if self.token is None

                   else 'Reauthenticating', 'host': self.host})

        self.token = None

        response = self._request("POST", url, headers=headers, auth=self.auth)

        if response:

            self.token = response['token-id']

            LOG.debug(_("Successfully authenticated with CSR %s"), self.host)

            return True

        LOG.error(_("Failed authentication with CSR %(host)s [%(status)s]"),

                  {'host': self.host, 'status': self.status})

**** CubicPower OpenStack Study ****

    def _do_request(self, method, resource, payload=None, more_headers=None,

                    full_url=False):

        """Perform a REST request to a CSR resource.

        If this is the first time interacting with the CSR, a token will

        be obtained. If the request fails, due to an expired token, the

        token will be obtained and the request will be retried once more.

        """

        if self.token is None:

            if not self.authenticate():

                return

        if full_url:

            url = resource

        else:

            url = ('https://%(host)s/api/v1/%(resource)s' %

                   {'host': self.host, 'resource': resource})

        headers = {'Accept': 'application/json', 'X-auth-token': self.token}

        if more_headers:

            headers.update(more_headers)

        if payload:

            payload = jsonutils.dumps(payload)

        response = self._request(method, url, data=payload, headers=headers)

        if self.status == requests.codes.UNAUTHORIZED:

            if not self.authenticate():

                return

            headers['X-auth-token'] = self.token

            response = self._request(method, url, data=payload,

                                     headers=headers)

        if self.status != requests.codes.REQUEST_TIMEOUT:

            return response

        LOG.error(_("%(method)s: Request timeout for CSR(%(host)s)"),

                  {'method': method, 'host': self.host})

**** CubicPower OpenStack Study ****

    def get_request(self, resource, full_url=False):

        """Perform a REST GET requests for a CSR resource."""

        return self._do_request('GET', resource, full_url=full_url)

**** CubicPower OpenStack Study ****

    def post_request(self, resource, payload=None):

        """Perform a POST request to a CSR resource."""

        return self._do_request('POST', resource, payload=payload,

                                more_headers=HEADER_CONTENT_TYPE_JSON)

**** CubicPower OpenStack Study ****

    def put_request(self, resource, payload=None):

        """Perform a PUT request to a CSR resource."""

        return self._do_request('PUT', resource, payload=payload,

                                more_headers=HEADER_CONTENT_TYPE_JSON)

**** CubicPower OpenStack Study ****

    def delete_request(self, resource):

        """Perform a DELETE request on a CSR resource."""

        return self._do_request('DELETE', resource,

                                more_headers=HEADER_CONTENT_TYPE_JSON)

**** CubicPower OpenStack Study ****

    def create_ike_policy(self, policy_info):

        base_ike_policy_info = {u'version': u'v1',

                                u'local-auth-method': u'pre-share'}

        base_ike_policy_info.update(policy_info)

        return self.post_request('vpn-svc/ike/policies',

                                 payload=base_ike_policy_info)

**** CubicPower OpenStack Study ****

    def create_ipsec_policy(self, policy_info):

        base_ipsec_policy_info = {u'mode': u'tunnel'}

        base_ipsec_policy_info.update(policy_info)

        return self.post_request('vpn-svc/ipsec/policies',

                                 payload=base_ipsec_policy_info)

**** CubicPower OpenStack Study ****

    def create_pre_shared_key(self, psk_info):

        return self.post_request('vpn-svc/ike/keyrings', payload=psk_info)

**** CubicPower OpenStack Study ****

    def create_ipsec_connection(self, connection_info):

        base_conn_info = {u'vpn-type': u'site-to-site',

                          u'ip-version': u'ipv4'}

        connection_info.update(base_conn_info)

        # TODO(pcm) pass in value, when CSR is embedded as Neutron router.

        # Currently, get this from .INI file.

        connection_info[u'local-device'][u'tunnel-ip-address'] = self.tunnel_ip

        return self.post_request('vpn-svc/site-to-site',

                                 payload=connection_info)

**** CubicPower OpenStack Study ****

    def configure_ike_keepalive(self, keepalive_info):

        base_keepalive_info = {u'periodic': True}

        keepalive_info.update(base_keepalive_info)

        return self.put_request('vpn-svc/ike/keepalive', keepalive_info)

**** CubicPower OpenStack Study ****

    def create_static_route(self, route_info):

        return self.post_request('routing-svc/static-routes',

                                 payload=route_info)

**** CubicPower OpenStack Study ****

    def delete_static_route(self, route_id):

        return self.delete_request('routing-svc/static-routes/%s' % route_id)

**** CubicPower OpenStack Study ****

    def delete_ipsec_connection(self, conn_id):

        return self.delete_request('vpn-svc/site-to-site/%s' % conn_id)

**** CubicPower OpenStack Study ****

    def delete_ipsec_policy(self, policy_id):

        return self.delete_request('vpn-svc/ipsec/policies/%s' % policy_id)

**** CubicPower OpenStack Study ****

    def delete_ike_policy(self, policy_id):

        return self.delete_request('vpn-svc/ike/policies/%s' % policy_id)

**** CubicPower OpenStack Study ****

    def delete_pre_shared_key(self, key_id):

        return self.delete_request('vpn-svc/ike/keyrings/%s' % key_id)

**** CubicPower OpenStack Study ****

    def read_tunnel_statuses(self):

        results = self.get_request('vpn-svc/site-to-site/active/sessions')

        if self.status != requests.codes.OK or not results:

            return []

        tunnels = [(t[u'vpn-interface-name'], t[u'status'])

                   for t in results['items']]

        return tunnels