test_thread_start_api.py 20 KB

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