exception.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. from __future__ import unicode_literals
  2. import logging
  3. import sys
  4. import warnings
  5. from functools import wraps
  6. from django.conf import settings
  7. from django.core import signals
  8. from django.core.exceptions import PermissionDenied, SuspiciousOperation
  9. from django.http import Http404
  10. from django.http.multipartparser import MultiPartParserError
  11. from django.urls import get_resolver, get_urlconf
  12. from django.utils import six
  13. from django.utils.decorators import available_attrs
  14. from django.utils.deprecation import RemovedInDjango20Warning
  15. from django.utils.encoding import force_text
  16. from django.views import debug
  17. logger = logging.getLogger('django.request')
  18. def convert_exception_to_response(get_response):
  19. """
  20. Wrap the given get_response callable in exception-to-response conversion.
  21. All exceptions will be converted. All known 4xx exceptions (Http404,
  22. PermissionDenied, MultiPartParserError, SuspiciousOperation) will be
  23. converted to the appropriate response, and all other exceptions will be
  24. converted to 500 responses.
  25. This decorator is automatically applied to all middleware to ensure that
  26. no middleware leaks an exception and that the next middleware in the stack
  27. can rely on getting a response instead of an exception.
  28. """
  29. @wraps(get_response, assigned=available_attrs(get_response))
  30. def inner(request):
  31. try:
  32. response = get_response(request)
  33. except Exception as exc:
  34. response = response_for_exception(request, exc)
  35. return response
  36. return inner
  37. def response_for_exception(request, exc):
  38. if isinstance(exc, Http404):
  39. if settings.DEBUG:
  40. response = debug.technical_404_response(request, exc)
  41. else:
  42. response = get_exception_response(request, get_resolver(get_urlconf()), 404, exc)
  43. elif isinstance(exc, PermissionDenied):
  44. logger.warning(
  45. 'Forbidden (Permission denied): %s', request.path,
  46. extra={'status_code': 403, 'request': request},
  47. )
  48. response = get_exception_response(request, get_resolver(get_urlconf()), 403, exc)
  49. elif isinstance(exc, MultiPartParserError):
  50. logger.warning(
  51. 'Bad request (Unable to parse request body): %s', request.path,
  52. extra={'status_code': 400, 'request': request},
  53. )
  54. response = get_exception_response(request, get_resolver(get_urlconf()), 400, exc)
  55. elif isinstance(exc, SuspiciousOperation):
  56. # The request logger receives events for any problematic request
  57. # The security logger receives events for all SuspiciousOperations
  58. security_logger = logging.getLogger('django.security.%s' % exc.__class__.__name__)
  59. security_logger.error(
  60. force_text(exc),
  61. extra={'status_code': 400, 'request': request},
  62. )
  63. if settings.DEBUG:
  64. response = debug.technical_500_response(request, *sys.exc_info(), status_code=400)
  65. else:
  66. response = get_exception_response(request, get_resolver(get_urlconf()), 400, exc)
  67. elif isinstance(exc, SystemExit):
  68. # Allow sys.exit() to actually exit. See tickets #1023 and #4701
  69. raise
  70. else:
  71. signals.got_request_exception.send(sender=None, request=request)
  72. response = handle_uncaught_exception(request, get_resolver(get_urlconf()), sys.exc_info())
  73. return response
  74. def get_exception_response(request, resolver, status_code, exception, sender=None):
  75. try:
  76. callback, param_dict = resolver.resolve_error_handler(status_code)
  77. # Unfortunately, inspect.getargspec result is not trustable enough
  78. # depending on the callback wrapping in decorators (frequent for handlers).
  79. # Falling back on try/except:
  80. try:
  81. response = callback(request, **dict(param_dict, exception=exception))
  82. except TypeError:
  83. warnings.warn(
  84. "Error handlers should accept an exception parameter. Update "
  85. "your code as this parameter will be required in Django 2.0",
  86. RemovedInDjango20Warning, stacklevel=2
  87. )
  88. response = callback(request, **param_dict)
  89. except Exception:
  90. signals.got_request_exception.send(sender=sender, request=request)
  91. response = handle_uncaught_exception(request, resolver, sys.exc_info())
  92. return response
  93. def handle_uncaught_exception(request, resolver, exc_info):
  94. """
  95. Processing for any otherwise uncaught exceptions (those that will
  96. generate HTTP 500 responses).
  97. """
  98. if settings.DEBUG_PROPAGATE_EXCEPTIONS:
  99. raise
  100. logger.error(
  101. 'Internal Server Error: %s', request.path,
  102. exc_info=exc_info,
  103. extra={'status_code': 500, 'request': request},
  104. )
  105. if settings.DEBUG:
  106. return debug.technical_500_response(request, *exc_info)
  107. # If Http500 handler is not installed, reraise the last exception.
  108. if resolver.urlconf_module is None:
  109. six.reraise(*exc_info)
  110. # Return an HttpResponse that displays a friendly error message.
  111. callback, param_dict = resolver.resolve_error_handler(500)
  112. return callback(request, **param_dict)