test_dates.py 35 KB

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