test_dates.py 31 KB

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