test_readtracker.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. from datetime import timedelta
  2. from django.contrib.auth import get_user_model
  3. from django.test import TestCase
  4. from django.utils import timezone
  5. from misago.acl import add_acl
  6. from misago.categories.models import Category
  7. from misago.threads import testutils
  8. from misago.users.models import AnonymousUser
  9. from misago.readtracker import categoriestracker, threadstracker
  10. class ReadTrackerTests(TestCase):
  11. def setUp(self):
  12. self.categories = [f for f in Category.objects.filter(role='forum')[:1]]
  13. self.category = self.categories[0]
  14. User = get_user_model()
  15. self.user = User.objects.create_user("Bob", "bob@test.com", "Pass.123")
  16. self.anon = AnonymousUser()
  17. def post_thread(self, datetime):
  18. return testutils.post_thread(
  19. category=self.category,
  20. started_on=datetime
  21. )
  22. class CategorysTrackerTests(ReadTrackerTests):
  23. def test_anon_empty_category_read(self):
  24. """anon users content is always read"""
  25. categoriestracker.make_read_aware(self.anon, self.categories)
  26. self.assertIsNone(self.category.last_post_on)
  27. self.assertTrue(self.category.is_read)
  28. def test_anon_category_with_recent_reply_read(self):
  29. """anon users content is always read"""
  30. categoriestracker.make_read_aware(self.anon, self.categories)
  31. self.category.last_post_on = timezone.now()
  32. self.assertTrue(self.category.is_read)
  33. def test_empty_category_is_read(self):
  34. """empty category is read for signed in user"""
  35. categoriestracker.make_read_aware(self.user, self.categories)
  36. self.assertTrue(self.category.is_read)
  37. def test_make_read_aware_sets_read_flag_for_empty_category(self):
  38. """make_read_aware sets read flag on empty category"""
  39. categoriestracker.make_read_aware(self.anon, self.categories)
  40. self.assertTrue(self.category.is_read)
  41. categoriestracker.make_read_aware(self.user, self.categories)
  42. self.assertTrue(self.category.is_read)
  43. def test_make_read_aware_sets_read_flag_for_category_with_old_thread(self):
  44. """make_read_aware sets read flag on category with old thread"""
  45. self.category.last_post_on = self.user.reads_cutoff - timedelta(days=1)
  46. categoriestracker.make_read_aware(self.user, self.categories)
  47. self.assertTrue(self.category.is_read)
  48. def test_make_read_aware_sets_unread_flag_for_category_with_new_thread(self):
  49. """make_read_aware sets unread flag on category with new thread"""
  50. self.category.last_post_on = self.user.reads_cutoff + timedelta(days=1)
  51. categoriestracker.make_read_aware(self.user, self.categories)
  52. self.assertFalse(self.category.is_read)
  53. def test_sync_record_for_empty_category(self):
  54. """sync_record sets read flag on empty category"""
  55. add_acl(self.user, self.categories)
  56. categoriestracker.sync_record(self.user, self.category)
  57. self.user.categoryread_set.get(category=self.category)
  58. categoriestracker.make_read_aware(self.user, self.categories)
  59. self.assertTrue(self.category.is_read)
  60. def test_sync_record_for_category_with_old_thread_and_reply(self):
  61. """
  62. sync_record sets read flag on category with old thread,
  63. then changes flag to unread when new reply is posted
  64. """
  65. self.post_thread(self.user.reads_cutoff - timedelta(days=1))
  66. add_acl(self.user, self.categories)
  67. categoriestracker.sync_record(self.user, self.category)
  68. self.user.categoryread_set.get(category=self.category)
  69. categoriestracker.make_read_aware(self.user, self.categories)
  70. self.assertTrue(self.category.is_read)
  71. thread = self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  72. categoriestracker.sync_record(self.user, self.category)
  73. categoriestracker.make_read_aware(self.user, self.categories)
  74. self.assertFalse(self.category.is_read)
  75. def test_sync_record_for_category_with_new_thread(self):
  76. """
  77. sync_record sets read flag on category with old thread,
  78. then keeps flag to unread when new reply is posted
  79. """
  80. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  81. add_acl(self.user, self.categories)
  82. categoriestracker.sync_record(self.user, self.category)
  83. self.user.categoryread_set.get(category=self.category)
  84. categoriestracker.make_read_aware(self.user, self.categories)
  85. self.assertFalse(self.category.is_read)
  86. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  87. categoriestracker.sync_record(self.user, self.category)
  88. categoriestracker.make_read_aware(self.user, self.categories)
  89. self.assertFalse(self.category.is_read)
  90. def test_sync_record_for_category_with_deleted_threads(self):
  91. """unread category reverts to read after its emptied"""
  92. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  93. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  94. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  95. add_acl(self.user, self.categories)
  96. categoriestracker.sync_record(self.user, self.category)
  97. categoriestracker.make_read_aware(self.user, self.categories)
  98. self.assertFalse(self.category.is_read)
  99. self.category.thread_set.all().delete()
  100. self.category.synchronize()
  101. self.category.save()
  102. categoriestracker.make_read_aware(self.user, self.categories)
  103. self.assertTrue(self.category.is_read)
  104. def test_sync_record_for_category_with_many_threads(self):
  105. """sync_record sets unread flag on category with many threads"""
  106. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  107. self.post_thread(self.user.reads_cutoff - timedelta(days=1))
  108. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  109. self.post_thread(self.user.reads_cutoff - timedelta(days=1))
  110. add_acl(self.user, self.categories)
  111. categoriestracker.sync_record(self.user, self.category)
  112. self.user.categoryread_set.get(category=self.category)
  113. categoriestracker.make_read_aware(self.user, self.categories)
  114. self.assertFalse(self.category.is_read)
  115. self.post_thread(self.user.reads_cutoff + timedelta(days=1))
  116. categoriestracker.sync_record(self.user, self.category)
  117. categoriestracker.make_read_aware(self.user, self.categories)
  118. self.assertFalse(self.category.is_read)
  119. class ThreadsTrackerTests(ReadTrackerTests):
  120. def setUp(self):
  121. super(ThreadsTrackerTests, self).setUp()
  122. self.thread = self.post_thread(timezone.now() - timedelta(days=10))
  123. def reply_thread(self, is_hidden=False, is_moderated=False):
  124. self.post = testutils.reply_thread(
  125. thread=self.thread,
  126. is_hidden=is_hidden,
  127. is_moderated=is_moderated,
  128. posted_on=timezone.now())
  129. return self.post
  130. def test_thread_read_for_guest(self):
  131. """threads are always read for guests"""
  132. threadstracker.make_read_aware(self.anon, self.thread)
  133. self.assertTrue(self.thread.is_read)
  134. self.reply_thread()
  135. threadstracker.make_read_aware(self.anon, [self.thread])
  136. self.assertTrue(self.thread.is_read)
  137. def test_thread_read_for_user(self):
  138. """thread is read for user"""
  139. threadstracker.make_read_aware(self.user, self.thread)
  140. self.assertTrue(self.thread.is_read)
  141. def test_thread_replied_unread_for_user(self):
  142. """replied thread is unread for user"""
  143. self.reply_thread(self.thread)
  144. threadstracker.make_read_aware(self.user, self.thread)
  145. self.assertFalse(self.thread.is_read)
  146. def _test_thread_read(self):
  147. """thread read flag is set for user, then its set as unread by reply"""
  148. self.reply_thread(self.thread)
  149. add_acl(self.user, self.categories)
  150. threadstracker.make_read_aware(self.user, self.thread)
  151. self.assertFalse(self.thread.is_read)
  152. threadstracker.read_thread(self.user, self.thread, self.post)
  153. threadstracker.make_read_aware(self.user, self.thread)
  154. self.assertTrue(self.thread.is_read)
  155. categoriestracker.make_read_aware(self.user, self.categories)
  156. self.assertTrue(self.category.is_read)
  157. self.thread.last_post_on = timezone.now()
  158. self.thread.save()
  159. self.category.synchronize()
  160. self.category.save()
  161. self.reply_thread()
  162. threadstracker.make_read_aware(self.user, self.thread)
  163. self.assertFalse(self.thread.is_read)
  164. categoriestracker.make_read_aware(self.user, self.categories)
  165. self.assertFalse(self.category.is_read)
  166. posts = [post for post in self.thread.post_set.order_by('id')]
  167. threadstracker.make_posts_read_aware(self.user, self.thread, posts)
  168. for post in posts[:-1]:
  169. self.assertTrue(post.is_read)
  170. self.assertFalse(posts[-1].is_read)