test_thread_start_api.py 20 KB

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