test_dates.py 33 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661
  1. from __future__ import unicode_literals
  2. import datetime
  3. from django.core.exceptions import ImproperlyConfigured
  4. from django.test import TestCase, override_settings, skipUnlessDBFeature
  5. from django.test.utils import requires_tz_support
  6. from django.utils import timezone
  7. from .models import Book, BookSigning
  8. def _make_books(n, base_date):
  9. for i in range(n):
  10. Book.objects.create(
  11. name='Book %d' % i,
  12. slug='book-%d' % i,
  13. pages=100 + i,
  14. pubdate=base_date - datetime.timedelta(days=i))
  15. @override_settings(ROOT_URLCONF='generic_views.urls')
  16. class ArchiveIndexViewTests(TestCase):
  17. fixtures = ['generic-views-test-data.json']
  18. def test_archive_view(self):
  19. res = self.client.get('/dates/books/')
  20. self.assertEqual(res.status_code, 200)
  21. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  22. self.assertEqual(list(res.context['latest']), list(Book.objects.all()))
  23. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  24. def test_archive_view_context_object_name(self):
  25. res = self.client.get('/dates/books/context_object_name/')
  26. self.assertEqual(res.status_code, 200)
  27. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  28. self.assertEqual(list(res.context['thingies']), list(Book.objects.all()))
  29. self.assertFalse('latest' in res.context)
  30. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  31. def test_empty_archive_view(self):
  32. Book.objects.all().delete()
  33. res = self.client.get('/dates/books/')
  34. self.assertEqual(res.status_code, 404)
  35. def test_allow_empty_archive_view(self):
  36. Book.objects.all().delete()
  37. res = self.client.get('/dates/books/allow_empty/')
  38. self.assertEqual(res.status_code, 200)
  39. self.assertEqual(list(res.context['date_list']), [])
  40. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  41. def test_archive_view_template(self):
  42. res = self.client.get('/dates/books/template_name/')
  43. self.assertEqual(res.status_code, 200)
  44. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  45. self.assertEqual(list(res.context['latest']), list(Book.objects.all()))
  46. self.assertTemplateUsed(res, 'generic_views/list.html')
  47. def test_archive_view_template_suffix(self):
  48. res = self.client.get('/dates/books/template_name_suffix/')
  49. self.assertEqual(res.status_code, 200)
  50. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  51. self.assertEqual(list(res.context['latest']), list(Book.objects.all()))
  52. self.assertTemplateUsed(res, 'generic_views/book_detail.html')
  53. def test_archive_view_invalid(self):
  54. self.assertRaises(ImproperlyConfigured, self.client.get, '/dates/books/invalid/')
  55. def test_archive_view_by_month(self):
  56. res = self.client.get('/dates/books/by_month/')
  57. self.assertEqual(res.status_code, 200)
  58. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'month', 'DESC')))
  59. def test_paginated_archive_view(self):
  60. _make_books(20, base_date=datetime.date.today())
  61. res = self.client.get('/dates/books/paginated/')
  62. self.assertEqual(res.status_code, 200)
  63. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  64. self.assertEqual(list(res.context['latest']), list(Book.objects.all()[0:10]))
  65. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  66. res = self.client.get('/dates/books/paginated/?page=2')
  67. self.assertEqual(res.status_code, 200)
  68. self.assertEqual(res.context['page_obj'].number, 2)
  69. self.assertEqual(list(res.context['latest']), list(Book.objects.all()[10:20]))
  70. def test_paginated_archive_view_does_not_load_entire_table(self):
  71. # Regression test for #18087
  72. _make_books(20, base_date=datetime.date.today())
  73. # 1 query for years list + 1 query for books
  74. with self.assertNumQueries(2):
  75. self.client.get('/dates/books/')
  76. # same as above + 1 query to test if books exist + 1 query to count them
  77. with self.assertNumQueries(4):
  78. self.client.get('/dates/books/paginated/')
  79. def test_no_duplicate_query(self):
  80. # Regression test for #18354
  81. with self.assertNumQueries(2):
  82. self.client.get('/dates/books/reverse/')
  83. def test_datetime_archive_view(self):
  84. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  85. res = self.client.get('/dates/booksignings/')
  86. self.assertEqual(res.status_code, 200)
  87. @requires_tz_support
  88. @skipUnlessDBFeature('has_zoneinfo_database')
  89. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  90. def test_aware_datetime_archive_view(self):
  91. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  92. res = self.client.get('/dates/booksignings/')
  93. self.assertEqual(res.status_code, 200)
  94. def test_date_list_order(self):
  95. """date_list should be sorted descending in index"""
  96. _make_books(5, base_date=datetime.date(2011, 12, 25))
  97. res = self.client.get('/dates/books/')
  98. self.assertEqual(res.status_code, 200)
  99. self.assertEqual(list(res.context['date_list']), list(reversed(sorted(res.context['date_list']))))
  100. def test_archive_view_custom_sorting(self):
  101. Book.objects.create(name="Zebras for Dummies", pages=600, pubdate=datetime.date(2007, 5, 1))
  102. res = self.client.get('/dates/books/sortedbyname/')
  103. self.assertEqual(res.status_code, 200)
  104. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  105. self.assertEqual(list(res.context['latest']), list(Book.objects.order_by('name').all()))
  106. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  107. def test_archive_view_custom_sorting_dec(self):
  108. Book.objects.create(name="Zebras for Dummies", pages=600, pubdate=datetime.date(2007, 5, 1))
  109. res = self.client.get('/dates/books/sortedbynamedec/')
  110. self.assertEqual(res.status_code, 200)
  111. self.assertEqual(list(res.context['date_list']), list(Book.objects.dates('pubdate', 'year', 'DESC')))
  112. self.assertEqual(list(res.context['latest']), list(Book.objects.order_by('-name').all()))
  113. self.assertTemplateUsed(res, 'generic_views/book_archive.html')
  114. @override_settings(ROOT_URLCONF='generic_views.urls')
  115. class YearArchiveViewTests(TestCase):
  116. fixtures = ['generic-views-test-data.json']
  117. def test_year_view(self):
  118. res = self.client.get('/dates/books/2008/')
  119. self.assertEqual(res.status_code, 200)
  120. self.assertEqual(list(res.context['date_list']), [datetime.date(2008, 10, 1)])
  121. self.assertEqual(res.context['year'], datetime.date(2008, 1, 1))
  122. self.assertTemplateUsed(res, 'generic_views/book_archive_year.html')
  123. # Since allow_empty=False, next/prev years must be valid (#7164)
  124. self.assertEqual(res.context['next_year'], None)
  125. self.assertEqual(res.context['previous_year'], datetime.date(2006, 1, 1))
  126. def test_year_view_make_object_list(self):
  127. res = self.client.get('/dates/books/2006/make_object_list/')
  128. self.assertEqual(res.status_code, 200)
  129. self.assertEqual(list(res.context['date_list']), [datetime.date(2006, 5, 1)])
  130. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2006)))
  131. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2006)))
  132. self.assertTemplateUsed(res, 'generic_views/book_archive_year.html')
  133. def test_year_view_empty(self):
  134. res = self.client.get('/dates/books/1999/')
  135. self.assertEqual(res.status_code, 404)
  136. res = self.client.get('/dates/books/1999/allow_empty/')
  137. self.assertEqual(res.status_code, 200)
  138. self.assertEqual(list(res.context['date_list']), [])
  139. self.assertEqual(list(res.context['book_list']), [])
  140. # Since allow_empty=True, next/prev are allowed to be empty years (#7164)
  141. self.assertEqual(res.context['next_year'], datetime.date(2000, 1, 1))
  142. self.assertEqual(res.context['previous_year'], datetime.date(1998, 1, 1))
  143. def test_year_view_allow_future(self):
  144. # Create a new book in the future
  145. year = datetime.date.today().year + 1
  146. Book.objects.create(name="The New New Testement", pages=600, pubdate=datetime.date(year, 1, 1))
  147. res = self.client.get('/dates/books/%s/' % year)
  148. self.assertEqual(res.status_code, 404)
  149. res = self.client.get('/dates/books/%s/allow_empty/' % year)
  150. self.assertEqual(res.status_code, 200)
  151. self.assertEqual(list(res.context['book_list']), [])
  152. res = self.client.get('/dates/books/%s/allow_future/' % year)
  153. self.assertEqual(res.status_code, 200)
  154. self.assertEqual(list(res.context['date_list']), [datetime.date(year, 1, 1)])
  155. def test_year_view_paginated(self):
  156. res = self.client.get('/dates/books/2006/paginated/')
  157. self.assertEqual(res.status_code, 200)
  158. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2006)))
  159. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2006)))
  160. self.assertTemplateUsed(res, 'generic_views/book_archive_year.html')
  161. def test_year_view_custom_sort_order(self):
  162. # Zebras comes after Dreaming by name, but before on '-pubdate' which is the default sorting
  163. Book.objects.create(name="Zebras for Dummies", pages=600, pubdate=datetime.date(2006, 9, 1))
  164. res = self.client.get('/dates/books/2006/sortedbyname/')
  165. self.assertEqual(res.status_code, 200)
  166. self.assertEqual(list(res.context['date_list']), [datetime.date(2006, 5, 1), datetime.date(2006, 9, 1)])
  167. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2006).order_by('name')))
  168. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2006).order_by('name')))
  169. self.assertTemplateUsed(res, 'generic_views/book_archive_year.html')
  170. def test_year_view_two_custom_sort_orders(self):
  171. Book.objects.create(name="Zebras for Dummies", pages=300, pubdate=datetime.date(2006, 9, 1))
  172. Book.objects.create(name="Hunting Hippos", pages=400, pubdate=datetime.date(2006, 3, 1))
  173. res = self.client.get('/dates/books/2006/sortedbypageandnamedec/')
  174. self.assertEqual(res.status_code, 200)
  175. self.assertEqual(list(res.context['date_list']), [datetime.date(2006, 3, 1), datetime.date(2006, 5, 1), datetime.date(2006, 9, 1)])
  176. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2006).order_by('pages', '-name')))
  177. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2006).order_by('pages', '-name')))
  178. self.assertTemplateUsed(res, 'generic_views/book_archive_year.html')
  179. def test_year_view_invalid_pattern(self):
  180. res = self.client.get('/dates/books/no_year/')
  181. self.assertEqual(res.status_code, 404)
  182. def test_no_duplicate_query(self):
  183. # Regression test for #18354
  184. with self.assertNumQueries(4):
  185. self.client.get('/dates/books/2008/reverse/')
  186. def test_datetime_year_view(self):
  187. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  188. res = self.client.get('/dates/booksignings/2008/')
  189. self.assertEqual(res.status_code, 200)
  190. @skipUnlessDBFeature('has_zoneinfo_database')
  191. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  192. def test_aware_datetime_year_view(self):
  193. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  194. res = self.client.get('/dates/booksignings/2008/')
  195. self.assertEqual(res.status_code, 200)
  196. def test_date_list_order(self):
  197. """date_list should be sorted ascending in year view"""
  198. _make_books(10, base_date=datetime.date(2011, 12, 25))
  199. res = self.client.get('/dates/books/2011/')
  200. self.assertEqual(list(res.context['date_list']), list(sorted(res.context['date_list'])))
  201. @override_settings(ROOT_URLCONF='generic_views.urls')
  202. class MonthArchiveViewTests(TestCase):
  203. fixtures = ['generic-views-test-data.json']
  204. def test_month_view(self):
  205. res = self.client.get('/dates/books/2008/oct/')
  206. self.assertEqual(res.status_code, 200)
  207. self.assertTemplateUsed(res, 'generic_views/book_archive_month.html')
  208. self.assertEqual(list(res.context['date_list']), [datetime.date(2008, 10, 1)])
  209. self.assertEqual(list(res.context['book_list']),
  210. list(Book.objects.filter(pubdate=datetime.date(2008, 10, 1))))
  211. self.assertEqual(res.context['month'], datetime.date(2008, 10, 1))
  212. # Since allow_empty=False, next/prev months must be valid (#7164)
  213. self.assertEqual(res.context['next_month'], None)
  214. self.assertEqual(res.context['previous_month'], datetime.date(2006, 5, 1))
  215. def test_month_view_allow_empty(self):
  216. # allow_empty = False, empty month
  217. res = self.client.get('/dates/books/2000/jan/')
  218. self.assertEqual(res.status_code, 404)
  219. # allow_empty = True, empty month
  220. res = self.client.get('/dates/books/2000/jan/allow_empty/')
  221. self.assertEqual(res.status_code, 200)
  222. self.assertEqual(list(res.context['date_list']), [])
  223. self.assertEqual(list(res.context['book_list']), [])
  224. self.assertEqual(res.context['month'], datetime.date(2000, 1, 1))
  225. # Since allow_empty=True, next/prev are allowed to be empty months (#7164)
  226. self.assertEqual(res.context['next_month'], datetime.date(2000, 2, 1))
  227. self.assertEqual(res.context['previous_month'], datetime.date(1999, 12, 1))
  228. # allow_empty but not allow_future: next_month should be empty (#7164)
  229. url = datetime.date.today().strftime('/dates/books/%Y/%b/allow_empty/').lower()
  230. res = self.client.get(url)
  231. self.assertEqual(res.status_code, 200)
  232. self.assertEqual(res.context['next_month'], None)
  233. def test_month_view_allow_future(self):
  234. future = (datetime.date.today() + datetime.timedelta(days=60)).replace(day=1)
  235. urlbit = future.strftime('%Y/%b').lower()
  236. b = Book.objects.create(name="The New New Testement", pages=600, pubdate=future)
  237. # allow_future = False, future month
  238. res = self.client.get('/dates/books/%s/' % urlbit)
  239. self.assertEqual(res.status_code, 404)
  240. # allow_future = True, valid future month
  241. res = self.client.get('/dates/books/%s/allow_future/' % urlbit)
  242. self.assertEqual(res.status_code, 200)
  243. self.assertEqual(res.context['date_list'][0], b.pubdate)
  244. self.assertEqual(list(res.context['book_list']), [b])
  245. self.assertEqual(res.context['month'], future)
  246. # Since allow_future = True but not allow_empty, next/prev are not
  247. # allowed to be empty months (#7164)
  248. self.assertEqual(res.context['next_month'], None)
  249. self.assertEqual(res.context['previous_month'], datetime.date(2008, 10, 1))
  250. # allow_future, but not allow_empty, with a current month. So next
  251. # should be in the future (yup, #7164, again)
  252. res = self.client.get('/dates/books/2008/oct/allow_future/')
  253. self.assertEqual(res.status_code, 200)
  254. self.assertEqual(res.context['next_month'], future)
  255. self.assertEqual(res.context['previous_month'], datetime.date(2006, 5, 1))
  256. def test_month_view_paginated(self):
  257. res = self.client.get('/dates/books/2008/oct/paginated/')
  258. self.assertEqual(res.status_code, 200)
  259. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2008, pubdate__month=10)))
  260. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2008, pubdate__month=10)))
  261. self.assertTemplateUsed(res, 'generic_views/book_archive_month.html')
  262. def test_custom_month_format(self):
  263. res = self.client.get('/dates/books/2008/10/')
  264. self.assertEqual(res.status_code, 200)
  265. def test_month_view_invalid_pattern(self):
  266. res = self.client.get('/dates/books/2007/no_month/')
  267. self.assertEqual(res.status_code, 404)
  268. def test_previous_month_without_content(self):
  269. "Content can exist on any day of the previous month. Refs #14711"
  270. self.pubdate_list = [
  271. datetime.date(2010, month, day)
  272. for month, day in ((9, 1), (10, 2), (11, 3))
  273. ]
  274. for pubdate in self.pubdate_list:
  275. name = str(pubdate)
  276. Book.objects.create(name=name, slug=name, pages=100, pubdate=pubdate)
  277. res = self.client.get('/dates/books/2010/nov/allow_empty/')
  278. self.assertEqual(res.status_code, 200)
  279. self.assertEqual(res.context['previous_month'], datetime.date(2010, 10, 1))
  280. # The following test demonstrates the bug
  281. res = self.client.get('/dates/books/2010/nov/')
  282. self.assertEqual(res.status_code, 200)
  283. self.assertEqual(res.context['previous_month'], datetime.date(2010, 10, 1))
  284. # The bug does not occur here because a Book with pubdate of Sep 1 exists
  285. res = self.client.get('/dates/books/2010/oct/')
  286. self.assertEqual(res.status_code, 200)
  287. self.assertEqual(res.context['previous_month'], datetime.date(2010, 9, 1))
  288. def test_datetime_month_view(self):
  289. BookSigning.objects.create(event_date=datetime.datetime(2008, 2, 1, 12, 0))
  290. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  291. BookSigning.objects.create(event_date=datetime.datetime(2008, 6, 3, 12, 0))
  292. res = self.client.get('/dates/booksignings/2008/apr/')
  293. self.assertEqual(res.status_code, 200)
  294. @skipUnlessDBFeature('has_zoneinfo_database')
  295. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  296. def test_aware_datetime_month_view(self):
  297. BookSigning.objects.create(event_date=datetime.datetime(2008, 2, 1, 12, 0, tzinfo=timezone.utc))
  298. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  299. BookSigning.objects.create(event_date=datetime.datetime(2008, 6, 3, 12, 0, tzinfo=timezone.utc))
  300. res = self.client.get('/dates/booksignings/2008/apr/')
  301. self.assertEqual(res.status_code, 200)
  302. def test_date_list_order(self):
  303. """date_list should be sorted ascending in month view"""
  304. _make_books(10, base_date=datetime.date(2011, 12, 25))
  305. res = self.client.get('/dates/books/2011/dec/')
  306. self.assertEqual(list(res.context['date_list']), list(sorted(res.context['date_list'])))
  307. @override_settings(ROOT_URLCONF='generic_views.urls')
  308. class WeekArchiveViewTests(TestCase):
  309. fixtures = ['generic-views-test-data.json']
  310. def test_week_view(self):
  311. res = self.client.get('/dates/books/2008/week/39/')
  312. self.assertEqual(res.status_code, 200)
  313. self.assertTemplateUsed(res, 'generic_views/book_archive_week.html')
  314. self.assertEqual(res.context['book_list'][0], Book.objects.get(pubdate=datetime.date(2008, 10, 1)))
  315. self.assertEqual(res.context['week'], datetime.date(2008, 9, 28))
  316. # Since allow_empty=False, next/prev weeks must be valid
  317. self.assertEqual(res.context['next_week'], None)
  318. self.assertEqual(res.context['previous_week'], datetime.date(2006, 4, 30))
  319. def test_week_view_allow_empty(self):
  320. # allow_empty = False, empty week
  321. res = self.client.get('/dates/books/2008/week/12/')
  322. self.assertEqual(res.status_code, 404)
  323. # allow_empty = True, empty month
  324. res = self.client.get('/dates/books/2008/week/12/allow_empty/')
  325. self.assertEqual(res.status_code, 200)
  326. self.assertEqual(list(res.context['book_list']), [])
  327. self.assertEqual(res.context['week'], datetime.date(2008, 3, 23))
  328. # Since allow_empty=True, next/prev are allowed to be empty weeks
  329. self.assertEqual(res.context['next_week'], datetime.date(2008, 3, 30))
  330. self.assertEqual(res.context['previous_week'], datetime.date(2008, 3, 16))
  331. # allow_empty but not allow_future: next_week should be empty
  332. url = datetime.date.today().strftime('/dates/books/%Y/week/%U/allow_empty/').lower()
  333. res = self.client.get(url)
  334. self.assertEqual(res.status_code, 200)
  335. self.assertEqual(res.context['next_week'], None)
  336. def test_week_view_allow_future(self):
  337. # January 7th always falls in week 1, given Python's definition of week numbers
  338. future = datetime.date(datetime.date.today().year + 1, 1, 7)
  339. future_sunday = future - datetime.timedelta(days=(future.weekday() + 1) % 7)
  340. b = Book.objects.create(name="The New New Testement", pages=600, pubdate=future)
  341. res = self.client.get('/dates/books/%s/week/1/' % future.year)
  342. self.assertEqual(res.status_code, 404)
  343. res = self.client.get('/dates/books/%s/week/1/allow_future/' % future.year)
  344. self.assertEqual(res.status_code, 200)
  345. self.assertEqual(list(res.context['book_list']), [b])
  346. self.assertEqual(res.context['week'], future_sunday)
  347. # Since allow_future = True but not allow_empty, next/prev are not
  348. # allowed to be empty weeks
  349. self.assertEqual(res.context['next_week'], None)
  350. self.assertEqual(res.context['previous_week'], datetime.date(2008, 9, 28))
  351. # allow_future, but not allow_empty, with a current week. So next
  352. # should be in the future
  353. res = self.client.get('/dates/books/2008/week/39/allow_future/')
  354. self.assertEqual(res.status_code, 200)
  355. self.assertEqual(res.context['next_week'], future_sunday)
  356. self.assertEqual(res.context['previous_week'], datetime.date(2006, 4, 30))
  357. def test_week_view_paginated(self):
  358. week_start = datetime.date(2008, 9, 28)
  359. week_end = week_start + datetime.timedelta(days=7)
  360. res = self.client.get('/dates/books/2008/week/39/')
  361. self.assertEqual(res.status_code, 200)
  362. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__gte=week_start, pubdate__lt=week_end)))
  363. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__gte=week_start, pubdate__lt=week_end)))
  364. self.assertTemplateUsed(res, 'generic_views/book_archive_week.html')
  365. def test_week_view_invalid_pattern(self):
  366. res = self.client.get('/dates/books/2007/week/no_week/')
  367. self.assertEqual(res.status_code, 404)
  368. def test_week_start_Monday(self):
  369. # Regression for #14752
  370. res = self.client.get('/dates/books/2008/week/39/')
  371. self.assertEqual(res.status_code, 200)
  372. self.assertEqual(res.context['week'], datetime.date(2008, 9, 28))
  373. res = self.client.get('/dates/books/2008/week/39/monday/')
  374. self.assertEqual(res.status_code, 200)
  375. self.assertEqual(res.context['week'], datetime.date(2008, 9, 29))
  376. def test_datetime_week_view(self):
  377. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  378. res = self.client.get('/dates/booksignings/2008/week/13/')
  379. self.assertEqual(res.status_code, 200)
  380. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  381. def test_aware_datetime_week_view(self):
  382. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  383. res = self.client.get('/dates/booksignings/2008/week/13/')
  384. self.assertEqual(res.status_code, 200)
  385. @override_settings(ROOT_URLCONF='generic_views.urls')
  386. class DayArchiveViewTests(TestCase):
  387. fixtures = ['generic-views-test-data.json']
  388. def test_day_view(self):
  389. res = self.client.get('/dates/books/2008/oct/01/')
  390. self.assertEqual(res.status_code, 200)
  391. self.assertTemplateUsed(res, 'generic_views/book_archive_day.html')
  392. self.assertEqual(list(res.context['book_list']),
  393. list(Book.objects.filter(pubdate=datetime.date(2008, 10, 1))))
  394. self.assertEqual(res.context['day'], datetime.date(2008, 10, 1))
  395. # Since allow_empty=False, next/prev days must be valid.
  396. self.assertEqual(res.context['next_day'], None)
  397. self.assertEqual(res.context['previous_day'], datetime.date(2006, 5, 1))
  398. def test_day_view_allow_empty(self):
  399. # allow_empty = False, empty month
  400. res = self.client.get('/dates/books/2000/jan/1/')
  401. self.assertEqual(res.status_code, 404)
  402. # allow_empty = True, empty month
  403. res = self.client.get('/dates/books/2000/jan/1/allow_empty/')
  404. self.assertEqual(res.status_code, 200)
  405. self.assertEqual(list(res.context['book_list']), [])
  406. self.assertEqual(res.context['day'], datetime.date(2000, 1, 1))
  407. # Since it's allow empty, next/prev are allowed to be empty months (#7164)
  408. self.assertEqual(res.context['next_day'], datetime.date(2000, 1, 2))
  409. self.assertEqual(res.context['previous_day'], datetime.date(1999, 12, 31))
  410. # allow_empty but not allow_future: next_month should be empty (#7164)
  411. url = datetime.date.today().strftime('/dates/books/%Y/%b/%d/allow_empty/').lower()
  412. res = self.client.get(url)
  413. self.assertEqual(res.status_code, 200)
  414. self.assertEqual(res.context['next_day'], None)
  415. def test_day_view_allow_future(self):
  416. future = (datetime.date.today() + datetime.timedelta(days=60))
  417. urlbit = future.strftime('%Y/%b/%d').lower()
  418. b = Book.objects.create(name="The New New Testement", pages=600, pubdate=future)
  419. # allow_future = False, future month
  420. res = self.client.get('/dates/books/%s/' % urlbit)
  421. self.assertEqual(res.status_code, 404)
  422. # allow_future = True, valid future month
  423. res = self.client.get('/dates/books/%s/allow_future/' % urlbit)
  424. self.assertEqual(res.status_code, 200)
  425. self.assertEqual(list(res.context['book_list']), [b])
  426. self.assertEqual(res.context['day'], future)
  427. # allow_future but not allow_empty, next/prev must be valid
  428. self.assertEqual(res.context['next_day'], None)
  429. self.assertEqual(res.context['previous_day'], datetime.date(2008, 10, 1))
  430. # allow_future, but not allow_empty, with a current month.
  431. res = self.client.get('/dates/books/2008/oct/01/allow_future/')
  432. self.assertEqual(res.status_code, 200)
  433. self.assertEqual(res.context['next_day'], future)
  434. self.assertEqual(res.context['previous_day'], datetime.date(2006, 5, 1))
  435. # allow_future for yesterday, next_day is today (#17192)
  436. today = datetime.date.today()
  437. yesterday = today - datetime.timedelta(days=1)
  438. res = self.client.get('/dates/books/%s/allow_empty_and_future/'
  439. % yesterday.strftime('%Y/%b/%d').lower())
  440. self.assertEqual(res.context['next_day'], today)
  441. def test_day_view_paginated(self):
  442. res = self.client.get('/dates/books/2008/oct/1/')
  443. self.assertEqual(res.status_code, 200)
  444. self.assertEqual(list(res.context['book_list']), list(Book.objects.filter(pubdate__year=2008, pubdate__month=10, pubdate__day=1)))
  445. self.assertEqual(list(res.context['object_list']), list(Book.objects.filter(pubdate__year=2008, pubdate__month=10, pubdate__day=1)))
  446. self.assertTemplateUsed(res, 'generic_views/book_archive_day.html')
  447. def test_next_prev_context(self):
  448. res = self.client.get('/dates/books/2008/oct/01/')
  449. self.assertEqual(res.content, b"Archive for Oct. 1, 2008. Previous day is May 1, 2006\n")
  450. def test_custom_month_format(self):
  451. res = self.client.get('/dates/books/2008/10/01/')
  452. self.assertEqual(res.status_code, 200)
  453. def test_day_view_invalid_pattern(self):
  454. res = self.client.get('/dates/books/2007/oct/no_day/')
  455. self.assertEqual(res.status_code, 404)
  456. def test_today_view(self):
  457. res = self.client.get('/dates/books/today/')
  458. self.assertEqual(res.status_code, 404)
  459. res = self.client.get('/dates/books/today/allow_empty/')
  460. self.assertEqual(res.status_code, 200)
  461. self.assertEqual(res.context['day'], datetime.date.today())
  462. def test_datetime_day_view(self):
  463. BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  464. res = self.client.get('/dates/booksignings/2008/apr/2/')
  465. self.assertEqual(res.status_code, 200)
  466. @requires_tz_support
  467. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  468. def test_aware_datetime_day_view(self):
  469. bs = BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  470. res = self.client.get('/dates/booksignings/2008/apr/2/')
  471. self.assertEqual(res.status_code, 200)
  472. # 2008-04-02T00:00:00+03:00 (beginning of day) > 2008-04-01T22:00:00+00:00 (book signing event date)
  473. bs.event_date = datetime.datetime(2008, 4, 1, 22, 0, tzinfo=timezone.utc)
  474. bs.save()
  475. res = self.client.get('/dates/booksignings/2008/apr/2/')
  476. self.assertEqual(res.status_code, 200)
  477. # 2008-04-03T00:00:00+03:00 (end of day) > 2008-04-02T22:00:00+00:00 (book signing event date)
  478. bs.event_date = datetime.datetime(2008, 4, 2, 22, 0, tzinfo=timezone.utc)
  479. bs.save()
  480. res = self.client.get('/dates/booksignings/2008/apr/2/')
  481. self.assertEqual(res.status_code, 404)
  482. @override_settings(ROOT_URLCONF='generic_views.urls')
  483. class DateDetailViewTests(TestCase):
  484. fixtures = ['generic-views-test-data.json']
  485. def test_date_detail_by_pk(self):
  486. res = self.client.get('/dates/books/2008/oct/01/1/')
  487. self.assertEqual(res.status_code, 200)
  488. self.assertEqual(res.context['object'], Book.objects.get(pk=1))
  489. self.assertEqual(res.context['book'], Book.objects.get(pk=1))
  490. self.assertTemplateUsed(res, 'generic_views/book_detail.html')
  491. def test_date_detail_by_slug(self):
  492. res = self.client.get('/dates/books/2006/may/01/byslug/dreaming-in-code/')
  493. self.assertEqual(res.status_code, 200)
  494. self.assertEqual(res.context['book'], Book.objects.get(slug='dreaming-in-code'))
  495. def test_date_detail_custom_month_format(self):
  496. res = self.client.get('/dates/books/2008/10/01/1/')
  497. self.assertEqual(res.status_code, 200)
  498. self.assertEqual(res.context['book'], Book.objects.get(pk=1))
  499. def test_date_detail_allow_future(self):
  500. future = (datetime.date.today() + datetime.timedelta(days=60))
  501. urlbit = future.strftime('%Y/%b/%d').lower()
  502. b = Book.objects.create(name="The New New Testement", slug="new-new", pages=600, pubdate=future)
  503. res = self.client.get('/dates/books/%s/new-new/' % urlbit)
  504. self.assertEqual(res.status_code, 404)
  505. res = self.client.get('/dates/books/%s/%s/allow_future/' % (urlbit, b.id))
  506. self.assertEqual(res.status_code, 200)
  507. self.assertEqual(res.context['book'], b)
  508. self.assertTemplateUsed(res, 'generic_views/book_detail.html')
  509. def test_invalid_url(self):
  510. self.assertRaises(AttributeError, self.client.get, "/dates/books/2008/oct/01/nopk/")
  511. def test_get_object_custom_queryset(self):
  512. """
  513. Ensure that custom querysets are used when provided to
  514. BaseDateDetailView.get_object()
  515. Refs #16918.
  516. """
  517. res = self.client.get(
  518. '/dates/books/get_object_custom_queryset/2006/may/01/2/')
  519. self.assertEqual(res.status_code, 200)
  520. self.assertEqual(res.context['object'], Book.objects.get(pk=2))
  521. self.assertEqual(res.context['book'], Book.objects.get(pk=2))
  522. self.assertTemplateUsed(res, 'generic_views/book_detail.html')
  523. res = self.client.get(
  524. '/dates/books/get_object_custom_queryset/2008/oct/01/1/')
  525. self.assertEqual(res.status_code, 404)
  526. def test_datetime_date_detail(self):
  527. bs = BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0))
  528. res = self.client.get('/dates/booksignings/2008/apr/2/%d/' % bs.pk)
  529. self.assertEqual(res.status_code, 200)
  530. @requires_tz_support
  531. @override_settings(USE_TZ=True, TIME_ZONE='Africa/Nairobi')
  532. def test_aware_datetime_date_detail(self):
  533. bs = BookSigning.objects.create(event_date=datetime.datetime(2008, 4, 2, 12, 0, tzinfo=timezone.utc))
  534. res = self.client.get('/dates/booksignings/2008/apr/2/%d/' % bs.pk)
  535. self.assertEqual(res.status_code, 200)
  536. # 2008-04-02T00:00:00+03:00 (beginning of day) > 2008-04-01T22:00:00+00:00 (book signing event date)
  537. bs.event_date = datetime.datetime(2008, 4, 1, 22, 0, tzinfo=timezone.utc)
  538. bs.save()
  539. res = self.client.get('/dates/booksignings/2008/apr/2/%d/' % bs.pk)
  540. self.assertEqual(res.status_code, 200)
  541. # 2008-04-03T00:00:00+03:00 (end of day) > 2008-04-02T22:00:00+00:00 (book signing event date)
  542. bs.event_date = datetime.datetime(2008, 4, 2, 22, 0, tzinfo=timezone.utc)
  543. bs.save()
  544. res = self.client.get('/dates/booksignings/2008/apr/2/%d/' % bs.pk)
  545. self.assertEqual(res.status_code, 404)