#!/usr/bin/python # -*- coding: utf-8 -*- # Hive Appier Framework # Copyright (c) 2008-2024 Hive Solutions Lda. # # This file is part of Hive Appier Framework. # # Hive Appier Framework is free software: you can redistribute it and/or modify # it under the terms of the Apache License as published by the Apache # Foundation, either version 2.0 of the License, or (at your option) any # later version. # # Hive Appier Framework is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # Apache License for more details. # # You should have received a copy of the Apache License along with # Hive Appier Framework. If not, see . __author__ = "João Magalhães " """ The author(s) of the module """ __copyright__ = "Copyright (c) 2008-2024 Hive Solutions Lda." """ The copyright for the module """ __license__ = "Apache License, Version 2.0" """ The license for the module """ import json import uuid from . import common from . import legacy class AppierException(Exception): """ Top level exception to be used as the root of all the exceptions to be raised by the appier infra- structure. Should be compatible with HTTP status codes for proper HTTP serialization. """ message = None """ The message value stored to describe the current exception value """ code = None """ The internal error code to be used in objective serialization of the error (eg: HTTP) """ headers = None """ Optional list of MIME compliant headers to be sent in a client/server paradigm """ meta = None """ The meta information associated with the error that should be considered private in context, this should be structured data ready to be serializable """ def __init__(self, *args, **kwargs): Exception.__init__(self, *args) self.name = self._name() self.message = kwargs.get("message", self.name) self.code = kwargs.get("code", 500) self.headers = kwargs.get("headers", None) self.meta = kwargs.get("meta", None) self.args = args self.kwargs = kwargs self._uid = None def __str__(self): if legacy.PYTHON_3: return self.__unicode__() is_unicode = legacy.is_unicode(self.message) if is_unicode: return self.message.encode("utf-8") return self.message def __unicode__(self): is_unicode = legacy.is_unicode(self.message) if not is_unicode: return self.message.decode("utf-8") return self.message def get_meta(self, name, default=None): if not self.meta: return default return self.meta.get(name, default) def set_meta(self, name, value): if not self.meta: self.meta = {} self.meta[name] = value def del_meta(self, name): if not self.meta: return if not name in self.meta: return del self.meta[name] @property def uid(self): if self._uid: return self._uid self._uid = uuid.uuid4() return self._uid def _name(self): cls = self.__class__ return common.util().camel_to_readable(cls.__name__) class OperationalError(AppierException): """ Error raised for a runtime error and as a result of an operational routine that failed. This should not be used for coherent development bugs, that are raised continuously. """ pass class SecurityError(AppierException): """ Error used to indicate security problems that may arise during the execution (runtime) of an appier application. This error should not be used to notify development related problems. """ pass class AssertionError(OperationalError): """ Error raised for failure to meet any pre-condition or assertion for a certain data set. """ def __init__(self, *args, **kwargs): kwargs["message"] = kwargs.get("message", "Assertion of data failed") kwargs["code"] = kwargs.get("code", None) OperationalError.__init__(self, *args, **kwargs) class ValidationError(OperationalError): """ Error raised when a validation on the model fails the error should associate a name in the model with a message describing the validation failure. """ errors = None """ The map containing an association between the name of a field and a list of validation errors for it """ model = None """ The model containing the values in it after the process of validation has completed """ def __init__(self, errors, model, *args, **kwargs): kwargs["message"] = kwargs.get("message", "Validation of submitted data failed") kwargs["code"] = kwargs.get("code", 400) OperationalError.__init__(self, *args, **kwargs) self.errors = errors self.model = model self.set_meta("errors", self.errors) def __str__(self): if legacy.PYTHON_3: return self.__unicode__() message = OperationalError.__str__(self) extended = common.is_devel() if not extended: return message errors_s = self.errors_s() if not errors_s: return message if self.model: message += " for model '%s' with id '%s'" % ( self.model.__class__._name(), self.model._id if hasattr(self.model, "_id") else "unset", ) errors_s = errors_s.encode("utf-8") message += " (" + errors_s + ")" return message def __unicode__(self): message = OperationalError.__unicode__(self) extended = common.is_devel() if not extended: return message errors_s = self.errors_s() if not errors_s: return message if self.model: message += " for model '%s' with id '%s'" % ( self.model.__class__._name(), self.model._id if hasattr(self.model, "_id") else "unset", ) message += " (" + errors_s + ")" return message def errors_s(self, encoding="utf-8"): if not self.errors: return "" buffer = [] is_first = True for name, errors in legacy.iteritems(self.errors): for error in errors: is_bytes = legacy.is_bytes(error) if is_bytes: error = error.decode(encoding) if is_first: is_first = False else: buffer.append(", ") buffer.append(name) buffer.append(" => ") buffer.append(error) return legacy.u("").join(buffer) class NotFoundError(OperationalError): """ Error originated from an operation that was not able to be performed because it was not able to found the requested entity/value as defined by specification. """ def __init__(self, *args, **kwargs): kwargs["code"] = kwargs.get("code", 404) OperationalError.__init__(self, *args, **kwargs) class NotImplementedError(OperationalError): """ Error to be raised when a certain feature or route is not yet implemented or is not meant to be implemented at the defined abstraction level. """ def __init__(self, *args, **kwargs): kwargs["code"] = kwargs.get("code", 501) OperationalError.__init__(self, *args, **kwargs) class BaseInternalError(RuntimeError): """ The base (internal) error class from which all the internal error classes should inherit, contains basic functionality to be inherited by all the internal "errors". """ message = None """ The message value stored to describe the current error, this should be a valid string """ meta = None """ The meta information associated with the error that should be considered private in context, this should be structured data ready to be serializable """ def __init__(self, message, meta=None): RuntimeError.__init__(self, message) self.message = message self.meta = meta def __str__(self): if legacy.PYTHON_3: return self.__unicode__() is_unicode = legacy.is_unicode(self.message) if is_unicode: return self.message.encode("utf-8") return self.message def __unicode__(self): is_unicode = legacy.is_unicode(self.message) if not is_unicode: return self.message.decode("utf-8") return self.message def get_meta(self, name, default=None): if not self.meta: return default return self.meta.get(name, default) def set_meta(self, name, value): if not self.meta: self.meta = {} self.meta[name] = value def del_meta(self, name): if not self.meta: return if not name in self.meta: return del self.meta[name] class ValidationInternalError(BaseInternalError): """ Error raised when a validation on the model fails the error should associate a name in the model with a message describing the validation failure. """ name = None """ The name of the attribute that failed the validation, for latter reference """ def __init__(self, name, message, meta=None): BaseInternalError.__init__(self, message, meta=meta) self.name = name class ValidationMultipleError(ValidationInternalError): """ Exception/error considered to be equivalent to the validation internal error, with the exception that it may handle multiple errors at the same time. """ errors = [] """ The sequence containing the multiple errors associated with the validation multiple error """ def __init__(self, name=None, message=None, meta=None): ValidationInternalError.__init__(self, name, message, meta=meta) self.errors = [] self.set_meta("errors", self.errors) def add_error(self, name, message): if not self.name: self.name = name if not self.message: self.message = message self.errors.append((name, message)) def add_exception(self, exception): self.add_error(exception.name, exception.message) class HTTPError(BaseInternalError): """ Top level HTTP error raised whenever a bad response is received from the server peer. This error is meant to be used by the client library. """ error = None """ The reference to the original and internal HTTP error that is going to be used in the reading of the underlying internal buffer """ _data = None """ The underlying/internal data attribute that is going to be used to cache the binary contents of the error associated with this exception (data) stream """ def __init__(self, error, code=None, message=None, extended=None, meta=None): message = message or "Problem in the HTTP request" if extended == None: extended = common.is_devel() if code: message = "[%d] %s" % (code, message) if extended: data = self.read(error=error) try: data = data.decode("utf-8") except Exception: data = legacy.str(data) if data: message = message + "\n" + data if message else data BaseInternalError.__init__(self, message, meta=meta) self.code = code self.error = error def read(self, error=None): error = error or self.error if not self._data == None: return self._data self._data = error.read() return self._data def read_json(self, error=None): error = error or self.error data = self.read(error=error) if legacy.is_bytes(data): data = data.decode("utf-8") try: data_j = json.loads(data) except Exception: data_j = None return data_j class APIError(BaseInternalError): """ Highest level error for API related problems that may be raised from the appier API infra-structure. These kind of errors should be encapsulated around proper structures """ def __init__(self, *args, **kwargs): message = kwargs.get("message", None) meta = kwargs.get("meta", None) BaseInternalError.__init__(self, message, meta=meta) class APIAccessError(APIError): """ General purpose access error exception, to be raised under situations where the access to a certain functionalities is denied for insufficient permissions/invalid credentials. """ original = None """ The reference to the original error/exception that originated this API access error, this may be unset in case no concrete error has originated this error """ def __init__(self, *args, **kwargs): self.original = kwargs.get("original", None) if self.original and hasattr(self.original, "message"): kwargs["message"] = self.original.message APIError.__init__(self, *args, **kwargs) class OAuthAccessError(APIAccessError): """ OAuth related problems that typically involve either outdated tokens or invalid ones. Triggering this exception should imply a revalidation of the current token. """ pass