test_thread_start_api.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  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_title_is_validated(self):
  88. """title is validated"""
  89. self.override_acl()
  90. response = self.client.post(
  91. self.api_link,
  92. data={
  93. 'category': self.category.pk,
  94. 'title': "------",
  95. 'post': "Lorem ipsum dolor met, sit amet elit!",
  96. }
  97. )
  98. self.assertEqual(response.status_code, 400)
  99. self.assertEqual(
  100. response.json(), {
  101. 'title': ["Thread title should contain alpha-numeric characters."],
  102. }
  103. )
  104. def test_post_is_validated(self):
  105. """post is validated"""
  106. self.override_acl()
  107. response = self.client.post(
  108. self.api_link,
  109. data={
  110. 'category': self.category.pk,
  111. 'title': "Lorem ipsum dolor met",
  112. 'post': "a",
  113. }
  114. )
  115. self.assertEqual(response.status_code, 400)
  116. self.assertEqual(
  117. response.json(), {
  118. 'post': ["Posted message should be at least 5 characters long (it has 1)."],
  119. }
  120. )
  121. def test_can_start_thread(self):
  122. """endpoint creates new thread"""
  123. self.override_acl()
  124. response = self.client.post(
  125. self.api_link,
  126. data={
  127. 'category': self.category.pk,
  128. 'title': "Hello, I am test thread!",
  129. 'post': "Lorem ipsum dolor met!",
  130. }
  131. )
  132. self.assertEqual(response.status_code, 200)
  133. thread = self.user.thread_set.all()[:1][0]
  134. response_json = response.json()
  135. self.assertEqual(response_json['url'], thread.get_absolute_url())
  136. self.override_acl()
  137. response = self.client.get(thread.get_absolute_url())
  138. self.assertContains(response, self.category.name)
  139. self.assertContains(response, thread.title)
  140. self.assertContains(response, "<p>Lorem ipsum dolor met!</p>")
  141. # api increased user's threads and posts counts
  142. self.reload_user()
  143. self.assertEqual(self.user.threads, 1)
  144. self.assertEqual(self.user.posts, 1)
  145. self.assertEqual(thread.category_id, self.category.pk)
  146. self.assertEqual(thread.title, "Hello, I am test thread!")
  147. self.assertEqual(thread.starter_id, self.user.id)
  148. self.assertEqual(thread.starter_name, self.user.username)
  149. self.assertEqual(thread.starter_slug, self.user.slug)
  150. self.assertEqual(thread.last_poster_id, self.user.id)
  151. self.assertEqual(thread.last_poster_name, self.user.username)
  152. self.assertEqual(thread.last_poster_slug, self.user.slug)
  153. post = self.user.post_set.all()[:1][0]
  154. self.assertEqual(post.category_id, self.category.pk)
  155. self.assertEqual(post.original, 'Lorem ipsum dolor met!')
  156. self.assertEqual(post.poster_id, self.user.id)
  157. self.assertEqual(post.poster_name, self.user.username)
  158. category = Category.objects.get(pk=self.category.pk)
  159. self.assertEqual(category.threads, 1)
  160. self.assertEqual(category.posts, 1)
  161. self.assertEqual(category.last_thread_id, thread.id)
  162. self.assertEqual(category.last_thread_title, thread.title)
  163. self.assertEqual(category.last_thread_slug, thread.slug)
  164. self.assertEqual(category.last_poster_id, self.user.id)
  165. self.assertEqual(category.last_poster_name, self.user.username)
  166. self.assertEqual(category.last_poster_slug, self.user.slug)
  167. def test_start_closed_thread_no_permission(self):
  168. """permission is checked before thread is closed"""
  169. self.override_acl({'can_close_threads': 0})
  170. response = self.client.post(
  171. self.api_link,
  172. data={
  173. 'category': self.category.pk,
  174. 'title': "Hello, I am test thread!",
  175. 'post': "Lorem ipsum dolor met!",
  176. 'close': True,
  177. }
  178. )
  179. self.assertEqual(response.status_code, 200)
  180. thread = self.user.thread_set.all()[:1][0]
  181. self.assertFalse(thread.is_closed)
  182. def test_start_closed_thread(self):
  183. """can post closed thread"""
  184. self.override_acl({'can_close_threads': 1})
  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.assertTrue(thread.is_closed)
  197. def test_start_unpinned_thread(self):
  198. """can post unpinned thread"""
  199. self.override_acl({'can_pin_threads': 1})
  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. 'pin': 0,
  207. }
  208. )
  209. self.assertEqual(response.status_code, 200)
  210. thread = self.user.thread_set.all()[:1][0]
  211. self.assertEqual(thread.weight, 0)
  212. def test_start_locally_pinned_thread(self):
  213. """can post locally pinned thread"""
  214. self.override_acl({'can_pin_threads': 1})
  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': 1,
  222. }
  223. )
  224. self.assertEqual(response.status_code, 200)
  225. thread = self.user.thread_set.all()[:1][0]
  226. self.assertEqual(thread.weight, 1)
  227. def test_start_globally_pinned_thread(self):
  228. """can post globally pinned thread"""
  229. self.override_acl({'can_pin_threads': 2})
  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': 2,
  237. }
  238. )
  239. self.assertEqual(response.status_code, 200)
  240. thread = self.user.thread_set.all()[:1][0]
  241. self.assertEqual(thread.weight, 2)
  242. def test_start_globally_pinned_thread_no_permission(self):
  243. """cant post globally pinned thread without permission"""
  244. self.override_acl({'can_pin_threads': 1})
  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, 0)
  257. def test_start_locally_pinned_thread_no_permission(self):
  258. """cant post locally pinned thread without permission"""
  259. self.override_acl({'can_pin_threads': 0})
  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': 1,
  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. def test_start_hidden_thread(self):
  273. """can post hidden thread"""
  274. self.override_acl({'can_hide_threads': 1})
  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. 'hide': 1,
  282. }
  283. )
  284. self.assertEqual(response.status_code, 200)
  285. thread = self.user.thread_set.all()[:1][0]
  286. self.assertTrue(thread.is_hidden)
  287. category = Category.objects.get(pk=self.category.pk)
  288. self.assertNotEqual(category.last_thread_id, thread.id)
  289. def test_start_hidden_thread_no_permission(self):
  290. """cant post hidden thread without permission"""
  291. self.override_acl({'can_hide_threads': 0})
  292. response = self.client.post(
  293. self.api_link,
  294. data={
  295. 'category': self.category.pk,
  296. 'title': "Hello, I am test thread!",
  297. 'post': "Lorem ipsum dolor met!",
  298. 'hide': 1,
  299. }
  300. )
  301. self.assertEqual(response.status_code, 200)
  302. thread = self.user.thread_set.all()[:1][0]
  303. self.assertFalse(thread.is_hidden)
  304. def test_post_unicode(self):
  305. """unicode characters can be posted"""
  306. self.override_acl()
  307. response = self.client.post(
  308. self.api_link,
  309. data={
  310. 'category': self.category.pk,
  311. 'title': "Brzęczyżczykiewicz",
  312. 'post': "Chrzążczyżewoszyce, powiat Łękółody.",
  313. }
  314. )
  315. self.assertEqual(response.status_code, 200)