test_threadview.py 23 KB

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