test_thread_model.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367
  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.categories.models import Category
  6. from ..models import Poll, Post, Thread, ThreadParticipant
  7. class ThreadModelTests(TestCase):
  8. def setUp(self):
  9. datetime = timezone.now()
  10. self.category = Category.objects.all_categories()[:1][0]
  11. self.thread = Thread(
  12. category=self.category,
  13. started_on=datetime,
  14. starter_name='Tester',
  15. starter_slug='tester',
  16. last_post_on=datetime,
  17. last_poster_name='Tester',
  18. last_poster_slug='tester'
  19. )
  20. self.thread.set_title("Test thread")
  21. self.thread.save()
  22. post = Post.objects.create(
  23. category=self.category,
  24. thread=self.thread,
  25. poster_name='Tester',
  26. poster_ip='127.0.0.1',
  27. original="Hello! I am test message!",
  28. parsed="<p>Hello! I am test message!</p>",
  29. checksum="nope",
  30. posted_on=datetime,
  31. updated_on=datetime
  32. )
  33. self.thread.first_post = post
  34. self.thread.last_post = post
  35. self.thread.save()
  36. def test_synchronize(self):
  37. """synchronize method updates thread data to reflect its contents"""
  38. User = get_user_model()
  39. user = User.objects.create_user("Bob", "bob@boberson.com", "Pass.123")
  40. self.assertEqual(self.thread.replies, 0)
  41. datetime = timezone.now() + timedelta(5)
  42. post = Post.objects.create(
  43. category=self.category,
  44. thread=self.thread,
  45. poster=user,
  46. poster_name=user.username,
  47. poster_ip='127.0.0.1',
  48. original="Hello! I am test message!",
  49. parsed="<p>Hello! I am test message!</p>",
  50. checksum="nope",
  51. posted_on=datetime,
  52. updated_on=datetime
  53. )
  54. # first sync call, updates last thread
  55. self.thread.synchronize()
  56. self.assertEqual(self.thread.last_post, post)
  57. self.assertEqual(self.thread.last_post_on, post.posted_on)
  58. self.assertEqual(self.thread.last_poster, user)
  59. self.assertEqual(self.thread.last_poster_name, user.username)
  60. self.assertEqual(self.thread.last_poster_slug, user.slug)
  61. self.assertFalse(self.thread.has_reported_posts)
  62. self.assertFalse(self.thread.has_unapproved_posts)
  63. self.assertFalse(self.thread.has_hidden_posts)
  64. self.assertEqual(self.thread.replies, 1)
  65. # add unapproved post
  66. unapproved_post = Post.objects.create(
  67. category=self.category,
  68. thread=self.thread,
  69. poster=user,
  70. poster_name=user.username,
  71. poster_ip='127.0.0.1',
  72. original="Hello! I am test message!",
  73. parsed="<p>Hello! I am test message!</p>",
  74. checksum="nope",
  75. posted_on=datetime + timedelta(5),
  76. updated_on=datetime + timedelta(5),
  77. is_unapproved=True
  78. )
  79. self.thread.synchronize()
  80. self.assertEqual(self.thread.last_post, post)
  81. self.assertEqual(self.thread.last_post_on, post.posted_on)
  82. self.assertEqual(self.thread.last_poster, user)
  83. self.assertEqual(self.thread.last_poster_name, user.username)
  84. self.assertEqual(self.thread.last_poster_slug, user.slug)
  85. self.assertFalse(self.thread.has_reported_posts)
  86. self.assertTrue(self.thread.has_unapproved_posts)
  87. self.assertFalse(self.thread.has_hidden_posts)
  88. self.assertEqual(self.thread.replies, 1)
  89. # add hidden post
  90. hidden_post = Post.objects.create(
  91. category=self.category,
  92. thread=self.thread,
  93. poster=user,
  94. poster_name=user.username,
  95. poster_ip='127.0.0.1',
  96. original="Hello! I am test message!",
  97. parsed="<p>Hello! I am test message!</p>",
  98. checksum="nope",
  99. posted_on=datetime + timedelta(10),
  100. updated_on=datetime + timedelta(10),
  101. is_hidden=True
  102. )
  103. self.thread.synchronize()
  104. self.assertEqual(self.thread.last_post, hidden_post)
  105. self.assertEqual(self.thread.last_post_on, hidden_post.posted_on)
  106. self.assertEqual(self.thread.last_poster, user)
  107. self.assertEqual(self.thread.last_poster_name, user.username)
  108. self.assertEqual(self.thread.last_poster_slug, user.slug)
  109. self.assertFalse(self.thread.has_reported_posts)
  110. self.assertTrue(self.thread.has_unapproved_posts)
  111. self.assertTrue(self.thread.has_hidden_posts)
  112. self.assertEqual(self.thread.replies, 2)
  113. # unhide post
  114. hidden_post.is_hidden = False
  115. hidden_post.save()
  116. # last post changed to unhidden one
  117. self.thread.synchronize()
  118. self.assertEqual(self.thread.last_post, hidden_post)
  119. self.assertEqual(self.thread.last_post_on, hidden_post.posted_on)
  120. self.assertEqual(self.thread.last_poster, user)
  121. self.assertEqual(self.thread.last_poster_name, user.username)
  122. self.assertEqual(self.thread.last_poster_slug, user.slug)
  123. self.assertFalse(self.thread.has_reported_posts)
  124. self.assertTrue(self.thread.has_unapproved_posts)
  125. self.assertFalse(self.thread.has_hidden_posts)
  126. self.assertEqual(self.thread.replies, 2)
  127. # unmoderate post
  128. unapproved_post.is_unapproved = False
  129. unapproved_post.save()
  130. # last post not changed, but flags and count did
  131. self.thread.synchronize()
  132. self.assertEqual(self.thread.last_post, hidden_post)
  133. self.assertEqual(self.thread.last_post_on, hidden_post.posted_on)
  134. self.assertEqual(self.thread.last_poster, user)
  135. self.assertEqual(self.thread.last_poster_name, user.username)
  136. self.assertEqual(self.thread.last_poster_slug, user.slug)
  137. self.assertFalse(self.thread.has_reported_posts)
  138. self.assertFalse(self.thread.has_unapproved_posts)
  139. self.assertFalse(self.thread.has_hidden_posts)
  140. self.assertEqual(self.thread.replies, 3)
  141. # add event post
  142. event = Post.objects.create(
  143. category=self.category,
  144. thread=self.thread,
  145. poster=user,
  146. poster_name=user.username,
  147. poster_ip='127.0.0.1',
  148. original="-",
  149. parsed="-",
  150. checksum="nope",
  151. posted_on=datetime + timedelta(10),
  152. updated_on=datetime + timedelta(10),
  153. is_event=True
  154. )
  155. self.thread.synchronize()
  156. self.assertEqual(self.thread.last_post, event)
  157. self.assertEqual(self.thread.last_post_on, event.posted_on)
  158. self.assertEqual(self.thread.last_poster, user)
  159. self.assertEqual(self.thread.last_poster_name, user.username)
  160. self.assertEqual(self.thread.last_poster_slug, user.slug)
  161. self.assertTrue(self.thread.last_post_is_event)
  162. self.assertTrue(self.thread.has_events)
  163. self.assertFalse(self.thread.has_reported_posts)
  164. self.assertFalse(self.thread.has_unapproved_posts)
  165. self.assertFalse(self.thread.has_hidden_posts)
  166. # events don't count to reply count
  167. self.assertEqual(self.thread.replies, 3)
  168. # create another post to provoke other has_events resolution path
  169. Post.objects.create(
  170. category=self.category,
  171. thread=self.thread,
  172. poster=user,
  173. poster_name=user.username,
  174. poster_ip='127.0.0.1',
  175. original="Hello! I am test message!",
  176. parsed="<p>Hello! I am test message!</p>",
  177. checksum="nope",
  178. posted_on=datetime,
  179. updated_on=datetime
  180. )
  181. self.thread.synchronize()
  182. self.assertFalse(self.thread.last_post_is_event)
  183. self.assertTrue(self.thread.has_events)
  184. # remove event
  185. event.delete()
  186. self.thread.synchronize()
  187. self.assertFalse(self.thread.last_post_is_event)
  188. self.assertFalse(self.thread.has_events)
  189. # has poll flag
  190. self.assertFalse(self.thread.has_poll)
  191. Poll.objects.create(
  192. thread=self.thread,
  193. category=self.category,
  194. poster_name='test',
  195. poster_slug='test',
  196. poster_ip='127.0.0.1',
  197. choices=[]
  198. )
  199. self.thread.synchronize()
  200. self.assertTrue(self.thread.has_poll)
  201. def test_set_first_post(self):
  202. """set_first_post sets first post and poster data on thread"""
  203. User = get_user_model()
  204. user = User.objects.create_user("Bob", "bob@boberson.com", "Pass.123")
  205. datetime = timezone.now() + timedelta(5)
  206. post = Post.objects.create(
  207. category=self.category,
  208. thread=self.thread,
  209. poster=user,
  210. poster_name=user.username,
  211. poster_ip='127.0.0.1',
  212. original="Hello! I am test message!",
  213. parsed="<p>Hello! I am test message!</p>",
  214. checksum="nope",
  215. posted_on=datetime,
  216. updated_on=datetime
  217. )
  218. self.thread.set_first_post(post)
  219. self.assertEqual(self.thread.first_post, post)
  220. self.assertEqual(self.thread.started_on, post.posted_on)
  221. self.assertEqual(self.thread.starter, user)
  222. self.assertEqual(self.thread.starter_name, user.username)
  223. self.assertEqual(self.thread.starter_slug, user.slug)
  224. def test_set_last_post(self):
  225. """set_last_post sets first post and poster data on thread"""
  226. User = get_user_model()
  227. user = User.objects.create_user("Bob", "bob@boberson.com", "Pass.123")
  228. datetime = timezone.now() + timedelta(5)
  229. post = Post.objects.create(
  230. category=self.category,
  231. thread=self.thread,
  232. poster=user,
  233. poster_name=user.username,
  234. poster_ip='127.0.0.1',
  235. original="Hello! I am test message!",
  236. parsed="<p>Hello! I am test message!</p>",
  237. checksum="nope",
  238. posted_on=datetime,
  239. updated_on=datetime
  240. )
  241. self.thread.set_last_post(post)
  242. self.assertEqual(self.thread.last_post, post)
  243. self.assertEqual(self.thread.last_post_on, post.posted_on)
  244. self.assertEqual(self.thread.last_poster, user)
  245. self.assertEqual(self.thread.last_poster_name, user.username)
  246. self.assertEqual(self.thread.last_poster_slug, user.slug)
  247. def test_move(self):
  248. """move(new_category) moves thread to other category"""
  249. # pick category instead of category (so we don't have to create one)
  250. root_category = Category.objects.root_category()
  251. Category(
  252. name='New Category',
  253. slug='new-category',
  254. ).insert_at(root_category, position='last-child', save=True)
  255. new_category = Category.objects.get(slug='new-category')
  256. self.thread.move(new_category)
  257. self.assertEqual(self.thread.category, new_category)
  258. for post in self.thread.post_set.all():
  259. self.assertEqual(post.category_id, new_category.id)
  260. def test_merge(self):
  261. """merge(other_thread) moves other thread content to this thread"""
  262. with self.assertRaises(ValueError):
  263. self.thread.merge(self.thread)
  264. datetime = timezone.now() + timedelta(5)
  265. other_thread = Thread(
  266. category=self.category,
  267. started_on=datetime,
  268. starter_name='Tester',
  269. starter_slug='tester',
  270. last_post_on=datetime,
  271. last_poster_name='Tester',
  272. last_poster_slug='tester'
  273. )
  274. other_thread.set_title("Other thread")
  275. other_thread.save()
  276. post = Post.objects.create(
  277. category=self.category,
  278. thread=other_thread,
  279. poster_name='Admin',
  280. poster_ip='127.0.0.1',
  281. original="Hello! I am other message!",
  282. parsed="<p>Hello! I am other message!</p>",
  283. checksum="nope",
  284. posted_on=datetime,
  285. updated_on=datetime
  286. )
  287. other_thread.first_post = post
  288. other_thread.last_post = post
  289. other_thread.save()
  290. self.thread.merge(other_thread)
  291. self.thread.synchronize()
  292. self.assertEqual(self.thread.replies, 1)
  293. self.assertEqual(self.thread.last_post, post)
  294. self.assertEqual(self.thread.last_post_on, post.posted_on)
  295. self.assertEqual(self.thread.last_poster_name, "Admin")
  296. self.assertEqual(self.thread.last_poster_slug, "admin")
  297. def test_delete_private_thread(self):
  298. """
  299. private thread gets deleted automatically
  300. when there are no participants left in it
  301. """
  302. User = get_user_model()
  303. user_a = User.objects.create_user(
  304. "Bob", "bob@boberson.com", "Pass.123")
  305. user_b = User.objects.create_user(
  306. "Weebl", "weebl@weeblson.com", "Pass.123")
  307. ThreadParticipant.objects.add_participants(self.thread, [user_a, user_b])
  308. self.assertEqual(self.thread.participants.count(), 2)
  309. user_a.delete()
  310. Thread.objects.get(id=self.thread.id)
  311. user_b.delete()
  312. with self.assertRaises(Thread.DoesNotExist):
  313. Thread.objects.get(id=self.thread.id)