Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

external_api.py 4.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. import re
  2. import sys
  3. from collections.abc import Mapping
  4. from typing import Any
  5. from flask import current_app, got_request_exception
  6. from flask_restx import Api
  7. from werkzeug.exceptions import HTTPException
  8. from werkzeug.http import HTTP_STATUS_CODES
  9. from core.errors.error import AppInvokeQuotaExceededError
  10. def http_status_message(code):
  11. return HTTP_STATUS_CODES.get(code, "")
  12. def register_external_error_handlers(api: Api) -> None:
  13. @api.errorhandler(HTTPException)
  14. def handle_http_exception(e: HTTPException):
  15. got_request_exception.send(current_app, exception=e)
  16. # If Werkzeug already prepared a Response, just use it.
  17. if getattr(e, "response", None) is not None:
  18. return e.response
  19. status_code = getattr(e, "code", 500) or 500
  20. # Build a safe, dict-like payload
  21. default_data = {
  22. "code": re.sub(r"(?<!^)(?=[A-Z])", "_", type(e).__name__).lower(),
  23. "message": getattr(e, "description", http_status_message(status_code)),
  24. "status": status_code,
  25. }
  26. if default_data["message"] == "Failed to decode JSON object: Expecting value: line 1 column 1 (char 0)":
  27. default_data["message"] = "Invalid JSON payload received or JSON payload is empty."
  28. # Use headers on the exception if present; otherwise none.
  29. headers = {}
  30. exc_headers = getattr(e, "headers", None)
  31. if exc_headers:
  32. headers.update(exc_headers)
  33. # Payload per status
  34. if status_code == 406 and api.default_mediatype is None:
  35. data = {"code": "not_acceptable", "message": default_data["message"], "status": status_code}
  36. return data, status_code, headers
  37. elif status_code == 400:
  38. msg = default_data["message"]
  39. if isinstance(msg, Mapping) and msg:
  40. # Convert param errors like {"field": "reason"} into a friendly shape
  41. param_key, param_value = next(iter(msg.items()))
  42. data = {
  43. "code": "invalid_param",
  44. "message": str(param_value),
  45. "params": param_key,
  46. "status": status_code,
  47. }
  48. else:
  49. data = {**default_data}
  50. data.setdefault("code", "unknown")
  51. return data, status_code, headers
  52. else:
  53. data = {**default_data}
  54. data.setdefault("code", "unknown")
  55. # If you need WWW-Authenticate for 401, add it to headers
  56. if status_code == 401:
  57. headers["WWW-Authenticate"] = 'Bearer realm="api"'
  58. return data, status_code, headers
  59. @api.errorhandler(ValueError)
  60. def handle_value_error(e: ValueError):
  61. got_request_exception.send(current_app, exception=e)
  62. status_code = 400
  63. data = {"code": "invalid_param", "message": str(e), "status": status_code}
  64. return data, status_code
  65. @api.errorhandler(AppInvokeQuotaExceededError)
  66. def handle_quota_exceeded(e: AppInvokeQuotaExceededError):
  67. got_request_exception.send(current_app, exception=e)
  68. status_code = 429
  69. data = {"code": "too_many_requests", "message": str(e), "status": status_code}
  70. return data, status_code
  71. @api.errorhandler(Exception)
  72. def handle_general_exception(e: Exception):
  73. got_request_exception.send(current_app, exception=e)
  74. status_code = 500
  75. data: dict[str, Any] = getattr(e, "data", {"message": http_status_message(status_code)})
  76. # 🔒 Normalize non-mapping data (e.g., if someone set e.data = Response)
  77. if not isinstance(data, Mapping):
  78. data = {"message": str(e)}
  79. data.setdefault("code", "unknown")
  80. data.setdefault("status", status_code)
  81. # Log stack
  82. exc_info: Any = sys.exc_info()
  83. if exc_info[1] is None:
  84. exc_info = None
  85. current_app.log_exception(exc_info)
  86. return data, status_code
  87. class ExternalApi(Api):
  88. def __init__(self, *args, **kwargs):
  89. super().__init__(*args, **kwargs)
  90. register_external_error_handlers(self)