test_thread_post_api.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. import json
  4. from django.conf import settings
  5. from misago.acl.testutils import override_acl
  6. from misago.categories.models import THREADS_ROOT_NAME, Category
  7. from misago.users.testutils import AuthenticatedUserTestCase
  8. from ..models import Thread
  9. from ..threadtypes import trees_map
  10. class StartThreadTests(AuthenticatedUserTestCase):
  11. def setUp(self):
  12. super(StartThreadTests, self).setUp()
  13. threads_tree_id = trees_map.get_tree_id_for_root(THREADS_ROOT_NAME)
  14. self.root = Category.objects.get(tree_id=threads_tree_id, level=0)
  15. self.category = Category.objects.get(slug='first-category')
  16. self.api_link = '/api/threads/'
  17. def override_acl(self, extra_acl=None):
  18. new_acl = self.user.acl
  19. new_acl['categories'][self.category.pk].update({
  20. 'can_see': 1,
  21. 'can_browse': 1,
  22. 'can_start_threads': 1,
  23. 'can_pin_threads': 0,
  24. 'can_close_threads': 0,
  25. 'can_hide_threads': 0,
  26. 'can_hide_own_threads': 0
  27. })
  28. if extra_acl:
  29. new_acl['categories'][self.category.pk].update(extra_acl)
  30. if 'can_see' in extra_acl and not extra_acl['can_see']:
  31. new_acl['visible_categories'].remove(self.category.pk)
  32. new_acl['browseable_categories'].remove(self.category.pk)
  33. if 'can_browse' in extra_acl and not extra_acl['can_browse']:
  34. new_acl['browseable_categories'].remove(self.category.pk)
  35. override_acl(self.user, new_acl)
  36. def test_cant_start_thread_as_guest(self):
  37. """user has to be authenticated to be able to post thread"""
  38. self.logout_user()
  39. response = self.client.post(self.api_link)
  40. self.assertEqual(response.status_code, 403)
  41. def test_cant_see(self):
  42. """has no permission to see selected category"""
  43. self.override_acl({'can_see': 0})
  44. response = self.client.post(self.api_link, {
  45. 'category': self.category.pk
  46. })
  47. self.assertContains(response, "Selected category is invalid.", status_code=400)
  48. def test_cant_browse(self):
  49. """has no permission to browse selected category"""
  50. self.override_acl({'can_browse': 0})
  51. response = self.client.post(self.api_link, {
  52. 'category': self.category.pk
  53. })
  54. self.assertContains(response, "Selected category is invalid.", status_code=400)
  55. def test_cant_start_thread(self):
  56. """permission to start thread in category is validated"""
  57. self.override_acl({'can_start_threads': 0})
  58. response = self.client.post(self.api_link, {
  59. 'category': self.category.pk
  60. })
  61. self.assertContains(response, "You don't have permission to start new threads", status_code=400)
  62. def test_cant_start_thread_in_locked_category(self):
  63. """can't post in closed category"""
  64. self.category.is_closed = True
  65. self.category.save()
  66. self.override_acl({'can_close_threads': 0})
  67. response = self.client.post(self.api_link, {
  68. 'category': self.category.pk
  69. })
  70. self.assertContains(response, "This category is closed.", status_code=400)
  71. def test_cant_start_thread_in_invalid_category(self):
  72. """can't post in invalid category"""
  73. self.category.is_closed = True
  74. self.category.save()
  75. self.override_acl({'can_close_threads': 0})
  76. response = self.client.post(self.api_link, {
  77. 'category': self.category.pk * 100000
  78. })
  79. self.assertContains(response, "Selected category doesn't exist", status_code=400)
  80. def test_empty_data(self):
  81. """no data sent handling has no showstoppers"""
  82. self.override_acl()
  83. response = self.client.post(self.api_link, data={})
  84. self.assertEqual(response.status_code, 400)
  85. self.assertEqual(json.loads(response.content), {
  86. 'category': [
  87. "You have to select category to post thread in."
  88. ],
  89. 'title':[
  90. "You have to enter thread title."
  91. ],
  92. 'post': [
  93. "You have to enter a message"
  94. ]
  95. })
  96. def test_title_is_validated(self):
  97. """title is validated"""
  98. self.override_acl()
  99. response = self.client.post(self.api_link, data={
  100. 'category': self.category.pk,
  101. 'title': '------',
  102. 'post': 'Lorem ipsum dolor met, sit amet elit!',
  103. })
  104. self.assertEqual(response.status_code, 400)
  105. self.assertEqual(json.loads(response.content), {
  106. 'title': [
  107. "Thread title should contain alpha-numeric characters."
  108. ]
  109. })
  110. def test_post_is_validated(self):
  111. """post is validated"""
  112. self.override_acl()
  113. response = self.client.post(self.api_link, data={
  114. 'category': self.category.pk,
  115. 'title': 'Lorem ipsum dolor met',
  116. 'post': 'a',
  117. })
  118. self.assertEqual(response.status_code, 400)
  119. self.assertEqual(json.loads(response.content), {
  120. 'post': [
  121. "Posted message should be at least 5 characters long (it has 1)."
  122. ]
  123. })
  124. def test_can_start_thread(self):
  125. """endpoint creates new thread"""
  126. self.override_acl()
  127. response = self.client.post(self.api_link, data={
  128. 'category': self.category.pk,
  129. 'title': "Hello, I am test thread!",
  130. 'post': "Lorem ipsum dolor met!"
  131. })
  132. self.assertEqual(response.status_code, 200)
  133. thread = self.user.thread_set.all()[:1][0]
  134. response_json = json.loads(response.content)
  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. self.reload_user()
  142. self.assertEqual(self.user.threads, 1)
  143. self.assertEqual(self.user.posts, 1)
  144. self.assertEqual(thread.category_id, self.category.pk)
  145. self.assertEqual(thread.title, "Hello, I am test thread!")
  146. self.assertEqual(thread.starter_id, self.user.id)
  147. self.assertEqual(thread.starter_name, self.user.username)
  148. self.assertEqual(thread.starter_slug, self.user.slug)
  149. self.assertEqual(thread.last_poster_id, self.user.id)
  150. self.assertEqual(thread.last_poster_name, self.user.username)
  151. self.assertEqual(thread.last_poster_slug, self.user.slug)
  152. post = self.user.post_set.all()[:1][0]
  153. self.assertEqual(post.category_id, self.category.pk)
  154. self.assertEqual(post.original, 'Lorem ipsum dolor met!')
  155. self.assertEqual(post.poster_id, self.user.id)
  156. self.assertEqual(post.poster_name, self.user.username)
  157. category = Category.objects.get(pk=self.category.pk)
  158. self.assertEqual(category.threads, 1)
  159. self.assertEqual(category.posts, 1)
  160. self.assertEqual(category.last_thread_id, thread.id)
  161. self.assertEqual(category.last_thread_title, thread.title)
  162. self.assertEqual(category.last_thread_slug, thread.slug)
  163. self.assertEqual(category.last_poster_id, self.user.id)
  164. self.assertEqual(category.last_poster_name, self.user.username)
  165. self.assertEqual(category.last_poster_slug, self.user.slug)
  166. def test_start_closed_thread_no_permission(self):
  167. """permission is checked before thread is closed"""
  168. self.override_acl({'can_close_threads': 0})
  169. response = self.client.post(self.api_link, data={
  170. 'category': self.category.pk,
  171. 'title': 'Hello, I am test thread!',
  172. 'post': 'Lorem ipsum dolor met!',
  173. 'close': True
  174. })
  175. self.assertEqual(response.status_code, 200)
  176. thread = self.user.thread_set.all()[:1][0]
  177. self.assertFalse(thread.is_closed)
  178. def test_start_closed_thread(self):
  179. """can post closed thread"""
  180. self.override_acl({'can_close_threads': 1})
  181. response = self.client.post(self.api_link, data={
  182. 'category': self.category.pk,
  183. 'title': 'Hello, I am test thread!',
  184. 'post': 'Lorem ipsum dolor met!',
  185. 'close': True
  186. })
  187. self.assertEqual(response.status_code, 200)
  188. thread = self.user.thread_set.all()[:1][0]
  189. self.assertTrue(thread.is_closed)
  190. def test_start_unpinned_thread(self):
  191. """can post unpinned thread"""
  192. self.override_acl({'can_pin_threads': 1})
  193. response = self.client.post(self.api_link, data={
  194. 'category': self.category.pk,
  195. 'title': 'Hello, I am test thread!',
  196. 'post': 'Lorem ipsum dolor met!',
  197. 'pin': 0
  198. })
  199. self.assertEqual(response.status_code, 200)
  200. thread = self.user.thread_set.all()[:1][0]
  201. self.assertEqual(thread.weight, 0)
  202. def test_start_locally_pinned_thread(self):
  203. """can post locally pinned thread"""
  204. self.override_acl({'can_pin_threads': 1})
  205. response = self.client.post(self.api_link, data={
  206. 'category': self.category.pk,
  207. 'title': 'Hello, I am test thread!',
  208. 'post': 'Lorem ipsum dolor met!',
  209. 'pin': 1
  210. })
  211. self.assertEqual(response.status_code, 200)
  212. thread = self.user.thread_set.all()[:1][0]
  213. self.assertEqual(thread.weight, 1)
  214. def test_start_globally_pinned_thread(self):
  215. """can post globally pinned thread"""
  216. self.override_acl({'can_pin_threads': 2})
  217. response = self.client.post(self.api_link, data={
  218. 'category': self.category.pk,
  219. 'title': 'Hello, I am test thread!',
  220. 'post': 'Lorem ipsum dolor met!',
  221. 'pin': 2
  222. })
  223. self.assertEqual(response.status_code, 200)
  224. thread = self.user.thread_set.all()[:1][0]
  225. self.assertEqual(thread.weight, 2)
  226. def test_start_globally_pinned_thread_no_permission(self):
  227. """cant post globally pinned thread without permission"""
  228. self.override_acl({'can_pin_threads': 1})
  229. response = self.client.post(self.api_link, data={
  230. 'category': self.category.pk,
  231. 'title': 'Hello, I am test thread!',
  232. 'post': 'Lorem ipsum dolor met!',
  233. 'pin': 2
  234. })
  235. self.assertEqual(response.status_code, 200)
  236. thread = self.user.thread_set.all()[:1][0]
  237. self.assertEqual(thread.weight, 0)
  238. def test_start_locally_pinned_thread_no_permission(self):
  239. """cant post locally pinned thread without permission"""
  240. self.override_acl({'can_pin_threads': 0})
  241. response = self.client.post(self.api_link, data={
  242. 'category': self.category.pk,
  243. 'title': 'Hello, I am test thread!',
  244. 'post': 'Lorem ipsum dolor met!',
  245. 'pin': 1
  246. })
  247. self.assertEqual(response.status_code, 200)
  248. thread = self.user.thread_set.all()[:1][0]
  249. self.assertEqual(thread.weight, 0)
  250. def test_start_hidden_thread(self):
  251. """can post hidden thread"""
  252. self.override_acl({'can_hide_threads': 1})
  253. response = self.client.post(self.api_link, data={
  254. 'category': self.category.pk,
  255. 'title': 'Hello, I am test thread!',
  256. 'post': 'Lorem ipsum dolor met!',
  257. 'hide': 1
  258. })
  259. self.assertEqual(response.status_code, 200)
  260. thread = self.user.thread_set.all()[:1][0]
  261. self.assertTrue(thread.is_hidden)
  262. category = Category.objects.get(pk=self.category.pk)
  263. self.assertNotEqual(category.last_thread_id, thread.id)
  264. def test_start_hidden_thread_no_permission(self):
  265. """cant post hidden thread without permission"""
  266. self.override_acl({'can_hide_threads': 0})
  267. response = self.client.post(self.api_link, data={
  268. 'category': self.category.pk,
  269. 'title': 'Hello, I am test thread!',
  270. 'post': 'Lorem ipsum dolor met!',
  271. 'hide': 1
  272. })
  273. self.assertEqual(response.status_code, 200)
  274. thread = self.user.thread_set.all()[:1][0]
  275. self.assertFalse(thread.is_hidden)
  276. def test_post_unicode(self):
  277. """unicode characters can be posted"""
  278. self.override_acl()
  279. response = self.client.post(self.api_link, data={
  280. 'category': self.category.pk,
  281. 'title': 'Brzęczyżczykiewicz',
  282. 'post': 'Chrzążczyżewoszyce, powiat Łękółody.'
  283. })
  284. self.assertEqual(response.status_code, 200)