test_response.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. from __future__ import unicode_literals
  2. from datetime import datetime
  3. import os
  4. import pickle
  5. import time
  6. from django.test import RequestFactory, SimpleTestCase
  7. from django.conf import settings
  8. from django.template import Context, engines
  9. from django.template.response import (TemplateResponse, SimpleTemplateResponse,
  10. ContentNotRenderedError)
  11. from django.test import ignore_warnings, override_settings
  12. from django.utils._os import upath
  13. from django.utils.deprecation import RemovedInDjango20Warning
  14. def test_processor(request):
  15. return {'processors': 'yes'}
  16. test_processor_name = 'template_tests.test_response.test_processor'
  17. # A test middleware that installs a temporary URLConf
  18. class CustomURLConfMiddleware(object):
  19. def process_request(self, request):
  20. request.urlconf = 'template_tests.alternate_urls'
  21. class SimpleTemplateResponseTest(SimpleTestCase):
  22. def _response(self, template='foo', *args, **kwargs):
  23. template = engines['django'].from_string(template)
  24. return SimpleTemplateResponse(template, *args, **kwargs)
  25. def test_template_resolving(self):
  26. response = SimpleTemplateResponse('first/test.html')
  27. response.render()
  28. self.assertEqual(response.content, b'First template\n')
  29. templates = ['foo.html', 'second/test.html', 'first/test.html']
  30. response = SimpleTemplateResponse(templates)
  31. response.render()
  32. self.assertEqual(response.content, b'Second template\n')
  33. response = self._response()
  34. response.render()
  35. self.assertEqual(response.content, b'foo')
  36. def test_explicit_baking(self):
  37. # explicit baking
  38. response = self._response()
  39. self.assertFalse(response.is_rendered)
  40. response.render()
  41. self.assertTrue(response.is_rendered)
  42. def test_render(self):
  43. # response is not re-rendered without the render call
  44. response = self._response().render()
  45. self.assertEqual(response.content, b'foo')
  46. # rebaking doesn't change the rendered content
  47. template = engines['django'].from_string('bar{{ baz }}')
  48. response.template_name = template
  49. response.render()
  50. self.assertEqual(response.content, b'foo')
  51. # but rendered content can be overridden by manually
  52. # setting content
  53. response.content = 'bar'
  54. self.assertEqual(response.content, b'bar')
  55. def test_iteration_unrendered(self):
  56. # unrendered response raises an exception on iteration
  57. response = self._response()
  58. self.assertFalse(response.is_rendered)
  59. def iteration():
  60. for x in response:
  61. pass
  62. self.assertRaises(ContentNotRenderedError, iteration)
  63. self.assertFalse(response.is_rendered)
  64. def test_iteration_rendered(self):
  65. # iteration works for rendered responses
  66. response = self._response().render()
  67. res = [x for x in response]
  68. self.assertEqual(res, [b'foo'])
  69. def test_content_access_unrendered(self):
  70. # unrendered response raises an exception when content is accessed
  71. response = self._response()
  72. self.assertFalse(response.is_rendered)
  73. self.assertRaises(ContentNotRenderedError, lambda: response.content)
  74. self.assertFalse(response.is_rendered)
  75. def test_content_access_rendered(self):
  76. # rendered response content can be accessed
  77. response = self._response().render()
  78. self.assertEqual(response.content, b'foo')
  79. def test_set_content(self):
  80. # content can be overridden
  81. response = self._response()
  82. self.assertFalse(response.is_rendered)
  83. response.content = 'spam'
  84. self.assertTrue(response.is_rendered)
  85. self.assertEqual(response.content, b'spam')
  86. response.content = 'baz'
  87. self.assertEqual(response.content, b'baz')
  88. def test_dict_context(self):
  89. response = self._response('{{ foo }}{{ processors }}',
  90. {'foo': 'bar'})
  91. self.assertEqual(response.context_data, {'foo': 'bar'})
  92. response.render()
  93. self.assertEqual(response.content, b'bar')
  94. @ignore_warnings(category=RemovedInDjango20Warning)
  95. def test_context_instance(self):
  96. response = self._response('{{ foo }}{{ processors }}',
  97. Context({'foo': 'bar'}))
  98. self.assertEqual(response.context_data.__class__, Context)
  99. response.render()
  100. self.assertEqual(response.content, b'bar')
  101. def test_kwargs(self):
  102. response = self._response(content_type='application/json', status=504)
  103. self.assertEqual(response['content-type'], 'application/json')
  104. self.assertEqual(response.status_code, 504)
  105. def test_args(self):
  106. response = SimpleTemplateResponse('', {}, 'application/json', 504)
  107. self.assertEqual(response['content-type'], 'application/json')
  108. self.assertEqual(response.status_code, 504)
  109. def test_post_callbacks(self):
  110. "Rendering a template response triggers the post-render callbacks"
  111. post = []
  112. def post1(obj):
  113. post.append('post1')
  114. def post2(obj):
  115. post.append('post2')
  116. response = SimpleTemplateResponse('first/test.html', {})
  117. response.add_post_render_callback(post1)
  118. response.add_post_render_callback(post2)
  119. # When the content is rendered, all the callbacks are invoked, too.
  120. response.render()
  121. self.assertEqual(response.content, b'First template\n')
  122. self.assertEqual(post, ['post1', 'post2'])
  123. def test_pickling(self):
  124. # Create a template response. The context is
  125. # known to be unpickleable (e.g., a function).
  126. response = SimpleTemplateResponse('first/test.html', {
  127. 'value': 123,
  128. 'fn': datetime.now,
  129. })
  130. self.assertRaises(ContentNotRenderedError,
  131. pickle.dumps, response)
  132. # But if we render the response, we can pickle it.
  133. response.render()
  134. pickled_response = pickle.dumps(response)
  135. unpickled_response = pickle.loads(pickled_response)
  136. self.assertEqual(unpickled_response.content, response.content)
  137. self.assertEqual(unpickled_response['content-type'], response['content-type'])
  138. self.assertEqual(unpickled_response.status_code, response.status_code)
  139. # ...and the unpickled response doesn't have the
  140. # template-related attributes, so it can't be re-rendered
  141. template_attrs = ('template_name', 'context_data', '_post_render_callbacks')
  142. for attr in template_attrs:
  143. self.assertFalse(hasattr(unpickled_response, attr))
  144. # ...and requesting any of those attributes raises an exception
  145. for attr in template_attrs:
  146. with self.assertRaises(AttributeError):
  147. getattr(unpickled_response, attr)
  148. def test_repickling(self):
  149. response = SimpleTemplateResponse('first/test.html', {
  150. 'value': 123,
  151. 'fn': datetime.now,
  152. })
  153. self.assertRaises(ContentNotRenderedError,
  154. pickle.dumps, response)
  155. response.render()
  156. pickled_response = pickle.dumps(response)
  157. unpickled_response = pickle.loads(pickled_response)
  158. pickle.dumps(unpickled_response)
  159. def test_pickling_cookie(self):
  160. response = SimpleTemplateResponse('first/test.html', {
  161. 'value': 123,
  162. 'fn': datetime.now,
  163. })
  164. response.cookies['key'] = 'value'
  165. response.render()
  166. pickled_response = pickle.dumps(response, pickle.HIGHEST_PROTOCOL)
  167. unpickled_response = pickle.loads(pickled_response)
  168. self.assertEqual(unpickled_response.cookies['key'].value, 'value')
  169. @override_settings(TEMPLATES=[{
  170. 'BACKEND': 'django.template.backends.django.DjangoTemplates',
  171. 'DIRS': [os.path.join(os.path.dirname(upath(__file__)), 'templates')],
  172. 'OPTIONS': {
  173. 'context_processors': [test_processor_name],
  174. },
  175. }])
  176. class TemplateResponseTest(SimpleTestCase):
  177. def setUp(self):
  178. self.factory = RequestFactory()
  179. def _response(self, template='foo', *args, **kwargs):
  180. self._request = self.factory.get('/')
  181. template = engines['django'].from_string(template)
  182. return TemplateResponse(self._request, template, *args, **kwargs)
  183. def test_render(self):
  184. response = self._response('{{ foo }}{{ processors }}').render()
  185. self.assertEqual(response.content, b'yes')
  186. def test_render_with_requestcontext(self):
  187. response = self._response('{{ foo }}{{ processors }}',
  188. {'foo': 'bar'}).render()
  189. self.assertEqual(response.content, b'baryes')
  190. @ignore_warnings(category=RemovedInDjango20Warning)
  191. def test_render_with_context(self):
  192. response = self._response('{{ foo }}{{ processors }}',
  193. Context({'foo': 'bar'})).render()
  194. self.assertEqual(response.content, b'bar')
  195. def test_context_processor_priority(self):
  196. # context processors should be overridden by passed-in context
  197. response = self._response('{{ foo }}{{ processors }}',
  198. {'processors': 'no'}).render()
  199. self.assertEqual(response.content, b'no')
  200. def test_kwargs(self):
  201. response = self._response(content_type='application/json',
  202. status=504)
  203. self.assertEqual(response['content-type'], 'application/json')
  204. self.assertEqual(response.status_code, 504)
  205. def test_args(self):
  206. response = TemplateResponse(self.factory.get('/'), '', {},
  207. 'application/json', 504)
  208. self.assertEqual(response['content-type'], 'application/json')
  209. self.assertEqual(response.status_code, 504)
  210. @ignore_warnings(category=RemovedInDjango20Warning)
  211. def test_custom_app(self):
  212. self._response('{{ foo }}', current_app="foobar")
  213. self.assertEqual(self._request.current_app, 'foobar')
  214. def test_pickling(self):
  215. # Create a template response. The context is
  216. # known to be unpickleable (e.g., a function).
  217. response = TemplateResponse(self.factory.get('/'),
  218. 'first/test.html', {
  219. 'value': 123,
  220. 'fn': datetime.now,
  221. }
  222. )
  223. self.assertRaises(ContentNotRenderedError,
  224. pickle.dumps, response)
  225. # But if we render the response, we can pickle it.
  226. response.render()
  227. pickled_response = pickle.dumps(response)
  228. unpickled_response = pickle.loads(pickled_response)
  229. self.assertEqual(unpickled_response.content, response.content)
  230. self.assertEqual(unpickled_response['content-type'], response['content-type'])
  231. self.assertEqual(unpickled_response.status_code, response.status_code)
  232. # ...and the unpickled response doesn't have the
  233. # template-related attributes, so it can't be re-rendered
  234. template_attrs = ('template_name', 'context_data',
  235. '_post_render_callbacks', '_request', '_current_app')
  236. for attr in template_attrs:
  237. self.assertFalse(hasattr(unpickled_response, attr))
  238. # ...and requesting any of those attributes raises an exception
  239. for attr in template_attrs:
  240. with self.assertRaises(AttributeError):
  241. getattr(unpickled_response, attr)
  242. def test_repickling(self):
  243. response = SimpleTemplateResponse('first/test.html', {
  244. 'value': 123,
  245. 'fn': datetime.now,
  246. })
  247. self.assertRaises(ContentNotRenderedError,
  248. pickle.dumps, response)
  249. response.render()
  250. pickled_response = pickle.dumps(response)
  251. unpickled_response = pickle.loads(pickled_response)
  252. pickle.dumps(unpickled_response)
  253. @override_settings(
  254. MIDDLEWARE_CLASSES=list(settings.MIDDLEWARE_CLASSES) + [
  255. 'template_tests.test_response.CustomURLConfMiddleware'
  256. ],
  257. ROOT_URLCONF='template_tests.urls',
  258. )
  259. class CustomURLConfTest(SimpleTestCase):
  260. def test_custom_urlconf(self):
  261. response = self.client.get('/template_response_view/')
  262. self.assertEqual(response.status_code, 200)
  263. self.assertContains(response, 'This is where you can find the snark: /snark/')
  264. @override_settings(
  265. CACHE_MIDDLEWARE_SECONDS=2.0,
  266. MIDDLEWARE_CLASSES=list(settings.MIDDLEWARE_CLASSES) + [
  267. 'django.middleware.cache.FetchFromCacheMiddleware',
  268. 'django.middleware.cache.UpdateCacheMiddleware',
  269. ],
  270. ROOT_URLCONF='template_tests.alternate_urls',
  271. )
  272. class CacheMiddlewareTest(SimpleTestCase):
  273. def test_middleware_caching(self):
  274. response = self.client.get('/template_response_view/')
  275. self.assertEqual(response.status_code, 200)
  276. time.sleep(1.0)
  277. response2 = self.client.get('/template_response_view/')
  278. self.assertEqual(response2.status_code, 200)
  279. self.assertEqual(response.content, response2.content)
  280. time.sleep(2.0)
  281. # Let the cache expire and test again
  282. response2 = self.client.get('/template_response_view/')
  283. self.assertEqual(response2.status_code, 200)
  284. self.assertNotEqual(response.content, response2.content)