test_thread_start_api.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. from django.urls 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. class StartThreadTests(AuthenticatedUserTestCase):
  8. def setUp(self):
  9. super(StartThreadTests, self).setUp()
  10. self.category = Category.objects.get(slug='first-category')
  11. self.api_link = reverse('misago:api:thread-list')
  12. def override_acl(self, extra_acl=None):
  13. new_acl = self.user.acl_cache
  14. new_acl['categories'][self.category.pk].update({
  15. 'can_see': 1,
  16. 'can_browse': 1,
  17. 'can_start_threads': 1,
  18. 'can_pin_threads': 0,
  19. 'can_close_threads': 0,
  20. 'can_hide_threads': 0,
  21. 'can_hide_own_threads': 0,
  22. })
  23. if extra_acl:
  24. new_acl['categories'][self.category.pk].update(extra_acl)
  25. if 'can_see' in extra_acl and not extra_acl['can_see']:
  26. new_acl['visible_categories'].remove(self.category.pk)
  27. new_acl['browseable_categories'].remove(self.category.pk)
  28. if 'can_browse' in extra_acl and not extra_acl['can_browse']:
  29. new_acl['browseable_categories'].remove(self.category.pk)
  30. override_acl(self.user, new_acl)
  31. def test_cant_start_thread_as_guest(self):
  32. """user has to be authenticated to be able to post thread"""
  33. self.logout_user()
  34. response = self.client.post(self.api_link)
  35. self.assertEqual(response.status_code, 403)
  36. def test_cant_see(self):
  37. """has no permission to see selected category"""
  38. self.override_acl({'can_see': 0})
  39. response = self.client.post(self.api_link, {
  40. 'category': self.category.pk,
  41. })
  42. self.assertContains(response, "Selected category is invalid.", status_code=400)
  43. def test_cant_browse(self):
  44. """has no permission to browse selected category"""
  45. self.override_acl({'can_browse': 0})
  46. response = self.client.post(self.api_link, {
  47. 'category': self.category.pk,
  48. })
  49. self.assertContains(response, "Selected category is invalid.", status_code=400)
  50. def test_cant_start_thread(self):
  51. """permission to start thread in category is validated"""
  52. self.override_acl({'can_start_threads': 0})
  53. response = self.client.post(self.api_link, {
  54. 'category': self.category.pk,
  55. })
  56. self.assertContains(
  57. response, "You don't have permission to start new threads", status_code=400
  58. )
  59. def test_cant_start_thread_in_locked_category(self):
  60. """can't post in closed category"""
  61. self.category.is_closed = True
  62. self.category.save()
  63. self.override_acl({'can_close_threads': 0})
  64. response = self.client.post(self.api_link, {
  65. 'category': self.category.pk,
  66. })
  67. self.assertContains(response, "This category is closed.", status_code=400)
  68. def test_cant_start_thread_in_invalid_category(self):
  69. """can't post in invalid category"""
  70. self.category.is_closed = True
  71. self.category.save()
  72. self.override_acl({'can_close_threads': 0})
  73. response = self.client.post(self.api_link, {'category': self.category.pk * 100000})
  74. self.assertContains(response, "Selected category doesn't exist", status_code=400)
  75. def test_empty_data(self):
  76. """no data sent handling has no showstoppers"""
  77. self.override_acl()
  78. response = self.client.post(self.api_link, data={})
  79. self.assertEqual(response.status_code, 400)
  80. self.assertEqual(
  81. response.json(), {
  82. 'category': ["You have to select category to post thread in."],
  83. 'title': ["You have to enter thread title."],
  84. 'post': ["You have to enter a message."],
  85. }
  86. )
  87. def test_invalid_data(self):
  88. """api errors for invalid request data"""
  89. self.override_acl()
  90. response = self.client.post(
  91. self.api_link,
  92. 'false',
  93. content_type="application/json",
  94. )
  95. self.assertContains(response, "Invalid data.", status_code=400)
  96. def test_title_is_validated(self):
  97. """title is validated"""
  98. self.override_acl()
  99. response = self.client.post(
  100. self.api_link,
  101. data={
  102. 'category': self.category.pk,
  103. 'title': "------",
  104. 'post': "Lorem ipsum dolor met, sit amet elit!",
  105. }
  106. )
  107. self.assertEqual(response.status_code, 400)
  108. self.assertEqual(
  109. response.json(), {
  110. 'title': ["Thread title should contain alpha-numeric characters."],
  111. }
  112. )
  113. def test_post_is_validated(self):
  114. """post is validated"""
  115. self.override_acl()
  116. response = self.client.post(
  117. self.api_link,
  118. data={
  119. 'category': self.category.pk,
  120. 'title': "Lorem ipsum dolor met",
  121. 'post': "a",
  122. }
  123. )
  124. self.assertEqual(response.status_code, 400)
  125. self.assertEqual(
  126. response.json(), {
  127. 'post': ["Posted message should be at least 5 characters long (it has 1)."],
  128. }
  129. )
  130. def test_can_start_thread(self):
  131. """endpoint creates new thread"""
  132. self.override_acl()
  133. response = self.client.post(
  134. self.api_link,
  135. data={
  136. 'category': self.category.pk,
  137. 'title': "Hello, I am test thread!",
  138. 'post': "Lorem ipsum dolor met!",
  139. }
  140. )
  141. self.assertEqual(response.status_code, 200)
  142. thread = self.user.thread_set.all()[:1][0]
  143. response_json = response.json()
  144. self.assertEqual(response_json['url'], thread.get_absolute_url())
  145. self.override_acl()
  146. response = self.client.get(thread.get_absolute_url())
  147. self.assertContains(response, self.category.name)
  148. self.assertContains(response, thread.title)
  149. self.assertContains(response, "<p>Lorem ipsum dolor met!</p>")
  150. # api increased user's threads and posts counts
  151. self.reload_user()
  152. self.assertEqual(self.user.threads, 1)
  153. self.assertEqual(self.user.posts, 1)
  154. self.assertEqual(thread.category_id, self.category.pk)
  155. self.assertEqual(thread.title, "Hello, I am test thread!")
  156. self.assertEqual(thread.starter_id, self.user.id)
  157. self.assertEqual(thread.starter_name, self.user.username)
  158. self.assertEqual(thread.starter_slug, self.user.slug)
  159. self.assertEqual(thread.last_poster_id, self.user.id)
  160. self.assertEqual(thread.last_poster_name, self.user.username)
  161. self.assertEqual(thread.last_poster_slug, self.user.slug)
  162. post = self.user.post_set.all()[:1][0]
  163. self.assertEqual(post.category_id, self.category.pk)
  164. self.assertEqual(post.original, 'Lorem ipsum dolor met!')
  165. self.assertEqual(post.poster_id, self.user.id)
  166. self.assertEqual(post.poster_name, self.user.username)
  167. category = Category.objects.get(pk=self.category.pk)
  168. self.assertEqual(category.threads, 1)
  169. self.assertEqual(category.posts, 1)
  170. self.assertEqual(category.last_thread_id, thread.id)
  171. self.assertEqual(category.last_thread_title, thread.title)
  172. self.assertEqual(category.last_thread_slug, thread.slug)
  173. self.assertEqual(category.last_poster_id, self.user.id)
  174. self.assertEqual(category.last_poster_name, self.user.username)
  175. self.assertEqual(category.last_poster_slug, self.user.slug)
  176. def test_start_closed_thread_no_permission(self):
  177. """permission is checked before thread is closed"""
  178. self.override_acl({'can_close_threads': 0})
  179. response = self.client.post(
  180. self.api_link,
  181. data={
  182. 'category': self.category.pk,
  183. 'title': "Hello, I am test thread!",
  184. 'post': "Lorem ipsum dolor met!",
  185. 'close': True,
  186. }
  187. )
  188. self.assertEqual(response.status_code, 200)
  189. thread = self.user.thread_set.all()[:1][0]
  190. self.assertFalse(thread.is_closed)
  191. def test_start_closed_thread(self):
  192. """can post closed thread"""
  193. self.override_acl({'can_close_threads': 1})
  194. response = self.client.post(
  195. self.api_link,
  196. data={
  197. 'category': self.category.pk,
  198. 'title': "Hello, I am test thread!",
  199. 'post': "Lorem ipsum dolor met!",
  200. 'close': True,
  201. }
  202. )
  203. self.assertEqual(response.status_code, 200)
  204. thread = self.user.thread_set.all()[:1][0]
  205. self.assertTrue(thread.is_closed)
  206. def test_start_unpinned_thread(self):
  207. """can post unpinned thread"""
  208. self.override_acl({'can_pin_threads': 1})
  209. response = self.client.post(
  210. self.api_link,
  211. data={
  212. 'category': self.category.pk,
  213. 'title': "Hello, I am test thread!",
  214. 'post': "Lorem ipsum dolor met!",
  215. 'pin': 0,
  216. }
  217. )
  218. self.assertEqual(response.status_code, 200)
  219. thread = self.user.thread_set.all()[:1][0]
  220. self.assertEqual(thread.weight, 0)
  221. def test_start_locally_pinned_thread(self):
  222. """can post locally pinned thread"""
  223. self.override_acl({'can_pin_threads': 1})
  224. response = self.client.post(
  225. self.api_link,
  226. data={
  227. 'category': self.category.pk,
  228. 'title': "Hello, I am test thread!",
  229. 'post': "Lorem ipsum dolor met!",
  230. 'pin': 1,
  231. }
  232. )
  233. self.assertEqual(response.status_code, 200)
  234. thread = self.user.thread_set.all()[:1][0]
  235. self.assertEqual(thread.weight, 1)
  236. def test_start_globally_pinned_thread(self):
  237. """can post globally pinned thread"""
  238. self.override_acl({'can_pin_threads': 2})
  239. response = self.client.post(
  240. self.api_link,
  241. data={
  242. 'category': self.category.pk,
  243. 'title': "Hello, I am test thread!",
  244. 'post': "Lorem ipsum dolor met!",
  245. 'pin': 2,
  246. }
  247. )
  248. self.assertEqual(response.status_code, 200)
  249. thread = self.user.thread_set.all()[:1][0]
  250. self.assertEqual(thread.weight, 2)
  251. def test_start_globally_pinned_thread_no_permission(self):
  252. """cant post globally pinned thread without permission"""
  253. self.override_acl({'can_pin_threads': 1})
  254. response = self.client.post(
  255. self.api_link,
  256. data={
  257. 'category': self.category.pk,
  258. 'title': "Hello, I am test thread!",
  259. 'post': "Lorem ipsum dolor met!",
  260. 'pin': 2,
  261. }
  262. )
  263. self.assertEqual(response.status_code, 200)
  264. thread = self.user.thread_set.all()[:1][0]
  265. self.assertEqual(thread.weight, 0)
  266. def test_start_locally_pinned_thread_no_permission(self):
  267. """cant post locally pinned thread without permission"""
  268. self.override_acl({'can_pin_threads': 0})
  269. response = self.client.post(
  270. self.api_link,
  271. data={
  272. 'category': self.category.pk,
  273. 'title': "Hello, I am test thread!",
  274. 'post': "Lorem ipsum dolor met!",
  275. 'pin': 1,
  276. }
  277. )
  278. self.assertEqual(response.status_code, 200)
  279. thread = self.user.thread_set.all()[:1][0]
  280. self.assertEqual(thread.weight, 0)
  281. def test_start_hidden_thread(self):
  282. """can post hidden thread"""
  283. self.override_acl({'can_hide_threads': 1})
  284. response = self.client.post(
  285. self.api_link,
  286. data={
  287. 'category': self.category.pk,
  288. 'title': "Hello, I am test thread!",
  289. 'post': "Lorem ipsum dolor met!",
  290. 'hide': 1,
  291. }
  292. )
  293. self.assertEqual(response.status_code, 200)
  294. thread = self.user.thread_set.all()[:1][0]
  295. self.assertTrue(thread.is_hidden)
  296. category = Category.objects.get(pk=self.category.pk)
  297. self.assertNotEqual(category.last_thread_id, thread.id)
  298. def test_start_hidden_thread_no_permission(self):
  299. """cant post hidden thread without permission"""
  300. self.override_acl({'can_hide_threads': 0})
  301. response = self.client.post(
  302. self.api_link,
  303. data={
  304. 'category': self.category.pk,
  305. 'title': "Hello, I am test thread!",
  306. 'post': "Lorem ipsum dolor met!",
  307. 'hide': 1,
  308. }
  309. )
  310. self.assertEqual(response.status_code, 200)
  311. thread = self.user.thread_set.all()[:1][0]
  312. self.assertFalse(thread.is_hidden)
  313. def test_post_unicode(self):
  314. """unicode characters can be posted"""
  315. self.override_acl()
  316. response = self.client.post(
  317. self.api_link,
  318. data={
  319. 'category': self.category.pk,
  320. 'title': "Brzęczyżczykiewicz",
  321. 'post': "Chrzążczyżewoszyce, powiat Łękółody.",
  322. }
  323. )
  324. self.assertEqual(response.status_code, 200)
  325. def test_category_moderation_queue(self):
  326. """start unapproved thread in category that requires approval"""
  327. self.category.require_threads_approval = True
  328. self.category.save()
  329. response = self.client.post(
  330. self.api_link,
  331. data={
  332. 'category': self.category.pk,
  333. 'title': "Hello, I am test thread!",
  334. 'post': "Lorem ipsum dolor met!",
  335. }
  336. )
  337. self.assertEqual(response.status_code, 200)
  338. thread = self.user.thread_set.all()[:1][0]
  339. self.assertTrue(thread.is_unapproved)
  340. self.assertTrue(thread.has_unapproved_posts)
  341. post = self.user.post_set.all()[:1][0]
  342. self.assertTrue(post.is_unapproved)
  343. category = Category.objects.get(slug='first-category')
  344. self.assertEqual(category.threads, self.category.threads)
  345. self.assertEqual(category.posts, self.category.posts)
  346. self.assertFalse(category.last_thread_id == thread.id)
  347. def test_category_moderation_queue_bypass(self):
  348. """bypass moderation queue due to user's acl"""
  349. override_acl(self.user, {'can_approve_content': 1})
  350. self.category.require_threads_approval = True
  351. self.category.save()
  352. response = self.client.post(
  353. self.api_link,
  354. data={
  355. 'category': self.category.pk,
  356. 'title': "Hello, I am test thread!",
  357. 'post': "Lorem ipsum dolor met!",
  358. }
  359. )
  360. self.assertEqual(response.status_code, 200)
  361. thread = self.user.thread_set.all()[:1][0]
  362. self.assertFalse(thread.is_unapproved)
  363. self.assertFalse(thread.has_unapproved_posts)
  364. post = self.user.post_set.all()[:1][0]
  365. self.assertFalse(post.is_unapproved)
  366. category = Category.objects.get(slug='first-category')
  367. self.assertEqual(category.threads, self.category.threads + 1)
  368. self.assertEqual(category.posts, self.category.posts + 1)
  369. self.assertEqual(category.last_thread_id, thread.id)
  370. def test_user_moderation_queue(self):
  371. """start unapproved thread in category that requires approval"""
  372. self.override_acl({'require_threads_approval': 1})
  373. response = self.client.post(
  374. self.api_link,
  375. data={
  376. 'category': self.category.pk,
  377. 'title': "Hello, I am test thread!",
  378. 'post': "Lorem ipsum dolor met!",
  379. }
  380. )
  381. self.assertEqual(response.status_code, 200)
  382. thread = self.user.thread_set.all()[:1][0]
  383. self.assertTrue(thread.is_unapproved)
  384. self.assertTrue(thread.has_unapproved_posts)
  385. post = self.user.post_set.all()[:1][0]
  386. self.assertTrue(post.is_unapproved)
  387. category = Category.objects.get(slug='first-category')
  388. self.assertEqual(category.threads, self.category.threads)
  389. self.assertEqual(category.posts, self.category.posts)
  390. self.assertFalse(category.last_thread_id == thread.id)
  391. def test_user_moderation_queue_bypass(self):
  392. """bypass moderation queue due to user's acl"""
  393. override_acl(self.user, {'can_approve_content': 1})
  394. self.override_acl({'require_threads_approval': 1})
  395. response = self.client.post(
  396. self.api_link,
  397. data={
  398. 'category': self.category.pk,
  399. 'title': "Hello, I am test thread!",
  400. 'post': "Lorem ipsum dolor met!",
  401. }
  402. )
  403. self.assertEqual(response.status_code, 200)
  404. thread = self.user.thread_set.all()[:1][0]
  405. self.assertFalse(thread.is_unapproved)
  406. self.assertFalse(thread.has_unapproved_posts)
  407. post = self.user.post_set.all()[:1][0]
  408. self.assertFalse(post.is_unapproved)
  409. category = Category.objects.get(slug='first-category')
  410. self.assertEqual(category.threads, self.category.threads + 1)
  411. self.assertEqual(category.posts, self.category.posts + 1)
  412. self.assertEqual(category.last_thread_id, thread.id)
  413. def test_omit_other_moderation_queues(self):
  414. """other queues are omitted"""
  415. self.category.require_replies_approval = True
  416. self.category.require_edits_approval = True
  417. self.category.save()
  418. self.override_acl({
  419. 'require_replies_approval': 1,
  420. 'require_edits_approval': 1,
  421. })
  422. response = self.client.post(
  423. self.api_link,
  424. data={
  425. 'category': self.category.pk,
  426. 'title': "Hello, I am test thread!",
  427. 'post': "Lorem ipsum dolor met!",
  428. }
  429. )
  430. self.assertEqual(response.status_code, 200)
  431. thread = self.user.thread_set.all()[:1][0]
  432. self.assertFalse(thread.is_unapproved)
  433. self.assertFalse(thread.has_unapproved_posts)
  434. post = self.user.post_set.all()[:1][0]
  435. self.assertFalse(post.is_unapproved)
  436. category = Category.objects.get(slug='first-category')
  437. self.assertEqual(category.threads, self.category.threads + 1)
  438. self.assertEqual(category.posts, self.category.posts + 1)
  439. self.assertEqual(category.last_thread_id, thread.id)