Source code for contentful.errors

"""
contentful.errors
~~~~~~~~~~~~~~~~~

This module implements the Error classes.

API Reference: https://www.contentful.com/developers/docs/references/content-delivery-api/#/introduction/errors

:copyright: (c) 2016 by Contentful GmbH.
:license: MIT, see LICENSE for more details.
"""


[docs]class HTTPError(Exception): """ Base HTTP Error Class """ def __init__(self, response): self.response = response self.status_code = response.status_code message = self._best_available_message(response) super(HTTPError, self).__init__(message) def _default_error_message(self): return "The following error was received: {0}".format(self.response.text) def _handle_details(self, details): return "{0}".format(details) def _has_additional_error_info(self): return False def _additional_error_info(self): return [] def _best_available_message(self, response): from .utils import json_error_class response_json = None error_message = [ "HTTP status code: {0}".format(self.status_code), ] try: response_json = response.json() message = response_json.get('message', None) details = response_json.get('details', None) request_id = response_json.get('requestId', None) if message is not None: error_message.append("Message: {0}".format(message)) else: error_message.append("Message: {0}".format(self._default_error_message())) if details is not None: error_message.append("Details: {0}".format(self._handle_details(details))) if request_id is not None: error_message.append("Request ID: {0}".format(request_id)) except json_error_class(): error_message.append("Message: {0}".format(self._default_error_message())) if self._has_additional_error_info(): error_message += self._additional_error_info() return "\n".join(error_message)
[docs]class BadRequestError(HTTPError): """ 400 """ def _default_error_message(self): return "The request was malformed or missing a required parameter." def _handle_details(self, details): from .utils import string_class if isinstance(details, string_class()): return details def _handle_detail(detail): if isinstance(detail, string_class()): return detail return detail.get('details', None) inner_details = [_handle_detail(detail) for detail in details['errors']] inner_details = [detail for detail in inner_details if detail is not None] # This works in both Py2 and Py3 return "\n\t".join(inner_details)
[docs]class UnauthorizedError(HTTPError): """ 401 """ def _default_error_message(self): return "The authorization token was invalid."
[docs]class AccessDeniedError(HTTPError): """ 403 """ def _default_error_message(self): return "The specified token does not have access to the requested resource." def _handle_details(self, details): return "\n\tReasons:\n\t\t{0}".format("\n\t\t".join(details['reasons']))
[docs]class NotFoundError(HTTPError): """ 404 """ def _default_error_message(self): return "The requested resource or endpoint could not be found." def _handle_details(self, details): from .utils import string_class if isinstance(details, string_class()): return details resource_type = None resource_id = None if 'sys' in details: resource_type = details['sys'].get('type', None) resource_id = details['sys'].get('id', None) else: resource_type = details['type'] resource_id = details.get('id', None) message = "The requested {0} could not be found.".format(resource_type) if resource_id is not None: message += " ID: {0}.".format(resource_id) return message
[docs]class RateLimitExceededError(HTTPError): """ 429 """ RATE_LIMIT_RESET_HEADER_KEY = 'x-contentful-ratelimit-reset' def _has_reset_time(self): return self.RATE_LIMIT_RESET_HEADER_KEY in self.response.headers
[docs] def reset_time(self): """Returns the reset time in seconds until next available request.""" return int(self.response.headers[ self.RATE_LIMIT_RESET_HEADER_KEY ])
def _has_additional_error_info(self): return self._has_reset_time() def _additional_error_info(self): return ["Time until reset (seconds): {0}".format(self.reset_time())] def _default_error_message(self): return "Rate limit exceeded. Too many requests."
[docs]class ServerError(HTTPError): """ 500 """ def _default_error_message(self): return "Internal server error."
[docs]class BadGatewayError(HTTPError): """ 502 """ def _default_error_message(self): return "The requested space is hibernated."
[docs]class ServiceUnavailableError(HTTPError): """ 503 """ def _default_error_message(self): return "The request was malformed or missing a required parameter."
[docs]class EntryNotFoundError(Exception): """ Error for entry not found. """ pass
[docs]def get_error(response): """Gets Error by HTTP Status Code""" errors = { 400: BadRequestError, 401: UnauthorizedError, 403: AccessDeniedError, 404: NotFoundError, 429: RateLimitExceededError, 500: ServerError, 502: BadGatewayError, 503: ServiceUnavailableError } error_class = HTTPError if response.status_code in errors: error_class = errors[response.status_code] return error_class(response)