views.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. import copy
  2. from django.core.urlresolvers import reverse as django_reverse
  3. from django.shortcuts import redirect
  4. from django.utils.translation import ugettext as _
  5. from misago.acl.builder import build_form
  6. from misago.admin import site
  7. from misago.core.admin.widgets import *
  8. from misago.forms import Form, YesNoSwitch
  9. from misago.models import Forum, ForumRole, Role
  10. from misago.utils.strings import slugify
  11. from misago.core.admin.roles.forms import RoleForm
  12. def reverse(route, target=None):
  13. if target:
  14. return django_reverse(route, kwargs={'target': target.pk, 'slug': slugify(target.name)})
  15. return django_reverse(route)
  16. """
  17. Views
  18. """
  19. class List(ListWidget):
  20. admin = site.get_action('roles')
  21. id = 'list'
  22. columns=(
  23. ('role', _("Role")),
  24. )
  25. nothing_checked_message = _('You have to check at least one role.')
  26. actions=(
  27. ('delete', _("Delete selected roles"), _("Are you sure you want to delete selected roles?")),
  28. )
  29. def sort_items(self, page_items, sorting_method):
  30. return page_items.order_by('name')
  31. def get_item_actions(self, item):
  32. return (
  33. self.action('list', _("Forums Permissions"), reverse('admin_roles_masks', item)),
  34. self.action('adjust', _("Role Permissions"), reverse('admin_roles_acl', item)),
  35. self.action('pencil', _("Edit Role"), reverse('admin_roles_edit', item)),
  36. self.action('remove', _("Delete Role"), reverse('admin_roles_delete', item), post=True, prompt=_("Are you sure you want to delete this role?")),
  37. )
  38. def action_delete(self, items, checked):
  39. for item in items:
  40. if unicode(item.pk) in checked:
  41. if item.token:
  42. return Message(_('You cannot delete system roles.'), 'error'), reverse('admin_roles')
  43. if item.protected and not self.request.user.is_god():
  44. return Message(_('You cannot delete protected roles.'), 'error'), reverse('admin_roles')
  45. if item.user_set.count() > 0:
  46. return Message(_('You cannot delete roles that are assigned to users.'), 'error'), reverse('admin_roles')
  47. Role.objects.filter(id__in=checked).delete()
  48. return Message(_('Selected roles have been deleted successfully.'), 'success'), reverse('admin_roles')
  49. class New(FormWidget):
  50. admin = site.get_action('roles')
  51. id = 'new'
  52. fallback = 'admin_roles'
  53. form = RoleForm
  54. submit_button = _("Save Role")
  55. def get_new_url(self, model):
  56. return reverse('admin_roles_new')
  57. def get_edit_url(self, model):
  58. return reverse('admin_roles_edit', model)
  59. def submit_form(self, form, target):
  60. new_role = Role(
  61. name = form.cleaned_data['name'],
  62. )
  63. new_role.save(force_insert=True)
  64. return new_role, Message(_('New Role has been created.'), 'success')
  65. class Edit(FormWidget):
  66. admin = site.get_action('roles')
  67. id = 'edit'
  68. name = _("Edit Role")
  69. fallback = 'admin_roles'
  70. form = RoleForm
  71. target_name = 'name'
  72. translate_target_name = True
  73. notfound_message = _('Requested Role could not be found.')
  74. submit_fallback = True
  75. def get_url(self, model):
  76. return reverse('admin_roles_edit', model)
  77. def get_edit_url(self, model):
  78. return self.get_url(model)
  79. def get_initial_data(self, model):
  80. if self.request.user.is_god():
  81. return {'name': model.name, 'protected': model.protected}
  82. return {'name': model.name}
  83. def get_and_validate_target(self, target):
  84. result = super(Edit, self).get_and_validate_target(target)
  85. if result and result.protected and not self.request.user.is_god():
  86. self.request.messages.set_flash(Message(_('Role "%(name)s" is protected, you cannot edit it.') % {'name': _(result.name)}), 'error', self.admin.id)
  87. return None
  88. return result
  89. def submit_form(self, form, target):
  90. target.name = form.cleaned_data['name']
  91. if self.request.user.is_god():
  92. target.protected = form.cleaned_data['protected']
  93. target.save(force_update=True)
  94. self.request.monitor['acl_version'] = int(self.request.monitor['acl_version']) + 1
  95. return target, Message(_('Changes in role "%(name)s" have been saved.') % {'name': self.original_name}, 'success')
  96. class Forums(ListWidget):
  97. admin = site.get_action('roles')
  98. id = 'forums'
  99. hide_actions = True
  100. name = _('Role Forums Permissions')
  101. table_form_button = _('Change Permissions')
  102. empty_message = _('No forums are currently defined.')
  103. template = 'forums'
  104. def get_url(self):
  105. return reverse('admin_roles_masks', self.role)
  106. def get_items(self):
  107. return Forum.objects.get(token='root').get_descendants()
  108. def sort_items(self, page_items, sorting_method):
  109. final_items = []
  110. for forum in Forum.objects.filter(token__in=['annoucements', 'reports', 'private']).order_by('token'):
  111. if forum.token == 'annoucements':
  112. forum.name = _("Global Annoucements")
  113. if forum.token == 'reports':
  114. forum.name = _("Reports")
  115. if forum.token == 'private':
  116. forum.name = _("Private Discussions")
  117. final_items.append(forum)
  118. for forum in page_items.order_by('lft').all():
  119. final_items.append(forum)
  120. return final_items
  121. def add_template_variables(self, variables):
  122. variables['target'] = _(self.role.name)
  123. return variables
  124. def get_table_form(self, page_items):
  125. perms = {}
  126. try:
  127. forums = self.role.permissions['forums']
  128. for fid in forums:
  129. perms[str(fid)] = str(forums[fid])
  130. except KeyError:
  131. pass
  132. perms_form = {}
  133. roles_select = [("0", _("No Access"))]
  134. for role in self.roles:
  135. roles_select.append((str(role.pk), _(role.name)))
  136. for item in page_items:
  137. perms_form['forum_' + str(item.pk)] = forms.ChoiceField(choices=roles_select,initial=(perms[str(item.pk)] if str(item.pk) in perms else "0"))
  138. # Turn dict into object
  139. return type('ChangeForumRolesForm', (Form,), perms_form)
  140. def table_action(self, page_items, cleaned_data):
  141. perms = {}
  142. for item in page_items:
  143. if cleaned_data['forum_' + str(item.pk)] != "0":
  144. perms[item.pk] = long(cleaned_data['forum_' + str(item.pk)])
  145. role_perms = self.role.permissions
  146. role_perms['forums'] = perms
  147. self.role.permissions = role_perms
  148. self.role.save(force_update=True)
  149. return Message(_('Forum permissions have been saved.'), 'success'), self.get_url()
  150. def __call__(self, request, slug, target):
  151. self.request = request
  152. try:
  153. self.role = Role.objects.get(id=target)
  154. if self.role and self.role.protected and not request.user.is_god():
  155. request.messages.set_flash(Message(_('Role "%(name)s" is protected, you cannot edit it.') % {'name': _(self.role.name)}), 'error', self.admin.id)
  156. return redirect(reverse('admin_roles'))
  157. except Role.DoesNotExist:
  158. request.messages.set_flash(Message(_('Requested Role could not be found.')), 'error', self.admin.id)
  159. return redirect(reverse('admin_roles'))
  160. self.roles = ForumRole.objects.order_by('name').all()
  161. if not self.roles:
  162. request.messages.set_flash(Message(_('No forum roles are currently set.')), 'error', self.admin.id)
  163. return redirect(reverse('admin_roles'))
  164. return super(Forums, self).__call__(request)
  165. class ACL(FormWidget):
  166. admin = site.get_action('roles')
  167. id = 'acl'
  168. name = _("Change Role Permissions")
  169. fallback = 'admin_roles'
  170. target_name = 'name'
  171. translate_target_name = True
  172. notfound_message = _('Requested Role could not be found.')
  173. submit_fallback = True
  174. template = 'acl_form'
  175. def get_form(self, target):
  176. self.form = build_form(self.request, target)
  177. return self.form
  178. def get_url(self, model):
  179. return reverse('admin_roles_acl', model)
  180. def get_edit_url(self, model):
  181. return self.get_url(model)
  182. def get_initial_data(self, model):
  183. raw_acl = model.permissions
  184. initial = {}
  185. for field in self.form.base_fields:
  186. if field in raw_acl:
  187. initial[field] = raw_acl[field]
  188. return initial
  189. def get_and_validate_target(self, target):
  190. result = super(ACL, self).get_and_validate_target(target)
  191. if result and result.protected and not self.request.user.is_god():
  192. self.request.messages.set_flash(Message(_('Role "%(name)s" is protected, you cannot edit it.') % {'name': _(result.name)}), 'error', self.admin.id)
  193. return None
  194. return result
  195. def submit_form(self, form, target):
  196. raw_acl = target.permissions
  197. for perm in form.cleaned_data:
  198. raw_acl[perm] = form.cleaned_data[perm]
  199. target.permissions = raw_acl
  200. target.save(force_update=True)
  201. self.request.monitor['acl_version'] = int(self.request.monitor['acl_version']) + 1
  202. return target, Message(_('Role "%(name)s" permissions have been changed.') % {'name': self.original_name}, 'success')
  203. class Delete(ButtonWidget):
  204. admin = site.get_action('roles')
  205. id = 'delete'
  206. fallback = 'admin_roles'
  207. notfound_message = _('Requested Role could not be found.')
  208. def action(self, target):
  209. if target.token:
  210. return Message(_('You cannot delete system roles.'), 'error'), reverse('admin_roles')
  211. if target.protected and not self.request.user.is_god():
  212. return Message(_('This role is protected.'), 'error'), reverse('admin_roles')
  213. if target.user_set.count() > 0:
  214. return Message(_('This role is assigned to one or more users.'), 'error'), reverse('admin_roles')
  215. target.delete()
  216. return Message(_('Role "%(name)s" has been deleted.') % {'name': _(target.name)}, 'success'), False