acl.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. from django import forms
  2. from django.db import models
  3. from django.db.models import Q
  4. from django.utils.translation import ugettext_lazy as _
  5. from misago.acl.builder import BaseACL
  6. from misago.acl.utils import ACLError403, ACLError404
  7. from misago.forms import YesNoSwitch
  8. def make_forum_form(request, role, form):
  9. form.base_fields['can_read_threads'] = forms.ChoiceField(choices=(
  10. ('0', _("No")),
  11. ('1', _("Yes, owned")),
  12. ('2', _("Yes, all")),
  13. ))
  14. form.base_fields['can_start_threads'] = forms.ChoiceField(choices=(
  15. ('0', _("No")),
  16. ('1', _("Yes, with moderation")),
  17. ('2', _("Yes")),
  18. ))
  19. form.base_fields['can_edit_own_threads'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  20. form.base_fields['can_soft_delete_own_threads'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  21. form.base_fields['can_write_posts'] = forms.ChoiceField(choices=(
  22. ('0', _("No")),
  23. ('1', _("Yes, with moderation")),
  24. ('2', _("Yes")),
  25. ))
  26. form.base_fields['can_edit_own_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  27. form.base_fields['can_soft_delete_own_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  28. form.base_fields['can_upvote_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  29. form.base_fields['can_downvote_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  30. form.base_fields['can_see_posts_scores'] = forms.ChoiceField(choices=(
  31. ('0', _("No")),
  32. ('1', _("Yes, final score")),
  33. ('2', _("Yes, both up and down-votes")),
  34. ))
  35. form.base_fields['can_see_votes'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  36. form.base_fields['can_make_polls'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  37. form.base_fields['can_vote_in_polls'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  38. form.base_fields['can_see_poll_votes'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  39. form.base_fields['can_see_attachments'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  40. form.base_fields['can_upload_attachments'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  41. form.base_fields['can_download_attachments'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  42. form.base_fields['attachment_size'] = forms.IntegerField(min_value=0,initial=100)
  43. form.base_fields['attachment_limit'] = forms.IntegerField(min_value=0,initial=3)
  44. form.base_fields['can_approve'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  45. form.base_fields['can_edit_labels'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  46. form.base_fields['can_see_changelog'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  47. form.base_fields['can_pin_threads'] = forms.ChoiceField(choices=(
  48. ('0', _("No")),
  49. ('1', _("Yes, to stickies")),
  50. ('2', _("Yes, to annoucements")),
  51. ))
  52. form.base_fields['can_edit_threads_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  53. form.base_fields['can_move_threads_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  54. form.base_fields['can_close_threads'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  55. form.base_fields['can_protect_posts'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  56. form.base_fields['can_delete_threads'] = forms.ChoiceField(choices=(
  57. ('0', _("No")),
  58. ('1', _("Yes, soft-delete")),
  59. ('2', _("Yes, hard-delete")),
  60. ))
  61. form.base_fields['can_delete_posts'] = forms.ChoiceField(choices=(
  62. ('0', _("No")),
  63. ('1', _("Yes, soft-delete")),
  64. ('2', _("Yes, hard-delete")),
  65. ))
  66. form.base_fields['can_delete_polls'] = forms.ChoiceField(choices=(
  67. ('0', _("No")),
  68. ('1', _("Yes, soft-delete")),
  69. ('2', _("Yes, hard-delete")),
  70. ))
  71. form.base_fields['can_delete_attachments'] = forms.BooleanField(widget=YesNoSwitch,initial=False,required=False)
  72. form.layout.append((
  73. _("Threads"),
  74. (
  75. ('can_read_threads', {'label': _("Can read threads")}),
  76. ('can_start_threads', {'label': _("Can start new threads")}),
  77. ('can_edit_own_threads', {'label': _("Can edit own threads")}),
  78. ('can_soft_delete_own_threads', {'label': _("Can soft-delete own threads")}),
  79. ),
  80. ),)
  81. form.layout.append((
  82. _("Posts"),
  83. (
  84. ('can_write_posts', {'label': _("Can write posts")}),
  85. ('can_edit_own_posts', {'label': _("Can edit own posts")}),
  86. ('can_soft_delete_own_posts', {'label': _("Can soft-delete own posts")}),
  87. ),
  88. ),)
  89. form.layout.append((
  90. _("Karma"),
  91. (
  92. ('can_upvote_posts', {'label': _("Can upvote posts")}),
  93. ('can_downvote_posts', {'label': _("Can downvote posts")}),
  94. ('can_see_posts_scores', {'label': _("Can see post score")}),
  95. ('can_see_votes', {'label': _("Can see who voted on post")}),
  96. ),
  97. ),)
  98. form.layout.append((
  99. _("Polls"),
  100. (
  101. ('can_make_polls', {'label': _("Can make polls")}),
  102. ('can_vote_in_polls', {'label': _("Can vote in polls")}),
  103. ('can_see_poll_votes', {'label': _("Can see who voted in poll")}),
  104. ),
  105. ),)
  106. form.layout.append((
  107. _("Attachments"),
  108. (
  109. ('can_see_attachments', {'label': _("Can see attachments")}),
  110. ('can_upload_attachments', {'label': _("Can upload attachments")}),
  111. ('can_download_attachments', {'label': _("Can download attachments")}),
  112. ('attachment_size', {'label': _("Max size of single attachment (in Kb)"), 'help_text': _("Enter zero for no limit.")}),
  113. ('attachment_limit', {'label': _("Max number of attachments per post"), 'help_text': _("Enter zero for no limit.")}),
  114. ),
  115. ),)
  116. form.layout.append((
  117. _("Moderation"),
  118. (
  119. ('can_approve', {'label': _("Can accept threads and posts")}),
  120. ('can_edit_labels', {'label': _("Can edit thread labels")}),
  121. ('can_see_changelog', {'label': _("Can see edits history")}),
  122. ('can_make_annoucements', {'label': _("Can make annoucements")}),
  123. ('can_pin_threads', {'label': _("Can change threads weight")}),
  124. ('can_edit_threads_posts', {'label': _("Can edit threads and posts")}),
  125. ('can_move_threads_posts', {'label': _("Can move, merge and split threads and posts")}),
  126. ('can_close_threads', {'label': _("Can close threads")}),
  127. ('can_protect_posts', {'label': _("Can protect posts"), 'help_text': _("Protected posts cannot be changed by their owners.")}),
  128. ('can_delete_threads', {'label': _("Can delete threads")}),
  129. ('can_delete_posts', {'label': _("Can delete posts")}),
  130. ('can_delete_polls', {'label': _("Can delete polls")}),
  131. ('can_delete_attachments', {'label': _("Can delete attachments")}),
  132. ),
  133. ),)
  134. class ThreadsACL(BaseACL):
  135. def get_role(self, forum):
  136. try:
  137. try:
  138. return self.acl[forum.pk]
  139. except AttributeError:
  140. return self.acl[forum]
  141. except KeyError:
  142. return {}
  143. def allow_thread_view(self, user, thread):
  144. try:
  145. forum_role = self.acl[thread.forum_id]
  146. if forum_role['can_read_threads'] == 0:
  147. raise ACLError403(_("You don't have permission to read threads in this forum."))
  148. if forum_role['can_read_threads'] == 1 and thread.weight < 2 and thread.start_poster_id != user.id:
  149. raise ACLError404()
  150. if thread.moderated and not (forum_role['can_approve'] or (user.is_authenticated() and user == thread.start_poster)):
  151. raise ACLError404()
  152. except KeyError:
  153. raise ACLError403(_("You don't have permission to read threads in this forum."))
  154. def allow_post_view(self, user, thread, post):
  155. forum_role = self.acl[thread.forum_id]
  156. if post.moderated and not (forum_role['can_approve'] or (user.is_authenticated() and user == post.user)):
  157. raise ACLError404()
  158. if post.deleted and not (forum_role['can_delete_posts'] or (user.is_authenticated() and user == post.user)):
  159. raise ACLError404()
  160. def get_readable_forums(self, acl):
  161. readable = []
  162. for forum in self.acl:
  163. if acl.forums.can_browse(forum) and self.acl[forum]['can_read_threads']:
  164. readable.append(forum)
  165. return readable
  166. def filter_threads(self, request, forum, queryset):
  167. try:
  168. forum_role = self.acl[forum.pk]
  169. if not forum_role['can_approve']:
  170. if request.user.is_authenticated():
  171. queryset = queryset.filter(Q(moderated=0) | Q(start_poster=request.user))
  172. else:
  173. queryset = queryset.filter(moderated=0)
  174. if forum_role['can_read_threads'] == 1:
  175. queryset = queryset.filter(Q(weight=2) | Q(start_poster_id=request.user.id))
  176. except KeyError:
  177. return False
  178. return queryset
  179. def filter_posts(self, request, thread, queryset):
  180. try:
  181. forum_role = self.acl[thread.forum.pk]
  182. if not forum_role['can_approve']:
  183. if request.user.is_authenticated():
  184. queryset = queryset.filter(Q(moderated=0) | Q(user=request.user))
  185. else:
  186. queryset = queryset.filter(moderated=0)
  187. except KeyError:
  188. return False
  189. return queryset
  190. def can_start_threads(self, forum):
  191. try:
  192. forum_role = self.acl[forum.pk]
  193. if forum_role['can_read_threads'] == 0 or forum_role['can_start_threads'] == 0:
  194. return False
  195. if forum.closed and forum_role['can_close_threads'] == 0:
  196. return False
  197. return True
  198. except KeyError:
  199. return False
  200. def allow_new_threads(self, forum):
  201. try:
  202. forum_role = self.acl[forum.pk]
  203. if forum_role['can_read_threads'] == 0 or forum_role['can_start_threads'] == 0:
  204. raise ACLError403(_("You don't have permission to start new threads in this forum."))
  205. if forum.closed and forum_role['can_close_threads'] == 0:
  206. raise ACLError403(_("This forum is closed, you can't start new threads in it."))
  207. except KeyError:
  208. raise ACLError403(_("You don't have permission to start new threads in this forum."))
  209. def can_edit_thread(self, user, forum, thread, post):
  210. try:
  211. forum_role = self.acl[thread.forum_id]
  212. if forum_role['can_close_threads'] == 0 and (forum.closed or thread.closed):
  213. return False
  214. if forum_role['can_edit_threads_posts']:
  215. return True
  216. if forum_role['can_edit_own_threads'] and not post.protected and post.user_id == user.pk:
  217. return True
  218. return False
  219. except KeyError:
  220. return False
  221. def allow_thread_edit(self, user, forum, thread, post):
  222. try:
  223. forum_role = self.acl[thread.forum_id]
  224. if not forum_role['can_close_threads']:
  225. if forum.closed:
  226. raise ACLError403(_("You can't edit threads in closed forums."))
  227. if thread.closed:
  228. raise ACLError403(_("You can't edit closed threads."))
  229. if not forum_role['can_edit_threads_posts']:
  230. if post.user_id != user.pk:
  231. raise ACLError403(_("You can't edit other members threads."))
  232. if not forum_role['can_edit_own_threads']:
  233. raise ACLError403(_("You can't edit your threads."))
  234. if post.protected:
  235. raise ACLError403(_("This thread is protected, you cannot edit it."))
  236. except KeyError:
  237. raise ACLError403(_("You don't have permission to edit threads in this forum."))
  238. def can_reply(self, forum, thread):
  239. try:
  240. forum_role = self.acl[forum.pk]
  241. if forum_role['can_write_posts'] == 0:
  242. return False
  243. if (forum.closed or thread.closed) and forum_role['can_close_threads'] == 0:
  244. return False
  245. return True
  246. except KeyError:
  247. return False
  248. def allow_reply(self, forum, thread):
  249. try:
  250. forum_role = self.acl[thread.forum.pk]
  251. if forum_role['can_write_posts'] == 0:
  252. raise ACLError403(_("You don't have permission to write replies in this forum."))
  253. if forum_role['can_close_threads'] == 0:
  254. if forum.closed:
  255. raise ACLError403(_("You can't write replies in closed forums."))
  256. if thread.closed:
  257. raise ACLError403(_("You can't write replies in closed threads."))
  258. except KeyError:
  259. raise ACLError403(_("You don't have permission to write replies in this forum."))
  260. def can_edit_reply(self, user, forum, thread, post):
  261. try:
  262. forum_role = self.acl[thread.forum_id]
  263. if forum_role['can_close_threads'] == 0 and (forum.closed or thread.closed):
  264. return False
  265. if forum_role['can_edit_threads_posts']:
  266. return True
  267. if forum_role['can_edit_own_posts'] and not post.protected and post.user_id == user.pk:
  268. return True
  269. return False
  270. except KeyError:
  271. return False
  272. def allow_reply_edit(self, user, forum, thread, post):
  273. try:
  274. forum_role = self.acl[thread.forum_id]
  275. if not forum_role['can_close_threads']:
  276. if forum.closed:
  277. raise ACLError403(_("You can't edit replies in closed forums."))
  278. if thread.closed:
  279. raise ACLError403(_("You can't edit replies in closed threads."))
  280. if not forum_role['can_edit_threads_posts']:
  281. if post.user_id != user.pk:
  282. raise ACLError403(_("You can't edit other members replies."))
  283. if not forum_role['can_edit_own_posts']:
  284. raise ACLError403(_("You can't edit your replies."))
  285. if post.protected:
  286. raise ACLError403(_("This reply is protected, you cannot edit it."))
  287. except KeyError:
  288. raise ACLError403(_("You don't have permission to edit replies in this forum."))
  289. def can_see_changelog(self, user, forum, post):
  290. try:
  291. forum_role = self.acl[forum.pk]
  292. return forum_role['can_see_changelog'] or user.pk == post.user_id
  293. except KeyError:
  294. return False
  295. def allow_changelog_view(self, user, forum, post):
  296. try:
  297. forum_role = self.acl[forum.pk]
  298. if not (forum_role['can_see_changelog'] or user.pk == post.user_id):
  299. raise ACLError403(_("You don't have permission to see history of changes made to this post."))
  300. except KeyError:
  301. raise ACLError403(_("You don't have permission to see history of changes made to this post."))
  302. def can_make_revert(self, forum, thread):
  303. try:
  304. forum_role = self.acl[forum.pk]
  305. if not forum_role['can_close_threads'] and (forum.closed or thread.closed):
  306. return False
  307. return forum_role['can_edit_threads_posts']
  308. except KeyError:
  309. return False
  310. def allow_revert(self, forum, thread):
  311. try:
  312. forum_role = self.acl[forum.pk]
  313. if not forum_role['can_close_threads']:
  314. if forum.closed:
  315. raise ACLError403(_("You can't make reverts in closed forums."))
  316. if thread.closed:
  317. raise ACLError403(_("You can't make reverts in closed threads."))
  318. if not forum_role['can_edit_threads_posts']:
  319. raise ACLError403(_("You don't have permission to make reverts in this forum."))
  320. except KeyError:
  321. raise ACLError403(_("You don't have permission to make reverts in this forum."))
  322. def can_mod_threads(self, forum):
  323. try:
  324. forum_role = self.acl[forum.pk]
  325. return (
  326. forum_role['can_approve']
  327. or forum_role['can_pin_threads']
  328. or forum_role['can_move_threads_posts']
  329. or forum_role['can_close_threads']
  330. or forum_role['can_delete_threads']
  331. )
  332. except KeyError:
  333. return False
  334. def can_mod_posts(self, thread):
  335. try:
  336. forum_role = self.acl[thread.forum.pk]
  337. return (
  338. forum_role['can_edit_threads_posts']
  339. or forum_role['can_move_threads_posts']
  340. or forum_role['can_close_threads']
  341. or forum_role['can_delete_threads']
  342. or forum_role['can_delete_posts']
  343. )
  344. except KeyError:
  345. return False
  346. def can_mod_thread(self, thread):
  347. pass
  348. def can_approve(self, forum):
  349. try:
  350. forum_role = self.acl[forum.pk]
  351. return forum_role['can_approve']
  352. except KeyError:
  353. return False
  354. def build_forums(acl, perms, forums, forum_roles):
  355. acl.threads = ThreadsACL()
  356. for forum in forums:
  357. forum_role = {
  358. 'can_read_threads': 0,
  359. 'can_start_threads': 0,
  360. 'can_edit_own_threads': False,
  361. 'can_soft_delete_own_threads': False,
  362. 'can_write_posts': 0,
  363. 'can_edit_own_posts': False,
  364. 'can_soft_delete_own_posts': False,
  365. 'can_upvote_posts': False,
  366. 'can_downvote_posts': False,
  367. 'can_see_posts_scores': 0,
  368. 'can_see_votes': False,
  369. 'can_make_polls': False,
  370. 'can_vote_in_polls': False,
  371. 'can_see_poll_votes': False,
  372. 'can_see_attachments': False,
  373. 'can_upload_attachments': False,
  374. 'can_download_attachments': False,
  375. 'attachment_size': 100,
  376. 'attachment_limit': 3,
  377. 'can_approve': False,
  378. 'can_edit_labels': False,
  379. 'can_see_changelog': False,
  380. 'can_make_annoucements': False,
  381. 'can_pin_threads': 0,
  382. 'can_edit_threads_posts': False,
  383. 'can_move_threads_posts': False,
  384. 'can_close_threads': False,
  385. 'can_protect_posts': False,
  386. 'can_delete_threads': 0,
  387. 'can_delete_posts': 0,
  388. 'can_delete_polls': 0,
  389. 'can_delete_attachments': False,
  390. }
  391. for perm in perms:
  392. try:
  393. role = forum_roles[perm['forums'][forum.pk]]
  394. for p in forum_role:
  395. try:
  396. if p in ['attachment_size', 'attachment_limit'] and role[p] == 0:
  397. forum_role[p] = 0
  398. elif int(role[p]) > forum_role[p]:
  399. forum_role[p] = int(role[p])
  400. except KeyError:
  401. pass
  402. except KeyError:
  403. pass
  404. acl.threads.acl[forum.pk] = forum_role