test_threadview.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561
  1. from unittest.mock import Mock
  2. from misago.acl import useracl
  3. from misago.acl.test import patch_user_acl
  4. from misago.categories.models import Category
  5. from misago.conf import settings
  6. from misago.threads import testutils
  7. from misago.threads.checksums import update_post_checksum
  8. from misago.threads.events import record_event
  9. from misago.threads.moderation import threads as threads_moderation
  10. from misago.threads.moderation import hide_post
  11. from misago.users.testutils import AuthenticatedUserTestCase
  12. cache_versions = {"acl": "abcdefgh"}
  13. def patch_category_acl(new_acl=None):
  14. def patch_acl(_, user_acl):
  15. category = Category.objects.get(slug='first-category')
  16. category_acl = user_acl['categories'][category.id]
  17. # reset category ACL to single predictable state
  18. category_acl.update({
  19. 'can_see': 1,
  20. 'can_browse': 1,
  21. 'can_see_all_threads': 1,
  22. 'can_see_own_threads': 0,
  23. 'can_hide_threads': 0,
  24. 'can_approve_content': 0,
  25. 'can_edit_posts': 0,
  26. 'can_hide_posts': 0,
  27. 'can_hide_own_posts': 0,
  28. 'can_close_threads': 0,
  29. 'post_edit_time': 0,
  30. 'can_hide_events': 0,
  31. })
  32. if new_acl:
  33. category_acl.update(new_acl)
  34. return patch_user_acl(patch_acl)
  35. class ThreadViewTestCase(AuthenticatedUserTestCase):
  36. def setUp(self):
  37. super().setUp()
  38. self.category = Category.objects.get(slug='first-category')
  39. self.thread = testutils.post_thread(category=self.category)
  40. class ThreadVisibilityTests(ThreadViewTestCase):
  41. def test_thread_displays(self):
  42. """thread view has no showstoppers"""
  43. response = self.client.get(self.thread.get_absolute_url())
  44. self.assertContains(response, self.thread.title)
  45. def test_view_shows_owner_thread(self):
  46. """view handles "owned threads" only"""
  47. with patch_category_acl({'can_see_all_threads': 0}):
  48. response = self.client.get(self.thread.get_absolute_url())
  49. self.assertEqual(response.status_code, 404)
  50. self.thread.starter = self.user
  51. self.thread.save()
  52. response = self.client.get(self.thread.get_absolute_url())
  53. self.assertContains(response, self.thread.title)
  54. def test_view_validates_category_permissions(self):
  55. """view validates category visiblity"""
  56. with patch_category_acl({'can_see': 0}):
  57. response = self.client.get(self.thread.get_absolute_url())
  58. self.assertEqual(response.status_code, 404)
  59. with patch_category_acl({'can_browse': 0}):
  60. response = self.client.get(self.thread.get_absolute_url())
  61. self.assertEqual(response.status_code, 404)
  62. def test_view_shows_unapproved_thread(self):
  63. """view handles unapproved thread"""
  64. with patch_category_acl({'can_approve_content': 0}):
  65. self.thread.is_unapproved = True
  66. self.thread.save()
  67. response = self.client.get(self.thread.get_absolute_url())
  68. self.assertEqual(response.status_code, 404)
  69. # grant permission to see unapproved content
  70. with patch_category_acl({'can_approve_content': 1}):
  71. response = self.client.get(self.thread.get_absolute_url())
  72. self.assertContains(response, self.thread.title)
  73. # make test user thread's owner and remove permission to see unapproved
  74. # user should be able to see thread as its author anyway
  75. self.thread.starter = self.user
  76. self.thread.save()
  77. with patch_category_acl({'can_approve_content': 0}):
  78. response = self.client.get(self.thread.get_absolute_url())
  79. self.assertContains(response, self.thread.title)
  80. def test_view_shows_hidden_thread(self):
  81. """view handles hidden thread"""
  82. with patch_category_acl({'can_hide_threads': 0}):
  83. self.thread.is_hidden = True
  84. self.thread.save()
  85. response = self.client.get(self.thread.get_absolute_url())
  86. self.assertEqual(response.status_code, 404)
  87. # threads owners are not extempt from hidden threads check
  88. self.thread.starter = self.user
  89. self.thread.save()
  90. response = self.client.get(self.thread.get_absolute_url())
  91. self.assertEqual(response.status_code, 404)
  92. # grant permission to see hidden content
  93. with patch_category_acl({'can_hide_threads': 1}):
  94. response = self.client.get(self.thread.get_absolute_url())
  95. self.assertContains(response, self.thread.title)
  96. class ThreadPostsVisibilityTests(ThreadViewTestCase):
  97. def test_post_renders(self):
  98. """post renders"""
  99. post = testutils.reply_thread(self.thread, poster=self.user)
  100. response = self.client.get(self.thread.get_absolute_url())
  101. self.assertContains(response, post.get_absolute_url())
  102. def test_invalid_post_renders(self):
  103. """invalid post renders"""
  104. post = testutils.reply_thread(self.thread, poster=self.user)
  105. post.parsed = 'fiddled post content'
  106. post.save()
  107. response = self.client.get(self.thread.get_absolute_url())
  108. self.assertContains(response, post.get_absolute_url())
  109. self.assertContains(response, "This post's contents cannot be displayed.")
  110. self.assertNotContains(response, post.parsed)
  111. def test_hidden_post_visibility(self):
  112. """hidden post renders correctly"""
  113. post = testutils.reply_thread(self.thread, message="Hello, I'm hidden post!")
  114. hide_post(self.user, post)
  115. response = self.client.get(self.thread.get_absolute_url())
  116. self.assertContains(response, post.get_absolute_url())
  117. self.assertContains(response, "This post is hidden. You cannot not see its contents.")
  118. self.assertNotContains(response, post.parsed)
  119. # posts authors are not extempt from seeing hidden posts content
  120. post.posted_by = self.user
  121. post.save()
  122. response = self.client.get(self.thread.get_absolute_url())
  123. self.assertContains(response, post.get_absolute_url())
  124. self.assertContains(response, "This post is hidden. You cannot not see its contents.")
  125. self.assertNotContains(response, post.parsed)
  126. # permission to hide own posts isn't enought to see post content
  127. with patch_category_acl({'can_hide_own_posts': 1}):
  128. response = self.client.get(self.thread.get_absolute_url())
  129. self.assertContains(response, post.get_absolute_url())
  130. self.assertContains(response, "This post is hidden. You cannot not see its contents.")
  131. self.assertNotContains(response, post.parsed)
  132. # post's content is displayed after permission to see posts is granted
  133. with patch_category_acl({'can_hide_posts': 1}):
  134. response = self.client.get(self.thread.get_absolute_url())
  135. self.assertContains(response, post.get_absolute_url())
  136. self.assertContains(
  137. response, "This post is hidden. Only users with permission may see its contents."
  138. )
  139. self.assertNotContains(response, "This post is hidden. You cannot not see its contents.")
  140. self.assertContains(response, post.parsed)
  141. def test_unapproved_post_visibility(self):
  142. """unapproved post renders for its author and users with perm to approve content"""
  143. post = testutils.reply_thread(self.thread, is_unapproved=True)
  144. # post is hdden because we aren't its author nor user with permission to approve
  145. response = self.client.get(self.thread.get_absolute_url())
  146. self.assertNotContains(response, post.get_absolute_url())
  147. # post displays because we have permission to approve unapproved content
  148. with patch_category_acl({'can_approve_content': 1}):
  149. response = self.client.get(self.thread.get_absolute_url())
  150. self.assertContains(response, post.get_absolute_url())
  151. self.assertContains(response, "This post is unapproved.")
  152. self.assertContains(response, post.parsed)
  153. # post displays because we are its author
  154. with patch_category_acl({'can_approve_content': 0}):
  155. post.poster = self.user
  156. post.save()
  157. response = self.client.get(self.thread.get_absolute_url())
  158. self.assertContains(response, post.get_absolute_url())
  159. self.assertContains(response, "This post is unapproved.")
  160. self.assertContains(response, post.parsed)
  161. class ThreadEventVisibilityTests(ThreadViewTestCase):
  162. def test_thread_events_render(self):
  163. """different thread events render"""
  164. TEST_ACTIONS = [
  165. (threads_moderation.pin_thread_globally, "Thread has been pinned globally."),
  166. (threads_moderation.pin_thread_locally, "Thread has been pinned locally."),
  167. (threads_moderation.unpin_thread, "Thread has been unpinned."),
  168. (threads_moderation.approve_thread, "Thread has been approved."),
  169. (threads_moderation.close_thread, "Thread has been closed."),
  170. (threads_moderation.open_thread, "Thread has been opened."),
  171. (threads_moderation.hide_thread, "Thread has been made hidden."),
  172. (threads_moderation.unhide_thread, "Thread has been revealed."),
  173. ]
  174. self.thread.is_unapproved = True
  175. self.thread.save()
  176. for action, message in TEST_ACTIONS:
  177. self.thread.post_set.filter(is_event=True).delete()
  178. with patch_category_acl({'can_approve_content': 1, 'can_hide_threads': 1}):
  179. user_acl = useracl.get_user_acl(self.user, cache_versions)
  180. request = Mock(user=self.user, user_acl=user_acl, user_ip="127.0.0.1")
  181. action(request, self.thread)
  182. event = self.thread.post_set.filter(is_event=True)[0]
  183. # event renders
  184. response = self.client.get(self.thread.get_absolute_url())
  185. self.assertContains(response, event.get_absolute_url())
  186. self.assertContains(response, message)
  187. # hidden events don't render without permission
  188. with patch_category_acl({'can_approve_content': 1, 'can_hide_threads': 1}):
  189. hide_post(self.user, event)
  190. response = self.client.get(self.thread.get_absolute_url())
  191. self.assertNotContains(response, event.get_absolute_url())
  192. self.assertNotContains(response, message)
  193. # hidden event renders with permission
  194. with patch_category_acl({
  195. 'can_approve_content': 1,
  196. 'can_hide_threads': 1,
  197. 'can_hide_events': 1,
  198. }):
  199. hide_post(self.user, event)
  200. response = self.client.get(self.thread.get_absolute_url())
  201. self.assertContains(response, event.get_absolute_url())
  202. self.assertContains(response, message)
  203. self.assertContains(response, "Hidden by")
  204. # Event is only loaded if thread has events flag
  205. with patch_category_acl({
  206. 'can_approve_content': 1,
  207. 'can_hide_threads': 1,
  208. 'can_hide_events': 1,
  209. }):
  210. self.thread.has_events = False
  211. self.thread.save()
  212. response = self.client.get(self.thread.get_absolute_url())
  213. self.assertNotContains(response, event.get_absolute_url())
  214. def test_events_limit(self):
  215. """forum will trim oldest events if theres more than allowed by config"""
  216. events_limit = settings.MISAGO_EVENTS_PER_PAGE
  217. events = []
  218. for _ in range(events_limit + 5):
  219. request = Mock(user=self.user, user_ip="127.0.0.1")
  220. event = record_event(request, self.thread, 'closed')
  221. events.append(event)
  222. # test that only events within limits were rendered
  223. response = self.client.get(self.thread.get_absolute_url())
  224. for event in events[5:]:
  225. self.assertContains(response, event.get_absolute_url())
  226. for event in events[:5]:
  227. self.assertNotContains(response, event.get_absolute_url())
  228. def test_events_dont_take_space(self):
  229. """events dont take space away from posts"""
  230. posts_limit = settings.MISAGO_POSTS_PER_PAGE
  231. events_limit = settings.MISAGO_EVENTS_PER_PAGE
  232. events = []
  233. for _ in range(events_limit + 5):
  234. request = Mock(user=self.user, user_ip="127.0.0.1")
  235. event = record_event(request, self.thread, 'closed')
  236. events.append(event)
  237. posts = []
  238. for _ in range(posts_limit - 1):
  239. post = testutils.reply_thread(self.thread)
  240. posts.append(post)
  241. # test that all events and posts within limits were rendered
  242. response = self.client.get(self.thread.get_absolute_url())
  243. for event in events[5:]:
  244. self.assertContains(response, event.get_absolute_url())
  245. for post in posts:
  246. self.assertContains(response, post.get_absolute_url())
  247. # add second page to thread with more events
  248. for _ in range(posts_limit):
  249. post = testutils.reply_thread(self.thread)
  250. for _ in range(events_limit):
  251. request = Mock(user=self.user, user_ip="127.0.0.1")
  252. event = record_event(request, self.thread, 'closed')
  253. events.append(event)
  254. # see first page
  255. response = self.client.get(self.thread.get_absolute_url())
  256. for event in events[5:events_limit]:
  257. self.assertContains(response, event.get_absolute_url())
  258. for post in posts[:posts_limit - 1]:
  259. self.assertContains(response, post.get_absolute_url())
  260. # see second page
  261. response = self.client.get('%s2/' % self.thread.get_absolute_url())
  262. for event in events[5 + events_limit:]:
  263. self.assertContains(response, event.get_absolute_url())
  264. for post in posts[posts_limit - 1:]:
  265. self.assertContains(response, post.get_absolute_url())
  266. def test_changed_thread_title_event_renders(self):
  267. """changed thread title event renders"""
  268. request = Mock(user=self.user, user_ip="127.0.0.1")
  269. threads_moderation.change_thread_title(
  270. request, self.thread, "Lorem renamed ipsum!"
  271. )
  272. event = self.thread.post_set.filter(is_event=True)[0]
  273. self.assertEqual(event.event_type, 'changed_title')
  274. # event renders
  275. response = self.client.get(self.thread.get_absolute_url())
  276. self.assertContains(response, event.get_absolute_url())
  277. self.assertContains(response, "title has been changed from")
  278. self.assertContains(response, self.thread.title)
  279. def test_thread_move_event_renders(self):
  280. """moved thread event renders"""
  281. self.thread.category = self.thread.category.parent
  282. self.thread.save()
  283. request = Mock(user=self.user, user_ip="127.0.0.1")
  284. threads_moderation.move_thread(request, self.thread, self.category)
  285. event = self.thread.post_set.filter(is_event=True)[0]
  286. self.assertEqual(event.event_type, 'moved')
  287. # event renders
  288. response = self.client.get(self.thread.get_absolute_url())
  289. self.assertContains(response, event.get_absolute_url())
  290. self.assertContains(response, "Thread has been moved from")
  291. def test_thread_merged_event_renders(self):
  292. """merged thread event renders"""
  293. request = Mock(user=self.user, user_ip="127.0.0.1")
  294. other_thread = testutils.post_thread(category=self.category)
  295. threads_moderation.merge_thread(request, self.thread, other_thread)
  296. event = self.thread.post_set.filter(is_event=True)[0]
  297. self.assertEqual(event.event_type, 'merged')
  298. # event renders
  299. response = self.client.get(self.thread.get_absolute_url())
  300. self.assertContains(response, event.get_absolute_url())
  301. self.assertContains(response, "thread has been merged into this thread")
  302. class ThreadAttachmentsViewTests(ThreadViewTestCase):
  303. def mock_attachment_cache(self, data):
  304. json = {
  305. 'url': {},
  306. 'size': 16914,
  307. 'filename': 'Archiwum.zip',
  308. 'filetype': 'ZIP',
  309. 'is_image': False,
  310. 'uploaded_on': '2016-10-22T21:17:40.408710Z',
  311. 'uploader_name': 'BobBoberson',
  312. }
  313. json.update(data)
  314. return json
  315. def test_attachments_display(self):
  316. """thread posts show list of attachments below them"""
  317. post = self.thread.first_post
  318. post.attachments_cache = [
  319. self.mock_attachment_cache({
  320. 'url': {
  321. 'index': '/attachment/loremipsum-123/',
  322. 'thumb': None,
  323. 'uploader': '/user/bobboberson-123/',
  324. },
  325. 'filename': 'Archiwum-1.zip',
  326. }),
  327. self.mock_attachment_cache({
  328. 'url': {
  329. 'index': '/attachment/loremipsum-223/',
  330. 'thumb': '/attachment/thumb/loremipsum-223/',
  331. 'uploader': '/user/bobboberson-223/',
  332. },
  333. 'is_image': True,
  334. 'filename': 'Archiwum-2.zip',
  335. }),
  336. self.mock_attachment_cache({
  337. 'url': {
  338. 'index': '/attachment/loremipsum-323/',
  339. 'thumb': None,
  340. 'uploader': '/user/bobboberson-323/',
  341. },
  342. 'filename': 'Archiwum-3.zip',
  343. }),
  344. ]
  345. post.save()
  346. # attachments render
  347. response = self.client.get(self.thread.get_absolute_url())
  348. for attachment in post.attachments_cache:
  349. self.assertContains(response, attachment['filename'])
  350. self.assertContains(response, attachment['uploader_name'])
  351. self.assertContains(response, attachment['url']['index'])
  352. self.assertContains(response, attachment['url']['uploader'])
  353. if attachment['url']['thumb']:
  354. self.assertContains(response, attachment['url']['thumb'])
  355. class ThreadPollViewTests(ThreadViewTestCase):
  356. def test_poll_voted_display(self):
  357. """view has no showstoppers when displaying voted poll"""
  358. poll = testutils.post_poll(self.thread, self.user)
  359. response = self.client.get(self.thread.get_absolute_url())
  360. self.assertContains(response, poll.question)
  361. self.assertContains(response, '4 votes')
  362. self.assertNotContains(response, 'Save your vote')
  363. def test_poll_unvoted_display(self):
  364. """view has no showstoppers when displaying poll vote form"""
  365. poll = testutils.post_poll(self.thread, self.user)
  366. poll.pollvote_set.all().delete()
  367. response = self.client.get(self.thread.get_absolute_url())
  368. self.assertContains(response, poll.question)
  369. self.assertContains(response, 'Save your vote')
  370. def test_poll_anonymous_view(self):
  371. """view has no showstoppers when displaying poll to anon user"""
  372. poll = testutils.post_poll(self.thread, self.user)
  373. self.logout_user()
  374. response = self.client.get(self.thread.get_absolute_url())
  375. self.assertContains(response, poll.question)
  376. self.assertContains(response, '4 votes')
  377. self.assertNotContains(response, 'Save your vote')
  378. class ThreadLikedPostsViewTests(ThreadViewTestCase):
  379. def test_liked_posts_display(self):
  380. """view has no showstoppers on displaying posts with likes"""
  381. testutils.like_post(self.thread.first_post, self.user)
  382. response = self.client.get(self.thread.get_absolute_url())
  383. self.assertContains(response, '"is_liked": true')
  384. def test_liked_posts_no_permission(self):
  385. """
  386. view has no showstoppers on displaying posts with likes without perm
  387. """
  388. testutils.like_post(self.thread.first_post, self.user)
  389. with patch_category_acl({'can_see_posts_likes': 0}):
  390. response = self.client.get(self.thread.get_absolute_url())
  391. self.assertNotContains(response, '"is_liked": true')
  392. self.assertNotContains(response, '"is_liked": false')
  393. self.assertContains(response, '"is_liked": null')
  394. class ThreadAnonViewTests(ThreadViewTestCase):
  395. def test_anonymous_user_view_no_showstoppers_display(self):
  396. """kitchensink thread view has no showstoppers for anons"""
  397. request = Mock(user=self.user, user_ip="127.0.0.1")
  398. poll = testutils.post_poll(self.thread, self.user)
  399. event = record_event(request, self.thread, 'closed')
  400. hidden_event = record_event(request, self.thread, 'opened')
  401. hide_post(self.user, hidden_event)
  402. unapproved_post = testutils.reply_thread(self.thread, is_unapproved=True)
  403. post = testutils.reply_thread(self.thread)
  404. self.logout_user()
  405. response = self.client.get(self.thread.get_absolute_url())
  406. self.assertContains(response, poll.question)
  407. self.assertContains(response, event.get_absolute_url())
  408. self.assertContains(response, post.get_absolute_url())
  409. self.assertNotContains(response, hidden_event.get_absolute_url())
  410. self.assertNotContains(response, unapproved_post.get_absolute_url())
  411. class ThreadUnicodeSupportTests(ThreadViewTestCase):
  412. def test_category_name(self):
  413. """unicode in category name causes no showstopper"""
  414. self.category.name = 'Łódź'
  415. self.category.save()
  416. with patch_category_acl():
  417. response = self.client.get(self.thread.get_absolute_url())
  418. self.assertEqual(response.status_code, 200)
  419. def test_thread_title(self):
  420. """unicode in thread title causes no showstopper"""
  421. self.thread.title = 'Łódź'
  422. self.thread.slug = 'Lodz'
  423. self.thread.save()
  424. with patch_category_acl():
  425. response = self.client.get(self.thread.get_absolute_url())
  426. self.assertEqual(response.status_code, 200)
  427. def test_post_content(self):
  428. """unicode in thread title causes no showstopper"""
  429. self.thread.first_post.original = 'Łódź'
  430. self.thread.first_post.parsed = '<p>Łódź</p>'
  431. update_post_checksum(self.thread.first_post)
  432. self.thread.first_post.save()
  433. with patch_category_acl():
  434. response = self.client.get(self.thread.get_absolute_url())
  435. self.assertEqual(response.status_code, 200)
  436. def test_user_rank(self):
  437. """unicode in user rank causes no showstopper"""
  438. self.user.title = 'Łódź'
  439. self.user.rank.name = 'Łódź'
  440. self.user.rank.title = 'Łódź'
  441. self.user.rank.save()
  442. self.user.save()
  443. with patch_category_acl():
  444. response = self.client.get(self.thread.get_absolute_url())
  445. self.assertEqual(response.status_code, 200)