PageRenderTime 42ms CodeModel.GetById 13ms RepoModel.GetById 0ms app.codeStats 0ms

/lib/django-1.4/tests/regressiontests/decorators/tests.py

https://github.com/theosp/google_appengine
Python | 275 lines | 173 code | 50 blank | 52 comment | 1 complexity | ceb2ed7edd6164422a2c57eb1eb78f3e MD5 | raw file
  1. import warnings
  2. from functools import wraps
  3. from django.contrib.admin.views.decorators import staff_member_required
  4. from django.contrib.auth.decorators import login_required, permission_required, user_passes_test
  5. from django.http import HttpResponse, HttpRequest, HttpResponseNotAllowed
  6. from django.middleware.clickjacking import XFrameOptionsMiddleware
  7. from django.test.utils import get_warnings_state, restore_warnings_state
  8. from django.utils.decorators import method_decorator
  9. from django.utils.functional import allow_lazy, lazy, memoize
  10. from django.utils.unittest import TestCase
  11. from django.views.decorators.cache import cache_page, never_cache, cache_control
  12. from django.views.decorators.clickjacking import xframe_options_deny, xframe_options_sameorigin, xframe_options_exempt
  13. from django.views.decorators.http import require_http_methods, require_GET, require_POST, require_safe, condition
  14. from django.views.decorators.vary import vary_on_headers, vary_on_cookie
  15. def fully_decorated(request):
  16. """Expected __doc__"""
  17. return HttpResponse('<html><body>dummy</body></html>')
  18. fully_decorated.anything = "Expected __dict__"
  19. def compose(*functions):
  20. # compose(f, g)(*args, **kwargs) == f(g(*args, **kwargs))
  21. functions = list(reversed(functions))
  22. def _inner(*args, **kwargs):
  23. result = functions[0](*args, **kwargs)
  24. for f in functions[1:]:
  25. result = f(result)
  26. return result
  27. return _inner
  28. full_decorator = compose(
  29. # django.views.decorators.http
  30. require_http_methods(["GET"]),
  31. require_GET,
  32. require_POST,
  33. require_safe,
  34. condition(lambda r: None, lambda r: None),
  35. # django.views.decorators.vary
  36. vary_on_headers('Accept-language'),
  37. vary_on_cookie,
  38. # django.views.decorators.cache
  39. cache_page(60*15),
  40. cache_control(private=True),
  41. never_cache,
  42. # django.contrib.auth.decorators
  43. # Apply user_passes_test twice to check #9474
  44. user_passes_test(lambda u:True),
  45. login_required,
  46. permission_required('change_world'),
  47. # django.contrib.admin.views.decorators
  48. staff_member_required,
  49. # django.utils.functional
  50. lambda f: memoize(f, {}, 1),
  51. allow_lazy,
  52. lazy,
  53. )
  54. fully_decorated = full_decorator(fully_decorated)
  55. class DecoratorsTest(TestCase):
  56. def setUp(self):
  57. self.warning_state = get_warnings_state()
  58. warnings.filterwarnings('ignore', category=PendingDeprecationWarning,
  59. module='django.views.decorators.cache')
  60. def tearDown(self):
  61. restore_warnings_state(self.warning_state)
  62. def test_attributes(self):
  63. """
  64. Tests that django decorators set certain attributes of the wrapped
  65. function.
  66. """
  67. self.assertEqual(fully_decorated.__name__, 'fully_decorated')
  68. self.assertEqual(fully_decorated.__doc__, 'Expected __doc__')
  69. self.assertEqual(fully_decorated.__dict__['anything'], 'Expected __dict__')
  70. def test_user_passes_test_composition(self):
  71. """
  72. Test that the user_passes_test decorator can be applied multiple times
  73. (#9474).
  74. """
  75. def test1(user):
  76. user.decorators_applied.append('test1')
  77. return True
  78. def test2(user):
  79. user.decorators_applied.append('test2')
  80. return True
  81. def callback(request):
  82. return request.user.decorators_applied
  83. callback = user_passes_test(test1)(callback)
  84. callback = user_passes_test(test2)(callback)
  85. class DummyUser(object): pass
  86. class DummyRequest(object): pass
  87. request = DummyRequest()
  88. request.user = DummyUser()
  89. request.user.decorators_applied = []
  90. response = callback(request)
  91. self.assertEqual(response, ['test2', 'test1'])
  92. def test_cache_page_new_style(self):
  93. """
  94. Test that we can call cache_page the new way
  95. """
  96. def my_view(request):
  97. return "response"
  98. my_view_cached = cache_page(123)(my_view)
  99. self.assertEqual(my_view_cached(HttpRequest()), "response")
  100. my_view_cached2 = cache_page(123, key_prefix="test")(my_view)
  101. self.assertEqual(my_view_cached2(HttpRequest()), "response")
  102. def test_cache_page_old_style(self):
  103. """
  104. Test that we can call cache_page the old way
  105. """
  106. def my_view(request):
  107. return "response"
  108. my_view_cached = cache_page(my_view, 123)
  109. self.assertEqual(my_view_cached(HttpRequest()), "response")
  110. my_view_cached2 = cache_page(my_view, 123, key_prefix="test")
  111. self.assertEqual(my_view_cached2(HttpRequest()), "response")
  112. my_view_cached3 = cache_page(my_view)
  113. self.assertEqual(my_view_cached3(HttpRequest()), "response")
  114. my_view_cached4 = cache_page()(my_view)
  115. self.assertEqual(my_view_cached4(HttpRequest()), "response")
  116. def test_require_safe_accepts_only_safe_methods(self):
  117. """
  118. Test for the require_safe decorator.
  119. A view returns either a response or an exception.
  120. Refs #15637.
  121. """
  122. def my_view(request):
  123. return HttpResponse("OK")
  124. my_safe_view = require_safe(my_view)
  125. request = HttpRequest()
  126. request.method = 'GET'
  127. self.assertTrue(isinstance(my_safe_view(request), HttpResponse))
  128. request.method = 'HEAD'
  129. self.assertTrue(isinstance(my_safe_view(request), HttpResponse))
  130. request.method = 'POST'
  131. self.assertTrue(isinstance(my_safe_view(request), HttpResponseNotAllowed))
  132. request.method = 'PUT'
  133. self.assertTrue(isinstance(my_safe_view(request), HttpResponseNotAllowed))
  134. request.method = 'DELETE'
  135. self.assertTrue(isinstance(my_safe_view(request), HttpResponseNotAllowed))
  136. # For testing method_decorator, a decorator that assumes a single argument.
  137. # We will get type arguments if there is a mismatch in the number of arguments.
  138. def simple_dec(func):
  139. def wrapper(arg):
  140. return func("test:" + arg)
  141. return wraps(func)(wrapper)
  142. simple_dec_m = method_decorator(simple_dec)
  143. # For testing method_decorator, two decorators that add an attribute to the function
  144. def myattr_dec(func):
  145. def wrapper(*args, **kwargs):
  146. return func(*args, **kwargs)
  147. wrapper.myattr = True
  148. return wraps(func)(wrapper)
  149. myattr_dec_m = method_decorator(myattr_dec)
  150. def myattr2_dec(func):
  151. def wrapper(*args, **kwargs):
  152. return func(*args, **kwargs)
  153. wrapper.myattr2 = True
  154. return wraps(func)(wrapper)
  155. myattr2_dec_m = method_decorator(myattr2_dec)
  156. class MethodDecoratorTests(TestCase):
  157. """
  158. Tests for method_decorator
  159. """
  160. def test_preserve_signature(self):
  161. class Test(object):
  162. @simple_dec_m
  163. def say(self, arg):
  164. return arg
  165. self.assertEqual("test:hello", Test().say("hello"))
  166. def test_preserve_attributes(self):
  167. # Sanity check myattr_dec and myattr2_dec
  168. @myattr_dec
  169. @myattr2_dec
  170. def func():
  171. pass
  172. self.assertEqual(getattr(func, 'myattr', False), True)
  173. self.assertEqual(getattr(func, 'myattr2', False), True)
  174. # Now check method_decorator
  175. class Test(object):
  176. @myattr_dec_m
  177. @myattr2_dec_m
  178. def method(self):
  179. "A method"
  180. pass
  181. self.assertEqual(getattr(Test().method, 'myattr', False), True)
  182. self.assertEqual(getattr(Test().method, 'myattr2', False), True)
  183. self.assertEqual(getattr(Test.method, 'myattr', False), True)
  184. self.assertEqual(getattr(Test.method, 'myattr2', False), True)
  185. self.assertEqual(Test.method.__doc__, 'A method')
  186. self.assertEqual(Test.method.im_func.__name__, 'method')
  187. class XFrameOptionsDecoratorsTests(TestCase):
  188. """
  189. Tests for the X-Frame-Options decorators.
  190. """
  191. def test_deny_decorator(self):
  192. """
  193. Ensures @xframe_options_deny properly sets the X-Frame-Options header.
  194. """
  195. @xframe_options_deny
  196. def a_view(request):
  197. return HttpResponse()
  198. r = a_view(HttpRequest())
  199. self.assertEqual(r['X-Frame-Options'], 'DENY')
  200. def test_sameorigin_decorator(self):
  201. """
  202. Ensures @xframe_options_sameorigin properly sets the X-Frame-Options
  203. header.
  204. """
  205. @xframe_options_sameorigin
  206. def a_view(request):
  207. return HttpResponse()
  208. r = a_view(HttpRequest())
  209. self.assertEqual(r['X-Frame-Options'], 'SAMEORIGIN')
  210. def test_exempt_decorator(self):
  211. """
  212. Ensures @xframe_options_exempt properly instructs the
  213. XFrameOptionsMiddleware to NOT set the header.
  214. """
  215. @xframe_options_exempt
  216. def a_view(request):
  217. return HttpResponse()
  218. req = HttpRequest()
  219. resp = a_view(req)
  220. self.assertEqual(resp.get('X-Frame-Options', None), None)
  221. self.assertTrue(resp.xframe_options_exempt)
  222. # Since the real purpose of the exempt decorator is to suppress
  223. # the middleware's functionality, let's make sure it actually works...
  224. r = XFrameOptionsMiddleware().process_response(req, resp)
  225. self.assertEqual(r.get('X-Frame-Options', None), None)