Source code for pypuppetdb.api

from __future__ import unicode_literals
from __future__ import absolute_import

import logging

import json
import requests

from pypuppetdb.errors import (
    ImproperlyConfiguredError,
    EmptyResponseError,
    UnsupportedVersionError,
    APIError,
    )

log = logging.getLogger(__name__)

API_VERSIONS = {
    2: 'v2',
    3: 'v3',
}

ENDPOINTS = {
    2: {
        'facts': 'facts',
        'fact-names': 'fact-names',
        'nodes': 'nodes',
        'resources': 'resources',
        'metrics': 'metrics',
        'mbean': 'metrics/mbean',
    },
    3: {
        'facts': 'facts',
        'fact-names': 'fact-names',
        'nodes': 'nodes',
        'resources': 'resources',
        'catalogs': 'catalogs',
        'metrics': 'metrics',
        'mbean': 'metrics/mbean',
        'reports': 'reports',
        'events': 'events',
        'event-counts': 'event-counts',
        'aggregate-event-counts': 'aggregate-event-counts',
        'server-time': 'server-time',
        'version': 'version',
    },
}

ERROR_STRINGS = {
    'timeout': 'Connection to PuppetDB timed out on',
    'refused': 'Could not reach PuppetDB on',
}


[docs]class BaseAPI(object): """This is a Base or Abstract class and is not meant to be instantiated or used directly. The BaseAPI object defines a set of methods that can be reused across different versions of the PuppetDB API. If querying for a certain resource is done in an identical fashion across different versions it will be implemented here and should be overridden in their respective versions if they deviate. If :attr:`ssl` is set to `True` but either :attr:`ssl_key` or\ :attr:`ssl_cert` are `None` this will raise an error. When at initialisation :obj:`api_version` isn't found in\ :const:`API_VERSIONS` this will raise an error. :param api_version: Version of the API we're initialising. :type api_version: :obj:`int` :param host: (optional) Hostname or IP of PuppetDB. :type host: :obj:`string` :param port: (optional) Port on which to talk to PuppetDB. :type port: :obj:`int` :param ssl_verify: (optional) Verify PuppetDB server certificate. :type ssl_verify: :obj:`bool` :param ssl_key: (optional) Path to our client secret key. :type ssl_key: :obj:`None` or :obj:`string` representing a filesystem\ path. :param ssl_cert: (optional) Path to our client certificate. :type ssl_cert: :obj:`None` or :obj:`string` representing a filesystem\ path. :param timeout: (optional) Number of seconds to wait for a response. :type timeout: :obj:`int` :raises: :class:`~pypuppetdb.errors.ImproperlyConfiguredError` :raises: :class:`~pypuppetdb.errors.UnsupportedVersionError` """ def __init__(self, api_version, host='localhost', port=8080, ssl_verify=True, ssl_key=None, ssl_cert=None, timeout=10): """Initialises our BaseAPI object passing the parameters needed in order to be able to create the connection strings, set up SSL and timeouts and so forth.""" if api_version in API_VERSIONS: self.api_version = API_VERSIONS[api_version] else: raise UnsupportedVersionError self.host = host self.port = port self.ssl_verify = ssl_verify self.ssl_key = ssl_key self.ssl_cert = ssl_cert self.timeout = timeout self.endpoints = ENDPOINTS[api_version] if self.ssl_key is not None and self.ssl_cert is not None: self.protocol = 'https' else: self.protocol = 'http' @property
[docs] def version(self): """The version of the API we're querying against. :returns: Current API version. :rtype: :obj:`string`""" return self.api_version
@property
[docs] def base_url(self): """A base_url that will be used to construct the final URL we're going to query against. :returns: A URL of the form: ``proto://host:port``. :rtype: :obj:`string` """ return '{proto}://{host}:{port}'.format( proto=self.protocol, host=self.host, port=self.port, )
@property
[docs] def total(self): """The total-count of the last request to PuppetDB if enabled as parameter in _query method :returns Number of total results :rtype :obj:`int` """ if self.last_total is not None: return int(self.last_total)
[docs] def _url(self, endpoint, path=None): """The complete URL we will end up querying. Depending on the endpoint we pass in this will result in different URL's with different prefixes. :param endpoint: The PuppetDB API endpoint we want to query. :type endpoint: :obj:`string` :param path: An additional path if we don't wish to query the\ bare endpoint. :type path: :obj:`string` :returns: A URL constructed from :func:`base_url` with the\ apropraite API version/prefix and the rest of the path added\ to it. :rtype: :obj:`string` """ log.debug('_url called with endpoint: {0} and path: {1}'.format( endpoint, path)) if endpoint in self.endpoints: api_prefix = self.api_version endpoint = self.endpoints[endpoint] else: # If we reach this we're trying to query an endpoint that doesn't # exist. This shouldn't happen unless someone made a booboo. raise APIError url = '{base_url}/{api_prefix}/{endpoint}'.format( base_url=self.base_url, api_prefix=api_prefix, endpoint=endpoint, ) if path is not None: url = '{0}/{1}'.format(url, path) return url
[docs] def _query(self, endpoint, path=None, query=None, order_by=None, limit=None, offset=None, include_total=False, summarize_by=None, count_by=None, count_filter=None): """This method actually querries PuppetDB. Provided an endpoint and an optional path and/or query it will fire a request at PuppetDB. If PuppetDB can be reached and answers within the timeout we'll decode the response and give it back or raise for the HTTP Status Code PuppetDB gave back. :param endpoint: The PuppetDB API endpoint we want to query. :type endpoint: :obj:`string` :param path: An additional path if we don't wish to query the\ bare endpoint. :type path: :obj:`string` :param query: (optional) A query to further narrow down the resultset. :type query: :obj:`string` :param order_by: (optional) Set the order parameters for the resultset. :type order_by: :obj:`string` :param limit: (optional) Tell PuppetDB to limit it's response to this\ number of objects. :type limit: :obj:`int` :param offset: (optional) Tell PuppetDB to start it's response from\ the given offset. This is useful for implementing pagination\ but is not supported just yet. :type offset: :obj:`string` :param include_total: (optional) Include the total number of results :type order_by: :obj:`bool` :param summarize_by: (optional) Specify what type of object you'd like\ to see counts at the event-counts and aggregate-event-counts \ endpoints :type summarize_by: :obj:`string` :param count_by: (optional) Specify what type of object is counted :type count_by: :obj:`string` :param count_filter: (optional) Specify a filter for the results :type count_filter: :obj:`string` :raises: :class:`~pypuppetdb.errors.EmptyResponseError` :returns: The decoded response from PuppetDB :rtype: :obj:`dict` or :obj:`list` """ log.debug('_query called with endpoint: {0}, path: {1}, query: {2}, ' 'limit: {3}, offset: {4}, summarize_by {5}, count_by {6}, ' 'count_filter: {7}'.format(endpoint, path, query, limit, offset, summarize_by, count_by, count_filter)) url = self._url(endpoint, path=path) headers = { 'content-type': 'application/json', 'accept': 'application/json', 'accept-charset': 'utf-8' } payload = {} if query is not None: payload['query'] = query if order_by is not None: payload['order-by'] = order_by if limit is not None: payload['limit'] = limit if include_total is True: payload['include-total'] = json.dumps(include_total) if offset is not None: payload['offset'] = offset if summarize_by is not None: payload['summarize-by'] = summarize_by if count_by is not None: payload['count-by'] = count_by if count_filter is not None: payload['count-filter'] = count_filter if not (payload): payload = None try: r = requests.get(url, params=payload, headers=headers, verify=self.ssl_verify, cert=(self.ssl_cert, self.ssl_key), timeout=self.timeout) r.raise_for_status() # get total number of results if requested with include-total # just a quick hack - needs improvement if 'X-Records' in r.headers: self.last_total = r.headers['X-Records'] else: self.last_total = None json_body = r.json() if json_body is not None: return json_body else: del json_body raise EmptyResponseError except requests.exceptions.Timeout: log.error("{0} {1}:{2} over {3}.".format(ERROR_STRINGS['timeout'], self.host, self.port, self.protocol.upper())) raise except requests.exceptions.ConnectionError: log.error("{0} {1}:{2} over {3}.".format(ERROR_STRINGS['refused'], self.host, self.port, self.protocol.upper())) raise except requests.exceptions.HTTPError as err: log.error("{0} {1}:{2} over {3}.".format(err.response.text, self.host, self.port, self.protocol.upper())) raise # Method stubs
def nodes(self): raise NotImplementedError def node(self): raise NotImplementedError def facts(self): raise NotImplementedError def resources(self): raise NotImplementedError
[docs] def metric(self, metric): """Query for a specific metrc. :param metric: The name of the metric we want. :type metric: :obj:`string` :returns: The return of :meth:`~pypuppetdb.api.BaseAPI._query`. """ return self._query('mbean', path=metric)
Read the Docs v: v0.1.0
Versions
latest
v0.1.0
v0.0.4
Downloads
On Read the Docs
Project Home
Builds

Free document hosting provided by Read the Docs.