"""
Types of Errors specified by personio-py
"""
from typing import Any, Dict, Type
from requests import Response
[docs]class PersonioError(Exception):
    """A generic error caused by personio-py""" 
[docs]class MissingCredentialsError(PersonioError):
    """you are missing some strictly required credentials""" 
[docs]class PersonioApiError(PersonioError):
    """
    An error response from the Personio HTTP API
    :param status_code: the HTTP status code of the API response
    :param message: the error message from the Personio API
    :param error_code: the error code that is provided with the Personio API response
           (not the HTTP status code!)
    :param errors: details about the errors that are provided by the Personio API as dictionary
    :param response: the HTTP response
    """
    def __init__(self, status_code: int, message: str, error_code: int = None,
                 errors: Dict[str, Any] = None, response: Response = None):
        super().__init__()
        self.status_code = status_code
        self.message = message
        self.error_code = error_code
        self.errors = errors
        self.response = response
[docs]    @classmethod
    def from_response(cls, response: Response):
        """
        Creates a ``PersonioApiError`` from the specified HTTP response.
        :param response: a HTTP error response from Personio
        :return: a PersonioApiError that matches the HTTP error
        """
        try:
            data: Dict = response.json()
            error = data.get('error', {})
            code = error.get('code')
            message = error.get('message')
            error_dict: Dict = error.get('errors')
            return PersonioApiError(
                status_code=response.status_code,
                message=message,
                error_code=code,
                errors=error_dict,
                response=response)
        except ValueError:
            return PersonioApiError(
                status_code=response.status_code,
                message=response.text,
                response=response) 
    def __str__(self):
        message = f"request failed with HTTP status code {self.status_code}: {self.message}"
        if self.error_code:
            message += f" (error code {self.error_code})"
        if self.errors:
            if message[-1] != '.':
                message += '.'
            message += f" Details: {self.errors}"
        return message 
[docs]class UnsupportedMethodError(PersonioError):
    """this method is not supported by this class (but it might be by a similar one)"""
    def __init__(self, method: str, clazz: Type):
        super().__init__()
        self.method = method
        self.clazz = clazz
    def __str__(self):
        return f"method '{self.method}' is not available for {self.clazz.__name__}"