test_thread_start_api.py 19 KB

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