test_threadview.py 23 KB

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