views.py 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279
  1. from __future__ import unicode_literals
  2. import datetime
  3. import decimal
  4. import logging
  5. import sys
  6. from django.core.exceptions import PermissionDenied, SuspiciousOperation
  7. from django.core.urlresolvers import get_resolver
  8. from django.http import Http404, HttpResponse, JsonResponse
  9. from django.shortcuts import render, render_to_response
  10. from django.template import TemplateDoesNotExist
  11. from django.views.debug import (
  12. SafeExceptionReporterFilter, technical_500_response,
  13. )
  14. from django.views.decorators.debug import (
  15. sensitive_post_parameters, sensitive_variables,
  16. )
  17. from django.views.generic import View
  18. from . import BrokenException, except_args
  19. def index_page(request):
  20. """Dummy index page"""
  21. return HttpResponse('<html><body>Dummy page</body></html>')
  22. def raises(request):
  23. # Make sure that a callable that raises an exception in the stack frame's
  24. # local vars won't hijack the technical 500 response. See:
  25. # http://code.djangoproject.com/ticket/15025
  26. def callable():
  27. raise Exception
  28. try:
  29. raise Exception
  30. except Exception:
  31. return technical_500_response(request, *sys.exc_info())
  32. def raises500(request):
  33. # We need to inspect the HTML generated by the fancy 500 debug view but
  34. # the test client ignores it, so we send it explicitly.
  35. try:
  36. raise Exception
  37. except Exception:
  38. return technical_500_response(request, *sys.exc_info())
  39. def raises400(request):
  40. raise SuspiciousOperation
  41. def raises403(request):
  42. raise PermissionDenied
  43. def raises404(request):
  44. resolver = get_resolver(None)
  45. resolver.resolve('/not-in-urls')
  46. def technical404(request):
  47. raise Http404("Testing technical 404.")
  48. class Http404View(View):
  49. def get(self, request):
  50. raise Http404("Testing class-based technical 404.")
  51. def view_exception(request, n):
  52. raise BrokenException(except_args[int(n)])
  53. def template_exception(request, n):
  54. return render_to_response('debug/template_exception.html',
  55. {'arg': except_args[int(n)]})
  56. def jsi18n(request):
  57. return render_to_response('jsi18n.html')
  58. def jsi18n_multi_catalogs(request):
  59. return render_to_response('jsi18n-multi-catalogs.html')
  60. def raises_template_does_not_exist(request, path='i_dont_exist.html'):
  61. # We need to inspect the HTML generated by the fancy 500 debug view but
  62. # the test client ignores it, so we send it explicitly.
  63. try:
  64. return render_to_response(path)
  65. except TemplateDoesNotExist:
  66. return technical_500_response(request, *sys.exc_info())
  67. def render_no_template(request):
  68. # If we do not specify a template, we need to make sure the debug
  69. # view doesn't blow up.
  70. return render(request, [], {})
  71. def send_log(request, exc_info):
  72. logger = logging.getLogger('django')
  73. # The default logging config has a logging filter to ensure admin emails are
  74. # only sent with DEBUG=False, but since someone might choose to remove that
  75. # filter, we still want to be able to test the behavior of error emails
  76. # with DEBUG=True. So we need to remove the filter temporarily.
  77. admin_email_handler = [
  78. h for h in logger.handlers
  79. if h.__class__.__name__ == "AdminEmailHandler"
  80. ][0]
  81. orig_filters = admin_email_handler.filters
  82. admin_email_handler.filters = []
  83. admin_email_handler.include_html = True
  84. logger.error('Internal Server Error: %s', request.path,
  85. exc_info=exc_info,
  86. extra={
  87. 'status_code': 500,
  88. 'request': request
  89. }
  90. )
  91. admin_email_handler.filters = orig_filters
  92. def non_sensitive_view(request):
  93. # Do not just use plain strings for the variables' values in the code
  94. # so that the tests don't return false positives when the function's source
  95. # is displayed in the exception report.
  96. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  97. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  98. try:
  99. raise Exception
  100. except Exception:
  101. exc_info = sys.exc_info()
  102. send_log(request, exc_info)
  103. return technical_500_response(request, *exc_info)
  104. @sensitive_variables('sauce')
  105. @sensitive_post_parameters('bacon-key', 'sausage-key')
  106. def sensitive_view(request):
  107. # Do not just use plain strings for the variables' values in the code
  108. # so that the tests don't return false positives when the function's source
  109. # is displayed in the exception report.
  110. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  111. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  112. try:
  113. raise Exception
  114. except Exception:
  115. exc_info = sys.exc_info()
  116. send_log(request, exc_info)
  117. return technical_500_response(request, *exc_info)
  118. @sensitive_variables()
  119. @sensitive_post_parameters()
  120. def paranoid_view(request):
  121. # Do not just use plain strings for the variables' values in the code
  122. # so that the tests don't return false positives when the function's source
  123. # is displayed in the exception report.
  124. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  125. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  126. try:
  127. raise Exception
  128. except Exception:
  129. exc_info = sys.exc_info()
  130. send_log(request, exc_info)
  131. return technical_500_response(request, *exc_info)
  132. def sensitive_args_function_caller(request):
  133. try:
  134. sensitive_args_function(''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']))
  135. except Exception:
  136. exc_info = sys.exc_info()
  137. send_log(request, exc_info)
  138. return technical_500_response(request, *exc_info)
  139. @sensitive_variables('sauce')
  140. def sensitive_args_function(sauce):
  141. # Do not just use plain strings for the variables' values in the code
  142. # so that the tests don't return false positives when the function's source
  143. # is displayed in the exception report.
  144. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  145. raise Exception
  146. def sensitive_kwargs_function_caller(request):
  147. try:
  148. sensitive_kwargs_function(''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']))
  149. except Exception:
  150. exc_info = sys.exc_info()
  151. send_log(request, exc_info)
  152. return technical_500_response(request, *exc_info)
  153. @sensitive_variables('sauce')
  154. def sensitive_kwargs_function(sauce=None):
  155. # Do not just use plain strings for the variables' values in the code
  156. # so that the tests don't return false positives when the function's source
  157. # is displayed in the exception report.
  158. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  159. raise Exception
  160. class UnsafeExceptionReporterFilter(SafeExceptionReporterFilter):
  161. """
  162. Ignores all the filtering done by its parent class.
  163. """
  164. def get_post_parameters(self, request):
  165. return request.POST
  166. def get_traceback_frame_variables(self, request, tb_frame):
  167. return tb_frame.f_locals.items()
  168. @sensitive_variables()
  169. @sensitive_post_parameters()
  170. def custom_exception_reporter_filter_view(request):
  171. # Do not just use plain strings for the variables' values in the code
  172. # so that the tests don't return false positives when the function's source
  173. # is displayed in the exception report.
  174. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  175. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  176. request.exception_reporter_filter = UnsafeExceptionReporterFilter()
  177. try:
  178. raise Exception
  179. except Exception:
  180. exc_info = sys.exc_info()
  181. send_log(request, exc_info)
  182. return technical_500_response(request, *exc_info)
  183. class Klass(object):
  184. @sensitive_variables('sauce')
  185. def method(self, request):
  186. # Do not just use plain strings for the variables' values in the code
  187. # so that the tests don't return false positives when the function's
  188. # source is displayed in the exception report.
  189. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  190. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  191. try:
  192. raise Exception
  193. except Exception:
  194. exc_info = sys.exc_info()
  195. send_log(request, exc_info)
  196. return technical_500_response(request, *exc_info)
  197. def sensitive_method_view(request):
  198. return Klass().method(request)
  199. @sensitive_variables('sauce')
  200. @sensitive_post_parameters('bacon-key', 'sausage-key')
  201. def multivalue_dict_key_error(request):
  202. cooked_eggs = ''.join(['s', 'c', 'r', 'a', 'm', 'b', 'l', 'e', 'd']) # NOQA
  203. sauce = ''.join(['w', 'o', 'r', 'c', 'e', 's', 't', 'e', 'r', 's', 'h', 'i', 'r', 'e']) # NOQA
  204. try:
  205. request.POST['bar']
  206. except Exception:
  207. exc_info = sys.exc_info()
  208. send_log(request, exc_info)
  209. return technical_500_response(request, *exc_info)
  210. def json_response_view(request):
  211. return JsonResponse({
  212. 'a': [1, 2, 3],
  213. 'foo': {'bar': 'baz'},
  214. # Make sure datetime and Decimal objects would be serialized properly
  215. 'timestamp': datetime.datetime(2013, 5, 19, 20),
  216. 'value': decimal.Decimal('3.14'),
  217. })