diff --git a/datadog/__init__.py b/datadog/__init__.py index 4d02cf9c1..cb4fb60e4 100644 --- a/datadog/__init__.py +++ b/datadog/__init__.py @@ -8,7 +8,6 @@ * datadog.dogshell: a command-line tool, wrapping datadog.api, to interact with Datadog REST API. """ # stdlib -from pkg_resources import get_distribution, DistributionNotFound import os import os.path @@ -16,22 +15,12 @@ from datadog import api from datadog.dogstatsd import DogStatsd, statsd # noqa from datadog.threadstats import ThreadStats # noqa -from datadog.util.hostname import get_hostname from datadog.util.compat import iteritems +from datadog.util.config import get_version +from datadog.util.hostname import get_hostname -try: - _dist = get_distribution("datadog") - # Normalize case for Windows systems - dist_loc = os.path.normcase(_dist.location) - here = os.path.normcase(__file__) - if not here.startswith(os.path.join(dist_loc, __name__)): - # not installed, but there is another version that *is*e - raise DistributionNotFound -except DistributionNotFound: - __version__ = 'Please install datadog with setup.py' -else: - __version__ = _dist.version +__version__ = get_version() def initialize(api_key=None, app_key=None, host_name=None, api_host=None, diff --git a/datadog/api/api_client.py b/datadog/api/api_client.py new file mode 100644 index 000000000..335bd22fc --- /dev/null +++ b/datadog/api/api_client.py @@ -0,0 +1,223 @@ +# stdlib +import logging +import time + +# datadog +from datadog.api import _api_version, _max_timeouts, _backoff_period +from datadog.api.exceptions import ClientError, ApiError, HttpBackoff, \ + HttpTimeout, ApiNotInitialized +from datadog.api.http_client import resolve_http_client +from datadog.util.compat import json, is_p3k + + +log = logging.getLogger('dd.datadogpy') + + +class APIClient(object): + """ + Datadog API client: format and submit API calls to Datadog. + Embeds a HTTP client. + """ + # HTTP transport parameters + _backoff_period = _backoff_period + _max_timeouts = _max_timeouts + _backoff_timestamp = None + _timeout_counter = 0 + _api_version = _api_version + + # Plugged HTTP client + _http_client = None + + @classmethod + def _get_http_client(cls): + """ + Getter for the embedded HTTP client. + """ + if not cls._http_client: + cls._http_client = resolve_http_client() + + return cls._http_client + + @classmethod + def submit(cls, method, path, body=None, attach_host_name=False, response_formatter=None, + error_formatter=None, **params): + """ + Make an HTTP API request + + :param method: HTTP method to use to contact API endpoint + :type method: HTTP method string + + :param path: API endpoint url + :type path: url + + :param body: dictionary to be sent in the body of the request + :type body: dictionary + + :param response_formatter: function to format JSON response from HTTP API request + :type response_formatter: JSON input function + + :param error_formatter: function to format JSON error response from HTTP API request + :type error_formatter: JSON input function + + :param attach_host_name: link the new resource object to the host name + :type attach_host_name: bool + + :param params: dictionary to be sent in the query string of the request + :type params: dictionary + + :returns: JSON or formated response from HTTP API request + """ + try: + # Check if it's ok to submit + if not cls._should_submit(): + _, backoff_time_left = cls._backoff_status() + raise HttpBackoff(backoff_time_left) + + # Import API, User and HTTP settings + from datadog.api import _api_key, _application_key, _api_host, \ + _mute, _host_name, _proxies, _max_retries, _timeout, \ + _cacert + + # Check keys and add then to params + if _api_key is None: + raise ApiNotInitialized("API key is not set." + " Please run 'initialize' method first.") + params['api_key'] = _api_key + if _application_key: + params['application_key'] = _application_key + + # Attach host name to body + if attach_host_name and body: + # Is it a 'series' list of objects ? + if 'series' in body: + # Adding the host name to all objects + for obj_params in body['series']: + if obj_params.get('host', "") == "": + obj_params['host'] = _host_name + else: + if body.get('host', "") == "": + body['host'] = _host_name + + # If defined, make sure tags are defined as a comma-separated string + if 'tags' in params and isinstance(params['tags'], list): + params['tags'] = ','.join(params['tags']) + + # Process the body, if necessary + headers = {} + if isinstance(body, dict): + body = json.dumps(body) + headers['Content-Type'] = 'application/json' + + # Construct the URL + url = "{api_host}/api/{api_version}/{path}".format( + api_host=_api_host, + api_version=cls._api_version, + path=path.lstrip("/"), + ) + + # Process requesting + start_time = time.time() + + result = cls._get_http_client().request( + method=method, url=url, + headers=headers, params=params, data=body, + timeout=_timeout, max_retries=_max_retries, + proxies=_proxies, verify=_cacert + ) + + # Request succeeded: log it and reset the timeout counter + duration = round((time.time() - start_time) * 1000., 4) + log.info("%s %s %s (%sms)" % (result.status_code, method, url, duration)) + cls._timeout_counter = 0 + + # Format response content + content = result.content + + if content: + try: + if is_p3k(): + response_obj = json.loads(content.decode('utf-8')) + else: + response_obj = json.loads(content) + except ValueError: + raise ValueError('Invalid JSON response: {0}'.format(content)) + + if response_obj and 'errors' in response_obj: + raise ApiError(response_obj) + else: + response_obj = None + if response_formatter is None: + return response_obj + else: + return response_formatter(response_obj) + + except HttpTimeout: + cls._timeout_counter += 1 + raise + except ClientError as e: + if _mute: + log.error(str(e)) + if error_formatter is None: + return {'errors': e.args[0]} + else: + return error_formatter({'errors': e.args[0]}) + else: + raise + except ApiError as e: + if _mute: + for error in e.args[0]['errors']: + log.error(str(error)) + if error_formatter is None: + return e.args[0] + else: + return error_formatter(e.args[0]) + else: + raise + + @classmethod + def _should_submit(cls): + """ + Returns True if we're in a state where we should make a request + (backoff expired, no backoff in effect), false otherwise. + """ + now = time.time() + should_submit = False + + # If we're not backing off, but the timeout counter exceeds the max + # number of timeouts, then enter the backoff state, recording the time + # we started backing off + if not cls._backoff_timestamp and cls._timeout_counter >= cls._max_timeouts: + log.info("Max number of datadog timeouts exceeded, backing off for {0} seconds" + .format(cls._backoff_period)) + cls._backoff_timestamp = now + should_submit = False + + # If we are backing off but the we've waiting sufficiently long enough + # (backoff_retry_age), exit the backoff state and reset the timeout + # counter so that we try submitting metrics again + elif cls._backoff_timestamp: + backed_off_time, backoff_time_left = cls._backoff_status() + if backoff_time_left < 0: + log.info("Exiting backoff state after {0} seconds, will try to submit metrics again" + .format(backed_off_time)) + cls._backoff_timestamp = None + cls._timeout_counter = 0 + should_submit = True + else: + log.info("In backoff state, won't submit metrics for another {0} seconds" + .format(backoff_time_left)) + should_submit = False + else: + should_submit = True + + return should_submit + + @classmethod + def _backoff_status(cls): + """ + Get a backoff report, i.e. backoff total and remaining time. + """ + now = time.time() + backed_off_time = now - cls._backoff_timestamp + backoff_time_left = cls._backoff_period - backed_off_time + return round(backed_off_time, 2), round(backoff_time_left, 2) diff --git a/datadog/api/base.py b/datadog/api/base.py deleted file mode 100644 index 839f3f018..000000000 --- a/datadog/api/base.py +++ /dev/null @@ -1,442 +0,0 @@ -# stdlib -import time -import logging -import requests - -# 3p -import simplejson as json - -# datadog -from datadog.api.exceptions import ClientError, ApiError, HttpBackoff, \ - HttpTimeout, ApiNotInitialized -from datadog.api import _api_version, _max_timeouts, _backoff_period -from datadog.util.compat import is_p3k - -log = logging.getLogger('dd.datadogpy') - - -class HTTPClient(object): - """ - HTTP client based on Requests library for Datadog API calls - """ - # http transport params - _backoff_period = _backoff_period - _max_timeouts = _max_timeouts - _backoff_timestamp = None - _timeout_counter = 0 - _api_version = _api_version - - @classmethod - def request(cls, method, path, body=None, attach_host_name=False, response_formatter=None, - error_formatter=None, **params): - """ - Make an HTTP API request - - :param method: HTTP method to use to contact API endpoint - :type method: HTTP method string - - :param path: API endpoint url - :type path: url - - :param body: dictionary to be sent in the body of the request - :type body: dictionary - - :param response_formatter: function to format JSON response from HTTP API request - :type response_formatter: JSON input function - - :param error_formatter: function to format JSON error response from HTTP API request - :type error_formatter: JSON input function - - :param attach_host_name: link the new resource object to the host name - :type attach_host_name: bool - - :param params: dictionary to be sent in the query string of the request - :type params: dictionary - - :returns: JSON or formated response from HTTP API request - """ - - try: - # Check if it's ok to submit - if not cls._should_submit(): - raise HttpBackoff("Too many timeouts. Won't try again for {1} seconds." - .format(*cls._backoff_status())) - - # Import API, User and HTTP settings - from datadog.api import _api_key, _application_key, _api_host, \ - _mute, _host_name, _proxies, _max_retries, _timeout, \ - _cacert - - # Check keys and add then to params - if _api_key is None: - raise ApiNotInitialized("API key is not set." - " Please run 'initialize' method first.") - params['api_key'] = _api_key - if _application_key: - params['application_key'] = _application_key - - # Construct the url - url = "%s/api/%s/%s" % (_api_host, cls._api_version, path.lstrip("/")) - - # Attach host name to body - if attach_host_name and body: - # Is it a 'series' list of objects ? - if 'series' in body: - # Adding the host name to all objects - for obj_params in body['series']: - if obj_params.get('host', "") == "": - obj_params['host'] = _host_name - else: - if body.get('host', "") == "": - body['host'] = _host_name - - # If defined, make sure tags are defined as a comma-separated string - if 'tags' in params and isinstance(params['tags'], list): - params['tags'] = ','.join(params['tags']) - - # Process the body, if necessary - headers = {} - if isinstance(body, dict): - body = json.dumps(body) - headers['Content-Type'] = 'application/json' - - # Process requesting - start_time = time.time() - try: - # Use a session to set a max_retries parameters - s = requests.Session() - http_adapter = requests.adapters.HTTPAdapter(max_retries=_max_retries) - s.mount('https://', http_adapter) - - # Request - result = s.request( - method, - url, - headers=headers, - params=params, - data=body, - timeout=_timeout, - proxies=_proxies, - verify=_cacert) - - result.raise_for_status() - except requests.ConnectionError as e: - raise ClientError("Could not request %s %s%s: %s" % (method, _api_host, url, e)) - except requests.exceptions.Timeout as e: - cls._timeout_counter += 1 - raise HttpTimeout('%s %s timed out after %d seconds.' % (method, url, _timeout)) - except requests.exceptions.HTTPError as e: - if e.response.status_code in (400, 403, 404, 409): - # This gets caught afterwards and raises an ApiError exception - pass - else: - raise - except TypeError as e: - raise TypeError( - "Your installed version of 'requests' library seems not compatible with" - "Datadog's usage. We recommand upgrading it ('pip install -U requests')." - "If you need help or have any question, please contact support@datadoghq.com") - - # Request succeeded: log it and reset the timeout counter - duration = round((time.time() - start_time) * 1000., 4) - log.info("%s %s %s (%sms)" % (result.status_code, method, url, duration)) - cls._timeout_counter = 0 - - # Format response content - content = result.content - - if content: - try: - if is_p3k(): - response_obj = json.loads(content.decode('utf-8')) - else: - response_obj = json.loads(content) - except ValueError: - raise ValueError('Invalid JSON response: {0}'.format(content)) - - if response_obj and 'errors' in response_obj: - raise ApiError(response_obj) - else: - response_obj = None - if response_formatter is None: - return response_obj - else: - return response_formatter(response_obj) - - except ClientError as e: - if _mute: - log.error(str(e)) - if error_formatter is None: - return {'errors': e.args[0]} - else: - return error_formatter({'errors': e.args[0]}) - else: - raise - except ApiError as e: - if _mute: - for error in e.args[0]['errors']: - log.error(str(error)) - if error_formatter is None: - return e.args[0] - else: - return error_formatter(e.args[0]) - else: - raise - - # Private functions - @classmethod - def _should_submit(cls): - """ Returns True if we're in a state where we should make a request - (backoff expired, no backoff in effect), false otherwise. - """ - now = time.time() - should_submit = False - - # If we're not backing off, but the timeout counter exceeds the max - # number of timeouts, then enter the backoff state, recording the time - # we started backing off - if not cls._backoff_timestamp and cls._timeout_counter >= cls._max_timeouts: - log.info("Max number of datadog timeouts exceeded, backing off for {0} seconds" - .format(cls._backoff_period)) - cls._backoff_timestamp = now - should_submit = False - - # If we are backing off but the we've waiting sufficiently long enough - # (backoff_retry_age), exit the backoff state and reset the timeout - # counter so that we try submitting metrics again - elif cls._backoff_timestamp: - backed_off_time, backoff_time_left = cls._backoff_status() - if backoff_time_left < 0: - log.info("Exiting backoff state after {0} seconds, will try to submit metrics again" - .format(backed_off_time)) - cls._backoff_timestamp = None - cls._timeout_counter = 0 - should_submit = True - else: - log.info("In backoff state, won't submit metrics for another {0} seconds" - .format(backoff_time_left)) - should_submit = False - else: - should_submit = True - - return should_submit - - @classmethod - def _backoff_status(cls): - now = time.time() - backed_off_time = now - cls._backoff_timestamp - backoff_time_left = cls._backoff_period - backed_off_time - return round(backed_off_time, 2), round(backoff_time_left, 2) - - -# API Resource types are listed below -class CreateableAPIResource(object): - """ - Creatable API Resource - """ - @classmethod - def create(cls, attach_host_name=False, method='POST', id=None, params=None, **body): - """ - Create a new API resource object - - :param attach_host_name: link the new resource object to the host name - :type attach_host_name: bool - - :param method: HTTP method to use to contact API endpoint - :type method: HTTP method string - - :param id: create a new resource object as a child of the given object - :type id: id - - :param params: new resource object source - :type params: dictionary - - :param body: new resource object attributes - :type body: dictionary - - :returns: JSON response from HTTP API request - """ - if params is None: - params = {} - if method == 'GET': - return HTTPClient.request('GET', cls._class_url, **body) - if id is None: - return HTTPClient.request('POST', cls._class_url, body, - attach_host_name=attach_host_name, **params) - else: - return HTTPClient.request('POST', cls._class_url + "/" + str(id), body, - attach_host_name=attach_host_name, **params) - - -class SendableAPIResource(object): - """ - Fork of CreateableAPIResource class with different method names - """ - @classmethod - def send(cls, attach_host_name=False, id=None, **body): - """ - Create an API resource object - - :param attach_host_name: link the new resource object to the host name - :type attach_host_name: bool - - :param id: create a new resource object as a child of the given object - :type id: id - - :param body: new resource object attributes - :type body: dictionary - - :returns: JSON response from HTTP API request - """ - if id is None: - return HTTPClient.request('POST', cls._class_url, body, - attach_host_name=attach_host_name) - else: - return HTTPClient.request('POST', cls._class_url + "/" + str(id), body, - attach_host_name=attach_host_name) - - -class UpdatableAPIResource(object): - """ - Updatable API Resource - """ - @classmethod - def update(cls, id, params=None, **body): - """ - Update an API resource object - - :param params: updated resource object source - :type params: dictionary - - :param body: updated resource object attributes - :type body: dictionary - - :returns: JSON response from HTTP API request - """ - if params is None: - params = {} - return HTTPClient.request('PUT', cls._class_url + "/" + str(id), body, **params) - - -class DeletableAPIResource(object): - """ - Deletable API Resource - """ - @classmethod - def delete(cls, id, **params): - """ - Delete an API resource object - - :param id: resource object to delete - :type id: id - - :returns: JSON response from HTTP API request - """ - return HTTPClient.request('DELETE', cls._class_url + "/" + str(id), **params) - - -class GetableAPIResource(object): - """ - Getable API Resource - """ - @classmethod - def get(cls, id, **params): - """ - Get information about an API resource object - - :param id: resource object id to retrieve - :type id: id - - :param params: parameters to filter API resource stream - :type params: dictionary - - :returns: JSON response from HTTP API request - """ - return HTTPClient.request('GET', cls._class_url + "/" + str(id), **params) - - -class ListableAPIResource(object): - """ - Listable API Resource - """ - @classmethod - def get_all(cls, **params): - """ - List API resource objects - - :param params: parameters to filter API resource stream - :type params: dictionary - - :returns: JSON response from HTTP API request - """ - return HTTPClient.request('GET', cls._class_url, **params) - - -class SearchableAPIResource(object): - """ - Fork of ListableAPIResource class with different method names - """ - @classmethod - def _search(cls, **params): - """ - Query an API resource stream - - :param params: parameters to filter API resource stream - :type params: dictionary - - :returns: JSON response from HTTP API request - """ - return HTTPClient.request('GET', cls._class_url, **params) - - -class ActionAPIResource(object): - """ - Actionable API Resource - """ - @classmethod - def _trigger_class_action(cls, method, name, id=None, **params): - """ - Trigger an action - - :param method: HTTP method to use to contact API endpoint - :type method: HTTP method string - - :param name: action name - :type name: string - - :param id: trigger the action for the specified resource object - :type id: id - - :param params: action parameters - :type params: dictionary - - :returns: JSON response from HTTP API request - """ - if id is None: - return HTTPClient.request(method, cls._class_url + "/" + name, params) - else: - return HTTPClient.request(method, cls._class_url + "/" + str(id) + "/" + name, params) - - @classmethod - def _trigger_action(cls, method, name, id=None, **params): - """ - Trigger an action - - :param method: HTTP method to use to contact API endpoint - :type method: HTTP method string - - :param name: action name - :type name: string - - :param id: trigger the action for the specified resource object - :type id: id - - :param params: action parameters - :type params: dictionary - - :returns: JSON response from HTTP API request - """ - if id is None: - return HTTPClient.request(method, name, params) - else: - return HTTPClient.request(method, name + "/" + str(id), params) diff --git a/datadog/api/comments.py b/datadog/api/comments.py index f0f4c01d3..39b9e4fc3 100644 --- a/datadog/api/comments.py +++ b/datadog/api/comments.py @@ -1,4 +1,4 @@ -from datadog.api.base import CreateableAPIResource, UpdatableAPIResource, \ +from datadog.api.resources import CreateableAPIResource, UpdatableAPIResource, \ DeletableAPIResource diff --git a/datadog/api/downtimes.py b/datadog/api/downtimes.py index 9af703feb..82a84fd63 100644 --- a/datadog/api/downtimes.py +++ b/datadog/api/downtimes.py @@ -1,4 +1,4 @@ -from datadog.api.base import GetableAPIResource, CreateableAPIResource,\ +from datadog.api.resources import GetableAPIResource, CreateableAPIResource,\ UpdatableAPIResource, ListableAPIResource, DeletableAPIResource diff --git a/datadog/api/events.py b/datadog/api/events.py index c3569d15b..831d0b8d6 100644 --- a/datadog/api/events.py +++ b/datadog/api/events.py @@ -1,6 +1,6 @@ -from datadog.util.compat import iteritems -from datadog.api.base import GetableAPIResource, CreateableAPIResource, \ +from datadog.api.resources import GetableAPIResource, CreateableAPIResource, \ SearchableAPIResource +from datadog.util.compat import iteritems class Event(GetableAPIResource, CreateableAPIResource, SearchableAPIResource): diff --git a/datadog/api/exceptions.py b/datadog/api/exceptions.py index 82dfd27fb..b0d8e26c7 100644 --- a/datadog/api/exceptions.py +++ b/datadog/api/exceptions.py @@ -1,34 +1,67 @@ -""" Module containing all the possible exceptions that datadog can raise. """ -__all__ = [ - 'DatadogException', - 'ClientError', - 'HttpTimeout', - 'HttpBackoff', - 'ApiError', - 'ApiNotInitialized', -] +API & HTTP Clients exceptions. +""" + + +class ClientError(Exception): + """ + HTTP connection to Datadog endpoint is not possible. + """ + def __init__(self, method, url, exception): + message = u"Could not request {method} {url}: {exception}. "\ + u"Please check the network connection or try again later. "\ + u"If the problem persists, please contact support@datadoghq.com".format( + method=method, url=url, exception=exception + ) + super(ClientError, self).__init__(message) -class DatadogException(Exception): - pass +class HttpTimeout(Exception): + """ + HTTP connection timeout. + """ + def __init__(self, method, url, timeout): + message = u"{method} {url} timed out after {timeout}. "\ + u"Please try again later. "\ + u"If the problem persists, please contact support@datadoghq.com".format( + method=method, url=url, timeout=timeout + ) + super(HttpTimeout, self).__init__(message) -class ClientError(DatadogException): - "When HTTP connection to Datadog endpoint is not possible" +class HttpBackoff(Exception): + """ + Backing off after too many timeouts. + """ + def __init__(self, backoff_period): + message = u"Too many timeouts. Won't try again for {backoff_period} seconds. ".format( + backoff_period=backoff_period) + super(HttpBackoff, self).__init__(message) -class HttpTimeout(DatadogException): - "HTTP connection timeout" +class HTTPError(Exception): + """ + Datadog returned a HTTP error. + """ + def __init__(self, status_code=None, reason=None): + reason = u" - {reason}".format(reason=reason) if reason else u"" + message = u"Datadog returned a bad HTTP response code: {status_code}{reason}. "\ + u"Please try again later. "\ + u"If the problem persists, please contact support@datadoghq.com".format( + status_code=status_code, + reason=reason, + ) + super(HTTPError, self).__init__(message) -class HttpBackoff(DatadogException): - "Backing off after too many timeouts" +class ApiError(Exception): + """ + Datadog returned an API error (known HTTPError). -class ApiError(DatadogException): - "Datadog API is returning an error" + Matches the following status codes: 400, 403, 404, 409. + """ -class ApiNotInitialized(DatadogException): +class ApiNotInitialized(Exception): "No API key is set" diff --git a/datadog/api/graphs.py b/datadog/api/graphs.py index 9a247f7d8..d938f0fe2 100644 --- a/datadog/api/graphs.py +++ b/datadog/api/graphs.py @@ -1,5 +1,5 @@ from datadog.util.compat import urlparse -from datadog.api.base import ( +from datadog.api.resources import ( CreateableAPIResource, ActionAPIResource, GetableAPIResource, diff --git a/datadog/api/hosts.py b/datadog/api/hosts.py index f14432c1f..90eab3d97 100644 --- a/datadog/api/hosts.py +++ b/datadog/api/hosts.py @@ -1,4 +1,4 @@ -from datadog.api.base import ActionAPIResource +from datadog.api.resources import ActionAPIResource class Host(ActionAPIResource): diff --git a/datadog/api/http_client.py b/datadog/api/http_client.py new file mode 100644 index 000000000..2b67829c0 --- /dev/null +++ b/datadog/api/http_client.py @@ -0,0 +1,164 @@ +""" +Available HTTP Client for Datadog API client. + +Priority: +1. `requests` 3p module +2. `urlfetch` 3p module - Google App Engine only +""" +# stdlib +import logging +import urllib + +# 3p +try: + import requests +except ImportError: + requests = None + +try: + from google.appengine.api import urlfetch, urlfetch_errors +except ImportError: + urlfetch, urlfetch_errors = None, None + +# datadog +from datadog.api.exceptions import ClientError, HTTPError, HttpTimeout + + +log = logging.getLogger('dd.datadogpy') + + +class HTTPClient(object): + """ + An abstract generic HTTP client. Subclasses must implement the `request` methods. + """ + @classmethod + def request(cls, method, url, headers, params, data, timeout, proxies, verify, max_retries): + """ + Main method to be implemented by HTTP clients. + + The returned data structure has the following fields: + * `content`: string containing the response from the server + * `status_code`: HTTP status code returned by the server + + Can raise the following exceptions: + * `ClientError`: server cannot be contacted + * `HttpTimeout`: connection timed out + * `HTTPError`: unexpected HTTP response code + """ + raise NotImplementedError( + u"Must be implemented by HTTPClient subclasses." + ) + + +class RequestClient(HTTPClient): + """ + HTTP client based on 3rd party `requests` module. + """ + @classmethod + def request(cls, method, url, headers, params, data, timeout, proxies, verify, max_retries): + """ + """ + # Use a session to set a max_retries parameters + s = requests.Session() + http_adapter = requests.adapters.HTTPAdapter(max_retries=max_retries) + s.mount('https://', http_adapter) + + try: + result = s.request( + method, url, + headers=headers, params=params, data=data, + timeout=timeout, + proxies=proxies, verify=verify) + + result.raise_for_status() + + except requests.ConnectionError as e: + raise ClientError(method, url, e) + except requests.exceptions.Timeout: + raise HttpTimeout(method, url, timeout) + except requests.exceptions.HTTPError as e: + if e.response.status_code in (400, 403, 404, 409): + # This gets caught afterwards and raises an ApiError exception + pass + else: + raise HTTPError(e.response.status_code, result.reason) + except TypeError as e: + raise TypeError( + u"Your installed version of `requests` library seems not compatible with" + u"Datadog's usage. We recommand upgrading it ('pip install -U requests')." + u"If you need help or have any question, please contact support@datadoghq.com" + ) + + return result + + +class URLFetchClient(HTTPClient): + """ + HTTP client based on Google App Engine `urlfetch` module. + """ + @classmethod + def request(cls, method, url, headers, params, data, timeout, proxies, verify, max_retries): + """ + Wrapper around `urlfetch.fetch` method. + + TO IMPLEMENT: + * `max_retries` + """ + # No local certificate file can be used on Google App Engine + validate_certificate = True if verify else False + + # Encode parameters in the url + url_with_params = "{url}?{params}".format( + url=url, + params=urllib.urlencode(params) + ) + + try: + result = urlfetch.fetch( + url=url_with_params, + method=method, + headers=headers, + validate_certificate=validate_certificate, + deadline=timeout, + payload=data + ) + + cls.raise_on_status(result) + + except urlfetch.DownloadError as e: + raise ClientError(method, url, e) + except urlfetch_errors.DeadlineExceededError: + raise HttpTimeout(method, url, timeout) + + return result + + @classmethod + def raise_on_status(cls, result): + """ + Raise on HTTP status code errors. + """ + status_code = result.status_code + + if (status_code / 100) != 2: + if status_code in (400, 403, 404, 409): + pass + else: + raise HTTPError(status_code) + + +def resolve_http_client(): + """ + Resolve an appropriate HTTP client based the defined priority and user environment. + """ + if requests: + log.debug(u"Use `requests` based HTTP client.") + return RequestClient + + if urlfetch and urlfetch_errors: + log.debug(u"Use `urlfetch` based HTTP client.") + return URLFetchClient + + raise ImportError( + u"Datadog API client was unable to resolve a HTTP client. " + u" Please install `requests` library." + ) diff --git a/datadog/api/infrastructure.py b/datadog/api/infrastructure.py index 63b324df7..01f8c9da4 100644 --- a/datadog/api/infrastructure.py +++ b/datadog/api/infrastructure.py @@ -1,4 +1,4 @@ -from datadog.api.base import SearchableAPIResource +from datadog.api.resources import SearchableAPIResource class Infrastructure(SearchableAPIResource): diff --git a/datadog/api/metrics.py b/datadog/api/metrics.py index f0853f6ec..b175e4d45 100644 --- a/datadog/api/metrics.py +++ b/datadog/api/metrics.py @@ -1,8 +1,10 @@ +# stdlib import time from numbers import Number -from datadog.api.base import SearchableAPIResource, SendableAPIResource +# datadog from datadog.api.exceptions import ApiError +from datadog.api.resources import SearchableAPIResource, SendableAPIResource class Metric(SearchableAPIResource, SendableAPIResource): diff --git a/datadog/api/monitors.py b/datadog/api/monitors.py index 5c4487905..b8cf1a44a 100644 --- a/datadog/api/monitors.py +++ b/datadog/api/monitors.py @@ -1,4 +1,4 @@ -from datadog.api.base import GetableAPIResource, CreateableAPIResource, \ +from datadog.api.resources import GetableAPIResource, CreateableAPIResource, \ UpdatableAPIResource, ListableAPIResource, DeletableAPIResource, \ ActionAPIResource diff --git a/datadog/api/resources.py b/datadog/api/resources.py new file mode 100644 index 000000000..9dbe8a0ef --- /dev/null +++ b/datadog/api/resources.py @@ -0,0 +1,223 @@ +""" +Datadog API resources. +""" +# stdlib +import logging + +# datadog +from datadog.api.api_client import APIClient + + +log = logging.getLogger('dd.datadogpy') + + +class CreateableAPIResource(object): + """ + Creatable API Resource + """ + @classmethod + def create(cls, attach_host_name=False, method='POST', id=None, params=None, **body): + """ + Create a new API resource object + + :param attach_host_name: link the new resource object to the host name + :type attach_host_name: bool + + :param method: HTTP method to use to contact API endpoint + :type method: HTTP method string + + :param id: create a new resource object as a child of the given object + :type id: id + + :param params: new resource object source + :type params: dictionary + + :param body: new resource object attributes + :type body: dictionary + + :returns: JSON response from HTTP API request + """ + if params is None: + params = {} + if method == 'GET': + return APIClient.submit('GET', cls._class_url, **body) + if id is None: + return APIClient.submit('POST', cls._class_url, body, + attach_host_name=attach_host_name, **params) + else: + return APIClient.submit('POST', cls._class_url + "/" + str(id), body, + attach_host_name=attach_host_name, **params) + + +class SendableAPIResource(object): + """ + Fork of CreateableAPIResource class with different method names + """ + @classmethod + def send(cls, attach_host_name=False, id=None, **body): + """ + Create an API resource object + + :param attach_host_name: link the new resource object to the host name + :type attach_host_name: bool + + :param id: create a new resource object as a child of the given object + :type id: id + + :param body: new resource object attributes + :type body: dictionary + + :returns: JSON response from HTTP API request + """ + if id is None: + return APIClient.submit('POST', cls._class_url, body, + attach_host_name=attach_host_name) + else: + return APIClient.submit('POST', cls._class_url + "/" + str(id), body, + attach_host_name=attach_host_name) + + +class UpdatableAPIResource(object): + """ + Updatable API Resource + """ + @classmethod + def update(cls, id, params=None, **body): + """ + Update an API resource object + + :param params: updated resource object source + :type params: dictionary + + :param body: updated resource object attributes + :type body: dictionary + + :returns: JSON response from HTTP API request + """ + if params is None: + params = {} + return APIClient.submit('PUT', cls._class_url + "/" + str(id), body, **params) + + +class DeletableAPIResource(object): + """ + Deletable API Resource + """ + @classmethod + def delete(cls, id, **params): + """ + Delete an API resource object + + :param id: resource object to delete + :type id: id + + :returns: JSON response from HTTP API request + """ + return APIClient.submit('DELETE', cls._class_url + "/" + str(id), **params) + + +class GetableAPIResource(object): + """ + Getable API Resource + """ + @classmethod + def get(cls, id, **params): + """ + Get information about an API resource object + + :param id: resource object id to retrieve + :type id: id + + :param params: parameters to filter API resource stream + :type params: dictionary + + :returns: JSON response from HTTP API request + """ + return APIClient.submit('GET', cls._class_url + "/" + str(id), **params) + + +class ListableAPIResource(object): + """ + Listable API Resource + """ + @classmethod + def get_all(cls, **params): + """ + List API resource objects + + :param params: parameters to filter API resource stream + :type params: dictionary + + :returns: JSON response from HTTP API request + """ + return APIClient.submit('GET', cls._class_url, **params) + + +class SearchableAPIResource(object): + """ + Fork of ListableAPIResource class with different method names + """ + @classmethod + def _search(cls, **params): + """ + Query an API resource stream + + :param params: parameters to filter API resource stream + :type params: dictionary + + :returns: JSON response from HTTP API request + """ + return APIClient.submit('GET', cls._class_url, **params) + + +class ActionAPIResource(object): + """ + Actionable API Resource + """ + @classmethod + def _trigger_class_action(cls, method, name, id=None, **params): + """ + Trigger an action + + :param method: HTTP method to use to contact API endpoint + :type method: HTTP method string + + :param name: action name + :type name: string + + :param id: trigger the action for the specified resource object + :type id: id + + :param params: action parameters + :type params: dictionary + + :returns: JSON response from HTTP API request + """ + if id is None: + return APIClient.submit(method, cls._class_url + "/" + name, params) + else: + return APIClient.submit(method, cls._class_url + "/" + str(id) + "/" + name, params) + + @classmethod + def _trigger_action(cls, method, name, id=None, **params): + """ + Trigger an action + + :param method: HTTP method to use to contact API endpoint + :type method: HTTP method string + + :param name: action name + :type name: string + + :param id: trigger the action for the specified resource object + :type id: id + + :param params: action parameters + :type params: dictionary + + :returns: JSON response from HTTP API request + """ + if id is None: + return APIClient.submit(method, name, params) + else: + return APIClient.submit(method, name + "/" + str(id), params) diff --git a/datadog/api/screenboards.py b/datadog/api/screenboards.py index c8d2df031..a5cd3c64b 100644 --- a/datadog/api/screenboards.py +++ b/datadog/api/screenboards.py @@ -1,4 +1,4 @@ -from datadog.api.base import GetableAPIResource, CreateableAPIResource, \ +from datadog.api.resources import GetableAPIResource, CreateableAPIResource, \ UpdatableAPIResource, DeletableAPIResource, ActionAPIResource, ListableAPIResource diff --git a/datadog/api/service_checks.py b/datadog/api/service_checks.py index 6bb33c63e..524f277e0 100644 --- a/datadog/api/service_checks.py +++ b/datadog/api/service_checks.py @@ -1,6 +1,6 @@ -from datadog.api.base import ActionAPIResource -from datadog.api.exceptions import ApiError from datadog.api.constants import CheckStatus +from datadog.api.exceptions import ApiError +from datadog.api.resources import ActionAPIResource class ServiceCheck(ActionAPIResource): diff --git a/datadog/api/tags.py b/datadog/api/tags.py index d03369f22..213168e08 100644 --- a/datadog/api/tags.py +++ b/datadog/api/tags.py @@ -1,4 +1,4 @@ -from datadog.api.base import CreateableAPIResource, UpdatableAPIResource,\ +from datadog.api.resources import CreateableAPIResource, UpdatableAPIResource,\ DeletableAPIResource, GetableAPIResource, ListableAPIResource diff --git a/datadog/api/timeboards.py b/datadog/api/timeboards.py index f8877abab..de6d444a4 100644 --- a/datadog/api/timeboards.py +++ b/datadog/api/timeboards.py @@ -1,4 +1,4 @@ -from datadog.api.base import GetableAPIResource, CreateableAPIResource, \ +from datadog.api.resources import GetableAPIResource, CreateableAPIResource, \ UpdatableAPIResource, ListableAPIResource, DeletableAPIResource diff --git a/datadog/api/users.py b/datadog/api/users.py index cd6ac9554..57e7588bd 100644 --- a/datadog/api/users.py +++ b/datadog/api/users.py @@ -1,4 +1,4 @@ -from datadog.api.base import ActionAPIResource, GetableAPIResource, \ +from datadog.api.resources import ActionAPIResource, GetableAPIResource, \ CreateableAPIResource, UpdatableAPIResource, ListableAPIResource, \ DeletableAPIResource diff --git a/datadog/dogshell/__init__.py b/datadog/dogshell/__init__.py index 3df5392f7..a5361a2b4 100644 --- a/datadog/dogshell/__init__.py +++ b/datadog/dogshell/__init__.py @@ -1,7 +1,6 @@ # stdlib import logging import os -import pkg_resources as pkg # 3p import argparse @@ -20,6 +19,7 @@ from datadog.dogshell.service_check import ServiceCheckClient from datadog.dogshell.tag import TagClient from datadog.dogshell.timeboard import TimeboardClient +from datadog.util.config import get_version logging.getLogger('dd.datadogpy').setLevel(logging.CRITICAL) @@ -44,8 +44,7 @@ def main(): parser.add_argument('--timeout', help="time to wait in seconds before timing" " out an API call (default 10)", default=10, type=int) parser.add_argument('-v', '--version', help='Dog API version', action='version', - version='%(prog)s {version}' - .format(version=pkg.require("datadog")[0].version)) + version='%(prog)s {0}'.format(get_version())) config = DogshellConfig() diff --git a/datadog/dogshell/comment.py b/datadog/dogshell/comment.py index 4c7259c5c..c67ba7b35 100644 --- a/datadog/dogshell/comment.py +++ b/datadog/dogshell/comment.py @@ -1,12 +1,10 @@ # stdlib import sys -# 3p -import simplejson as json - # datadog -from datadog.dogshell.common import report_errors, report_warnings from datadog import api +from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class CommentClient(object): diff --git a/datadog/dogshell/downtime.py b/datadog/dogshell/downtime.py index 34fffcd22..c99cc74f5 100644 --- a/datadog/dogshell/downtime.py +++ b/datadog/dogshell/downtime.py @@ -1,10 +1,10 @@ # 3p -import simplejson as json from datadog.util.format import pretty_json # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class DowntimeClient(object): diff --git a/datadog/dogshell/event.py b/datadog/dogshell/event.py index 5b6accefe..3b0b53e77 100644 --- a/datadog/dogshell/event.py +++ b/datadog/dogshell/event.py @@ -4,12 +4,10 @@ import re import sys -# 3p -import simplejson as json - # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json def prettyprint_event(event): diff --git a/datadog/dogshell/host.py b/datadog/dogshell/host.py index 531c3804e..26a43547c 100644 --- a/datadog/dogshell/host.py +++ b/datadog/dogshell/host.py @@ -1,10 +1,10 @@ # 3p -import simplejson as json from datadog.util.format import pretty_json # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class HostClient(object): diff --git a/datadog/dogshell/monitor.py b/datadog/dogshell/monitor.py index bf19c3c14..ae9d2139a 100644 --- a/datadog/dogshell/monitor.py +++ b/datadog/dogshell/monitor.py @@ -1,10 +1,10 @@ # 3p -import simplejson as json from datadog.util.format import pretty_json # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class MonitorClient(object): diff --git a/datadog/dogshell/screenboard.py b/datadog/dogshell/screenboard.py index fc215ae68..f31dd62ea 100644 --- a/datadog/dogshell/screenboard.py +++ b/datadog/dogshell/screenboard.py @@ -5,12 +5,12 @@ import webbrowser # 3p -import simplejson as json from datadog.util.format import pretty_json # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings, print_err +from datadog.util.compat import json from datetime import datetime diff --git a/datadog/dogshell/search.py b/datadog/dogshell/search.py index 4903fd296..113a58984 100644 --- a/datadog/dogshell/search.py +++ b/datadog/dogshell/search.py @@ -1,9 +1,7 @@ -# 3p -import simplejson as json - # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json # TODO IS there a test ? diff --git a/datadog/dogshell/service_check.py b/datadog/dogshell/service_check.py index e6489abab..40623c967 100644 --- a/datadog/dogshell/service_check.py +++ b/datadog/dogshell/service_check.py @@ -1,10 +1,10 @@ # 3p -import simplejson as json from datadog.util.format import pretty_json # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class ServiceCheckClient(object): diff --git a/datadog/dogshell/tag.py b/datadog/dogshell/tag.py index 83f881149..3205ea72e 100644 --- a/datadog/dogshell/tag.py +++ b/datadog/dogshell/tag.py @@ -1,9 +1,7 @@ -# 3p -import simplejson as json - # datadog from datadog import api from datadog.dogshell.common import report_errors, report_warnings +from datadog.util.compat import json class TagClient(object): diff --git a/datadog/dogshell/timeboard.py b/datadog/dogshell/timeboard.py index b0bd6b604..553988a11 100644 --- a/datadog/dogshell/timeboard.py +++ b/datadog/dogshell/timeboard.py @@ -6,12 +6,12 @@ # 3p import argparse -import simplejson as json # datadog from datadog import api -from datadog.util.format import pretty_json from datadog.dogshell.common import report_errors, report_warnings, print_err +from datadog.util.compat import json +from datadog.util.format import pretty_json from datetime import datetime diff --git a/datadog/dogshell/wrap.py b/datadog/dogshell/wrap.py index 203a2f34c..241f628f5 100644 --- a/datadog/dogshell/wrap.py +++ b/datadog/dogshell/wrap.py @@ -19,7 +19,6 @@ ''' # stdlib import optparse -import pkg_resources as pkg import subprocess import sys import threading @@ -27,6 +26,7 @@ # datadog from datadog import initialize, api +from datadog.util.config import get_version SUCCESS = 'success' @@ -223,7 +223,7 @@ def main(): quotes to prevent python as soon as there is a space in your command. \n \nNOTICE: In normal \ mode, the whole stderr is printed before stdout, in flush_live mode they will be mixed but there \ is not guarantee that messages sent by the command on both stderr and stdout are printed in the \ -order they were sent.", version="%prog {0}".format(pkg.require("datadog")[0].version)) +order they were sent.", version="%prog {0}".format(get_version())) parser.add_option('-n', '--name', action='store', type='string', help="the name of the event \ as it should appear on your Datadog stream") diff --git a/datadog/util/compat.py b/datadog/util/compat.py index 13273e715..aa4cb2840 100644 --- a/datadog/util/compat.py +++ b/datadog/util/compat.py @@ -1,6 +1,6 @@ # flake8: noqa - -""" Imports for compatibility with Py2 and Py3 +""" +Imports for compatibility with Py2, Py3 and Google App Engine. """ import sys import logging @@ -51,3 +51,14 @@ def iternext(iter): from urllib.parse import urlparse except ImportError: from urlparse import urlparse + +try: + import pkg_resources as pkg +except ImportError: + pkg = None + +# Prefer `simplejson` but fall back to stdlib `json` +try: + import simplejson as json +except ImportError: + import json diff --git a/datadog/util/config.py b/datadog/util/config.py index c303d9e29..e5b9b9d4a 100644 --- a/datadog/util/config.py +++ b/datadog/util/config.py @@ -3,7 +3,8 @@ import string import sys -from datadog.util.compat import configparser, StringIO, is_p3k +# datadog +from datadog.util.compat import configparser, StringIO, is_p3k, pkg # CONSTANTS DATADOG_CONF = "datadog.conf" @@ -125,3 +126,27 @@ def get_config(cfg_path=None, options=None): raise CfgNotFound return agentConfig + + +def get_version(): + """ + Resolve `datadog` package version. + """ + version = u"unknown" + + if not pkg: + return version + + try: + dist = pkg.get_distribution("datadog") + # Normalize case for Windows systems + dist_loc = os.path.normcase(dist.location) + here = os.path.normcase(__file__) + if not here.startswith(dist_loc): + # not installed, but there is another version that *is* + raise pkg.DistributionNotFound + version = dist.version + except pkg.DistributionNotFound: + version = u"Please install `datadog` with setup.py" + + return version diff --git a/datadog/util/format.py b/datadog/util/format.py index 996719158..19ab8187a 100644 --- a/datadog/util/format.py +++ b/datadog/util/format.py @@ -1,5 +1,5 @@ -# 3p -import simplejson as json +# datadog +from datadog.util.compat import json def pretty_json(obj): diff --git a/datadog/util/hostname.py b/datadog/util/hostname.py index 497ee9581..33f90ab79 100644 --- a/datadog/util/hostname.py +++ b/datadog/util/hostname.py @@ -5,11 +5,8 @@ import subprocess import types -# 3p -import simplejson as json - # datadog -from datadog.util.compat import url_lib, is_p3k, iteritems +from datadog.util.compat import url_lib, is_p3k, iteritems, json from datadog.util.config import get_config, get_os, CfgNotFound VALID_HOSTNAME_RFC_1123_PATTERN = re.compile(r"^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])$") # noqa @@ -104,12 +101,12 @@ def _get_hostname_unix(): hostname = socket_hostname if hostname is None: - log.critical("Unable to reliably determine host name. You can define one" - " in datadog.conf or in your hosts file") - raise Exception("Unable to reliably determine host name. You can define" - " one in datadog.conf or in your hosts file") - else: - return hostname + log.warning( + u"Unable to reliably determine host name. You can define one in your `hosts` file, " + u"or in `datadog.conf` file if you have Datadog Agent installed." + ) + + return hostname def get_ec2_instance_id(): diff --git a/tests/unit/api/helper.py b/tests/unit/api/helper.py index ebfed1a31..dc46edaff 100644 --- a/tests/unit/api/helper.py +++ b/tests/unit/api/helper.py @@ -4,14 +4,13 @@ # 3p from mock import patch, Mock import requests -import simplejson as json # datadog from datadog import initialize, api -from datadog.api.base import CreateableAPIResource, UpdatableAPIResource, DeletableAPIResource,\ - GetableAPIResource, ListableAPIResource, ActionAPIResource from datadog.api.exceptions import ApiError -from datadog.util.compat import iteritems +from datadog.api.resources import CreateableAPIResource, UpdatableAPIResource, DeletableAPIResource,\ + GetableAPIResource, ListableAPIResource, ActionAPIResource +from datadog.util.compat import iteritems, json API_KEY = "apikey"