-test_editpost_view.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. import json
  2. from django.conf import settings
  3. from django.core.urlresolvers import reverse
  4. from misago.acl.testutils import override_acl
  5. from misago.categories.models import Category
  6. from misago.users.testutils import AuthenticatedUserTestCase
  7. from misago.threads.models import Label, Thread, Post
  8. from misago.threads.testutils import post_thread
  9. class EditPostTests(AuthenticatedUserTestCase):
  10. ajax_header = {'HTTP_X_REQUESTED_WITH': 'XMLHttpRequest'}
  11. def setUp(self):
  12. super(EditPostTests, self).setUp()
  13. self.category = Category.objects.all_categories().filter(
  14. role='category')[:1][0]
  15. self.thread = post_thread(self.category, poster=self.user)
  16. self.link = reverse('misago:edit_post', kwargs={
  17. 'category_id': self.category.id,
  18. 'thread_id': self.thread.id,
  19. 'post_id': self.thread.first_post_id,
  20. })
  21. Label.objects.clear_cache()
  22. def tearDown(self):
  23. Label.objects.clear_cache()
  24. def override_category_acl(self, extra_acl=None):
  25. categories_acl = self.user.acl
  26. categories_acl['visible_categories'].append(self.category.pk)
  27. categories_acl['categories'][self.category.pk] = {
  28. 'can_see': 1,
  29. 'can_browse': 1,
  30. 'can_see_all_threads': 1,
  31. }
  32. if extra_acl:
  33. categories_acl['categories'][self.category.pk].update(extra_acl)
  34. override_acl(self.user, categories_acl)
  35. def test_cant_see(self):
  36. """has no permission to see category"""
  37. categories_acl = self.user.acl
  38. categories_acl['visible_categories'].remove(self.category.pk)
  39. categories_acl['categories'][self.category.pk] = {
  40. 'can_see': 0,
  41. 'can_browse': 0,
  42. 'can_see_all_threads': 1,
  43. 'can_reply_threads': 1,
  44. }
  45. override_acl(self.user, categories_acl)
  46. response = self.client.get(self.link, **self.ajax_header)
  47. self.assertEqual(response.status_code, 404)
  48. def test_cant_browse(self):
  49. """has no permission to browse category"""
  50. categories_acl = self.user.acl
  51. categories_acl['visible_categories'].append(self.category.pk)
  52. categories_acl['categories'][self.category.pk] = {
  53. 'can_see': 1,
  54. 'can_browse': 0,
  55. 'can_see_all_threads': 1,
  56. 'can_reply_threads': 1,
  57. }
  58. override_acl(self.user, categories_acl)
  59. response = self.client.get(self.link, **self.ajax_header)
  60. self.assertEqual(response.status_code, 404)
  61. def test_cant_edit_own_post_in_locked_category(self):
  62. """can't edit own post in closed category"""
  63. self.category.is_closed = True
  64. self.category.save()
  65. self.override_category_acl({'can_edit_threads': 1})
  66. response = self.client.get(self.link, **self.ajax_header)
  67. self.assertEqual(response.status_code, 403)
  68. def test_cant_edit_other_user_post_in_locked_category(self):
  69. """can't edit other user post in closed category"""
  70. self.category.is_closed = True
  71. self.category.save()
  72. self.thread.first_post.poster = None
  73. self.thread.first_post.save()
  74. self.thread.synchronize()
  75. self.thread.save()
  76. self.override_category_acl({'can_edit_threads': 2})
  77. response = self.client.get(self.link, **self.ajax_header)
  78. self.assertEqual(response.status_code, 403)
  79. def test_cant_edit_own_post_in_locked_thread(self):
  80. """can't edit own post in closed thread"""
  81. self.thread.is_closed = True
  82. self.thread.save()
  83. self.override_category_acl({'can_edit_threads': 1})
  84. response = self.client.get(self.link, **self.ajax_header)
  85. self.assertEqual(response.status_code, 403)
  86. def test_cant_edit_other_user_post_in_locked_thread(self):
  87. """can't edit other user post in closed thread"""
  88. self.override_category_acl({'can_edit_threads': 2})
  89. self.thread.first_post.poster = None
  90. self.thread.first_post.save()
  91. self.thread.is_closed = True
  92. self.thread.synchronize()
  93. self.thread.save()
  94. response = self.client.get(self.link, **self.ajax_header)
  95. self.assertEqual(response.status_code, 403)
  96. def test_cant_edit_own_post(self):
  97. """can't edit own post"""
  98. self.override_category_acl({'can_edit_posts': 0})
  99. response = self.client.get(self.link, **self.ajax_header)
  100. self.assertEqual(response.status_code, 403)
  101. def test_cant_edit_other_user_post(self):
  102. """can't edit other user post"""
  103. self.override_category_acl({'can_edit_posts': 1})
  104. self.thread.first_post.poster = None
  105. self.thread.first_post.save()
  106. self.thread.synchronize()
  107. self.thread.save()
  108. response = self.client.get(self.link, **self.ajax_header)
  109. self.assertEqual(response.status_code, 403)
  110. def test_cant_edit_protected_post(self):
  111. """can't edit post that was protected by moderator"""
  112. self.override_category_acl({'can_edit_posts': 1, 'can_protect_posts': 0})
  113. self.thread.first_post.is_protected = True
  114. self.thread.first_post.save()
  115. response = self.client.get(self.link, **self.ajax_header)
  116. self.assertEqual(response.status_code, 403)
  117. def test_can_edit_own_post(self):
  118. """can edit own post"""
  119. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 0})
  120. response = self.client.get(self.link, **self.ajax_header)
  121. self.assertEqual(response.status_code, 200)
  122. self.assertNotIn('thread-title', response.content)
  123. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 0})
  124. response = self.client.post(self.link, data={
  125. 'post': 'Edited reply!',
  126. 'submit': True,
  127. },
  128. **self.ajax_header)
  129. self.assertEqual(response.status_code, 200)
  130. response_dict = json.loads(response.content)
  131. self.assertIn('post_url', response_dict)
  132. post = Post.objects.get(id=self.thread.first_post_id)
  133. self.assertEqual(post.original, 'Edited reply!')
  134. self.assertEqual(post.edits, 1)
  135. def test_empty_edit_form(self):
  136. """empty edit form has no crashes"""
  137. self.override_category_acl({'can_edit_posts': 2, 'can_edit_threads': 2})
  138. response = self.client.post(self.link, data={
  139. 'submit': True,
  140. },
  141. **self.ajax_header)
  142. self.assertEqual(response.status_code, 200)
  143. def test_can_edit_other_user_post(self):
  144. """can edit other user post"""
  145. self.override_category_acl({'can_edit_posts': 2, 'can_edit_threads': 0})
  146. self.thread.first_post.poster = None
  147. self.thread.first_post.save()
  148. self.thread.synchronize()
  149. self.thread.save()
  150. response = self.client.get(self.link, **self.ajax_header)
  151. self.assertEqual(response.status_code, 200)
  152. self.assertNotIn('thread-title', response.content)
  153. self.override_category_acl({'can_edit_posts': 2, 'can_edit_threads': 0})
  154. response = self.client.post(self.link, data={
  155. 'post': 'Edited reply!',
  156. 'submit': True,
  157. },
  158. **self.ajax_header)
  159. self.assertEqual(response.status_code, 200)
  160. response_dict = json.loads(response.content)
  161. self.assertIn('post_url', response_dict)
  162. post = Post.objects.get(id=self.thread.first_post_id)
  163. self.assertEqual(post.original, 'Edited reply!')
  164. self.assertEqual(post.edits, 1)
  165. def test_can_edit_own_thread(self):
  166. """can edit own thread"""
  167. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 1})
  168. response = self.client.get(self.link, **self.ajax_header)
  169. self.assertEqual(response.status_code, 200)
  170. self.assertIn('thread-title', response.content)
  171. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 1})
  172. response = self.client.post(self.link, data={
  173. 'title': 'Edited title!',
  174. 'post': self.thread.first_post.original,
  175. 'submit': True,
  176. },
  177. **self.ajax_header)
  178. self.assertEqual(response.status_code, 200)
  179. response_dict = json.loads(response.content)
  180. self.assertIn('post_url', response_dict)
  181. thread = Thread.objects.get(id=self.thread.id)
  182. self.assertEqual(thread.title, 'Edited title!')
  183. post = Post.objects.get(id=self.thread.first_post_id)
  184. self.assertEqual(post.original, self.thread.first_post.original)
  185. self.assertEqual(post.edits, 1)
  186. def test_can_edit_other_user_thread(self):
  187. """can edit other user thread"""
  188. self.override_category_acl({'can_edit_posts': 2, 'can_edit_threads': 2})
  189. self.thread.first_post.poster = None
  190. self.thread.first_post.save()
  191. self.thread.synchronize()
  192. self.thread.save()
  193. response = self.client.get(self.link, **self.ajax_header)
  194. self.assertEqual(response.status_code, 200)
  195. self.assertIn('thread-title', response.content)
  196. self.override_category_acl({'can_edit_posts': 2, 'can_edit_threads': 2})
  197. response = self.client.post(self.link, data={
  198. 'title': 'Edited title!',
  199. 'post': self.thread.first_post.original,
  200. 'submit': True,
  201. },
  202. **self.ajax_header)
  203. self.assertEqual(response.status_code, 200)
  204. response_dict = json.loads(response.content)
  205. self.assertIn('post_url', response_dict)
  206. thread = Thread.objects.get(id=self.thread.id)
  207. self.assertEqual(thread.title, 'Edited title!')
  208. post = Post.objects.get(id=self.thread.first_post_id)
  209. self.assertEqual(post.original, self.thread.first_post.original)
  210. self.assertEqual(post.edits, 1)
  211. def test_no_change_edit(self):
  212. """user edited post but submited no changes"""
  213. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 1})
  214. response = self.client.get(self.link, **self.ajax_header)
  215. self.assertEqual(response.status_code, 200)
  216. self.assertIn('thread-title', response.content)
  217. self.override_category_acl({'can_edit_posts': 1, 'can_edit_threads': 1})
  218. response = self.client.post(self.link, data={
  219. 'title': self.thread.title,
  220. 'post': self.thread.first_post.original,
  221. 'submit': True,
  222. },
  223. **self.ajax_header)
  224. self.assertEqual(response.status_code, 200)
  225. response_dict = json.loads(response.content)
  226. self.assertIn('post_url', response_dict)
  227. post = Post.objects.get(id=self.thread.first_post_id)
  228. self.assertEqual(post.original, self.thread.first_post.original)
  229. self.assertEqual(post.edits, 0)
  230. def test_close_and_open_edit(self):
  231. """user edited post to close and open thread"""
  232. prefix = 'misago.threads.posting.threadclose.ThreadCloseFormMiddleware'
  233. field_name = '%s-is_closed' % prefix
  234. self.override_category_acl({'can_edit_posts': 1, 'can_close_threads': 1})
  235. response = self.client.get(self.link, **self.ajax_header)
  236. self.assertEqual(response.status_code, 200)
  237. self.assertIn(field_name, response.content)
  238. self.override_category_acl({'can_edit_posts': 1, 'can_close_threads': 1})
  239. response = self.client.post(self.link, data={
  240. 'post': self.thread.first_post.original,
  241. field_name: 1,
  242. 'submit': True,
  243. },
  244. **self.ajax_header)
  245. self.assertEqual(response.status_code, 200)
  246. response_dict = json.loads(response.content)
  247. self.assertIn('post_url', response_dict)
  248. thread = Thread.objects.get(id=self.thread.id)
  249. self.assertTrue(thread.is_closed)
  250. self.user.last_posted_on = None
  251. self.user.save()
  252. self.override_category_acl({'can_edit_posts': 1, 'can_close_threads': 1})
  253. response = self.client.post(self.link, data={
  254. 'post': self.thread.first_post.original,
  255. field_name: 0,
  256. 'submit': True,
  257. },
  258. **self.ajax_header)
  259. self.assertEqual(response.status_code, 200)
  260. response_dict = json.loads(response.content)
  261. self.assertIn('post_url', response_dict)
  262. thread = Thread.objects.get(id=self.thread.id)
  263. self.assertFalse(thread.is_closed)
  264. def test_pin_and_unpin_edit(self):
  265. """user edited post to pin and unpin thread"""
  266. prefix = 'misago.threads.posting.threadpin.ThreadPinFormMiddleware'
  267. field_name = '%s-is_pinned' % prefix
  268. self.override_category_acl({'can_edit_posts': 1, 'can_pin_threads': 1})
  269. response = self.client.get(self.link, **self.ajax_header)
  270. self.assertEqual(response.status_code, 200)
  271. self.assertIn(field_name, response.content)
  272. self.override_category_acl({'can_edit_posts': 1, 'can_pin_threads': 1})
  273. response = self.client.post(self.link, data={
  274. 'post': self.thread.first_post.original,
  275. field_name: 1,
  276. 'submit': True,
  277. },
  278. **self.ajax_header)
  279. self.assertEqual(response.status_code, 200)
  280. response_dict = json.loads(response.content)
  281. self.assertIn('post_url', response_dict)
  282. thread = Thread.objects.get(id=self.thread.id)
  283. self.assertTrue(thread.is_pinned)
  284. self.user.last_posted_on = None
  285. self.user.save()
  286. self.override_category_acl({'can_edit_posts': 1, 'can_pin_threads': 1})
  287. response = self.client.post(self.link, data={
  288. 'post': self.thread.first_post.original,
  289. field_name: 0,
  290. 'submit': True,
  291. },
  292. **self.ajax_header)
  293. self.assertEqual(response.status_code, 200)
  294. response_dict = json.loads(response.content)
  295. self.assertIn('post_url', response_dict)
  296. thread = Thread.objects.get(id=self.thread.id)
  297. self.assertFalse(thread.is_pinned)
  298. def test_label_and_unlabel_edit(self):
  299. """user edited thread to label and unlabel it"""
  300. prefix = 'misago.threads.posting.threadlabel.ThreadLabelFormMiddleware'
  301. field_name = '%s-label' % prefix
  302. label = Label.objects.create(name="Label", slug="label")
  303. label.categories.add(self.category)
  304. acls = {
  305. 'can_edit_posts': 1,
  306. 'can_edit_threads': 1,
  307. 'can_change_threads_labels': 1
  308. }
  309. self.override_category_acl(acls)
  310. response = self.client.get(self.link, **self.ajax_header)
  311. self.assertEqual(response.status_code, 200)
  312. self.assertIn(field_name, response.content)
  313. self.override_category_acl(acls)
  314. response = self.client.post(self.link, data={
  315. field_name: label.pk,
  316. 'title': self.thread.title,
  317. 'post': self.thread.first_post.original,
  318. 'submit': True,
  319. },
  320. **self.ajax_header)
  321. self.assertEqual(response.status_code, 200)
  322. thread = Thread.objects.get(id=self.thread.id)
  323. self.assertEqual(thread.label_id, label.id)
  324. self.user.last_posted_on = None
  325. self.user.save()
  326. self.override_category_acl(acls)
  327. response = self.client.post(self.link, data={
  328. field_name: 0,
  329. 'title': self.thread.title,
  330. 'post': self.thread.first_post.original,
  331. 'submit': True,
  332. },
  333. **self.ajax_header)
  334. self.assertEqual(response.status_code, 200)
  335. thread = Thread.objects.get(id=self.thread.id)
  336. self.assertIsNone(thread.label_id)
  337. def test_empty_form(self):
  338. """empty form has no errors"""
  339. acls = {
  340. 'can_edit_posts': 1,
  341. 'can_edit_threads': 1,
  342. 'can_change_threads_labels': 1
  343. }
  344. self.override_category_acl(acls)
  345. response = self.client.post(self.link, data={
  346. 'title': '',
  347. 'post': '',
  348. 'preview': True},
  349. **self.ajax_header)
  350. self.assertEqual(response.status_code, 200)
  351. self.override_category_acl(acls)
  352. response = self.client.post(self.link, data={
  353. 'title': '',
  354. 'post': '',
  355. 'submit': True},
  356. **self.ajax_header)
  357. self.assertEqual(response.status_code, 200)