views.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. from django.core.urlresolvers import reverse as django_reverse
  2. from django.db.models import Q
  3. from django.shortcuts import redirect
  4. from django.utils.translation import ugettext as _
  5. from misago import messages
  6. from misago.admin import site
  7. from misago.apps.admin.widgets import *
  8. from misago.conf import settings
  9. from misago.markdown import signature_markdown
  10. from misago.models import Forum, User
  11. from misago.monitor import monitor, UpdatingMonitor
  12. from misago.utils.strings import random_string
  13. from misago.apps.admin.users.forms import UserForm, NewUserForm, SearchUsersForm
  14. def reverse(route, target=None):
  15. if target:
  16. return django_reverse(route, kwargs={'target': target.pk, 'slug': target.username_slug})
  17. return django_reverse(route)
  18. """
  19. Views
  20. """
  21. class List(ListWidget):
  22. admin = site.get_action('users')
  23. id = 'list'
  24. columns = (
  25. ('username_slug', _("User Name"), 35),
  26. ('join_date', _("Join Date")),
  27. )
  28. default_sorting = 'username'
  29. sortables = {
  30. 'username_slug': 1,
  31. 'join_date': 0,
  32. }
  33. pagination = 25
  34. search_form = SearchUsersForm
  35. nothing_checked_message = _('You have to check at least one user.')
  36. actions = (
  37. ('activate', _("Activate users"), _("Are you sure you want to activate selected members?")),
  38. ('deactivate', _("Request e-mail validation"), _("Are you sure you want to deactivate selected members and request them to revalidate their e-mail addresses?")),
  39. ('remove_av', _("Remove and lock avatars"), _("Are you sure you want to remove selected members avatars and their ability to change them?")),
  40. ('remove_sig', _("Remove and lock signatures"), _("Are you sure you want to remove selected members signatures and their ability to edit them?")),
  41. ('remove_locks', _("Remove locks from avatars and signatures"), _("Are you sure you want to remove locks from selected members avatars and signatures?")),
  42. ('reset', _("Reset passwords"), _("Are you sure you want to reset selected members passwords?")),
  43. ('delete_content', _("Delete users with content"), _("Are you sure you want to delete selected users and their content?")),
  44. ('delete', _("Delete users"), _("Are you sure you want to delete selected users?")),
  45. )
  46. def set_filters(self, model, filters):
  47. if 'role' in filters:
  48. model = model.filter(roles__in=filters['role']).distinct()
  49. if 'rank' in filters:
  50. model = model.filter(rank__in=filters['rank'])
  51. if 'username' in filters:
  52. if ',' in filters['username']:
  53. qs = None
  54. for name in filters['username'].split(','):
  55. name = name.strip().lower()
  56. if name:
  57. if qs:
  58. qs = qs | Q(username_slug__contains=name)
  59. else:
  60. qs = Q(username_slug__contains=name)
  61. if qs:
  62. model = model.filter(qs)
  63. else:
  64. model = model.filter(username_slug__contains=filters['username'].lower())
  65. if 'email' in filters:
  66. if ',' in filters['email']:
  67. qs = None
  68. for name in filters['email'].split(','):
  69. name = name.strip().lower()
  70. if name:
  71. if qs:
  72. qs = qs | Q(email__contains=name)
  73. else:
  74. qs = Q(email__contains=name)
  75. if qs:
  76. model = model.filter(qs)
  77. else:
  78. model = model.filter(email__contains=filters['email'])
  79. if 'activation' in filters:
  80. model = model.filter(activation__in=filters['activation'])
  81. return model
  82. def prefetch_related(self, items):
  83. return items.prefetch_related('roles')
  84. def get_item_actions(self, item):
  85. return (
  86. self.action('pencil', _("Edit User Details"), reverse('admin_users_edit', item)),
  87. self.action('remove', _("Delete User"), reverse('admin_users_delete', item), post=True, prompt=_("Are you sure you want to delete this user account?")),
  88. )
  89. def action_activate(self, items, checked):
  90. for user in items:
  91. if user.pk in checked and user.activation > 0:
  92. with UpdatingMonitor() as cm:
  93. monitor.decrease('users_inactive')
  94. user.activation = user.ACTIVATION_NONE
  95. user.save(force_update=True)
  96. user.email_user(
  97. self.request,
  98. 'users/activation/admin_done',
  99. _("Your Account has been activated"),
  100. )
  101. return Message(_('Selected users accounts have been activated.'), messages.SUCCESS), reverse('admin_users')
  102. def action_deactivate(self, items, checked):
  103. # First loop - check for errors
  104. for user in items:
  105. if user.pk in checked:
  106. if user.is_protected() and not self.request.user.is_god():
  107. return Message(_('You cannot force validation of protected members e-mails.'), messages.ERROR), reverse('admin_users')
  108. # Second loop - reset passwords
  109. for user in items:
  110. if user.pk in checked:
  111. user.activation = user.ACTIVATION_USER
  112. user.token = token = random_string(12)
  113. user.save(force_update=True)
  114. user.email_user(
  115. self.request,
  116. 'users/activation/invalidated',
  117. _("Account Activation"),
  118. )
  119. return Message(_('Selected users accounts have been deactivated and new activation links have been sent to them.'), messages.SUCCESS), reverse('admin_users')
  120. def action_remove_av(self, items, checked):
  121. # First loop - check for errors
  122. for user in items:
  123. if user.pk in checked:
  124. if user.is_protected() and not self.request.user.is_god():
  125. return Message(_('You cannot remove and block protected members avatars.'), messages.ERROR), reverse('admin_users')
  126. # Second loop - reset passwords
  127. for user in items:
  128. if user.pk in checked:
  129. user.lock_avatar()
  130. user.save(force_update=True)
  131. return Message(_('Selected users avatars were deleted and locked.'), messages.SUCCESS), reverse('admin_users')
  132. def action_remove_sig(self, items, checked):
  133. # First loop - check for errors
  134. for user in items:
  135. if user.pk in checked:
  136. if user.is_protected() and not self.request.user.is_god():
  137. return Message(_('You cannot remove and block protected members signatures.'), messages.ERROR), reverse('admin_users')
  138. # Second loop - reset passwords
  139. for user in items:
  140. if user.pk in checked:
  141. user.signature_ban = True
  142. user.signature = ''
  143. user.signature_preparsed = ''
  144. user.save(force_update=True)
  145. return Message(_('Selected users signatures were deleted and locked.'), messages.SUCCESS), reverse('admin_users')
  146. def action_remove_locks(self, items, checked):
  147. for user in items:
  148. if user.pk in checked:
  149. user.default_avatar()
  150. user.avatar_ban = False
  151. user.signature_ban = False
  152. user.save(force_update=True)
  153. return Message(_('Selected users can now edit their avatars and signatures.'), messages.SUCCESS), reverse('admin_users')
  154. def action_reset(self, items, checked):
  155. # First loop - check for errors
  156. for user in items:
  157. if user.pk in checked:
  158. if user.is_protected() and not self.request.user.is_god():
  159. return Message(_('You cannot reset protected members passwords.'), messages.ERROR), reverse('admin_users')
  160. # Second loop - reset passwords
  161. for user in items:
  162. if user.pk in checked:
  163. new_password = random_string(8)
  164. user.set_password(new_password)
  165. user.save(force_update=True)
  166. user.email_user(
  167. self.request,
  168. 'users/password/new_admin',
  169. _("Your New Password"),
  170. {
  171. 'password': new_password,
  172. },
  173. )
  174. return Message(_('Selected users passwords have been reset successfully.'), messages.SUCCESS), reverse('admin_users')
  175. def action_delete_content(self, items, checked):
  176. for user in items:
  177. if user.pk in checked:
  178. if user.pk == self.request.user.id:
  179. return Message(_('You cannot delete yourself.'), messages.ERROR), reverse('admin_users')
  180. if user.is_protected():
  181. return Message(_('You cannot delete protected members.'), messages.ERROR), reverse('admin_users')
  182. for user in items:
  183. if user.pk in checked:
  184. user.delete_content()
  185. user.delete()
  186. for forum in Forum.objects.all():
  187. forum.sync()
  188. forum.save(force_update=True)
  189. User.objects.resync_monitor()
  190. return Message(_('Selected users and their content have been deleted successfully.'), messages.SUCCESS), reverse('admin_users')
  191. def action_delete(self, items, checked):
  192. for user in items:
  193. if user.pk in checked:
  194. if user.pk == self.request.user.id:
  195. return Message(_('You cannot delete yourself.'), messages.ERROR), reverse('admin_users')
  196. if user.is_protected():
  197. return Message(_('You cannot delete protected members.'), messages.ERROR), reverse('admin_users')
  198. for user in items:
  199. if user.pk in checked:
  200. user.delete()
  201. User.objects.resync_monitor()
  202. return Message(_('Selected users have been deleted successfully.'), messages.SUCCESS), reverse('admin_users')
  203. class New(FormWidget):
  204. admin = site.get_action('users')
  205. id = 'new'
  206. fallback = 'admin_users'
  207. form = NewUserForm
  208. template = 'new'
  209. submit_button = _("Save User")
  210. def get_new_link(self, model):
  211. return reverse('admin_users_new')
  212. def get_edit_link(self, model):
  213. return reverse('admin_users_edit', model)
  214. def submit_form(self, form, target):
  215. new_user = User.objects.create_user(
  216. form.cleaned_data['username'],
  217. form.cleaned_data['email'],
  218. form.cleaned_data['password'],
  219. settings.default_timezone,
  220. self.request.META['REMOTE_ADDR'],
  221. no_roles=True,
  222. request=self.request,
  223. )
  224. new_user.title = form.cleaned_data['title']
  225. new_user.rank = form.cleaned_data['rank']
  226. for role in form.cleaned_data['roles']:
  227. new_user.roles.add(role)
  228. new_user.make_acl_key(True)
  229. new_user.save(force_update=True)
  230. return new_user, Message(_('New User has been created.'), messages.SUCCESS)
  231. class Edit(FormWidget):
  232. admin = site.get_action('users')
  233. id = 'edit'
  234. name = _("Edit User")
  235. fallback = 'admin_users'
  236. form = UserForm
  237. target_name = 'username'
  238. notfound_message = _('Requested User could not be found.')
  239. submit_fallback = True
  240. def get_form_instance(self, form, model, initial, post=False):
  241. if post:
  242. return form(model, self.request.POST, request=self.request, initial=self.get_initial_data(model))
  243. return form(model, request=self.request, initial=self.get_initial_data(model))
  244. def get_link(self, model):
  245. return reverse('admin_users_edit', model)
  246. def get_edit_link(self, model):
  247. return self.get_link(model)
  248. def get_initial_data(self, model):
  249. return {
  250. 'username': model.username,
  251. 'title': model.title,
  252. 'email': model.email,
  253. 'rank': model.rank,
  254. 'roles': model.roles.all(),
  255. 'avatar_ban': model.avatar_ban,
  256. 'avatar_ban_reason_user': model.avatar_ban_reason_user,
  257. 'avatar_ban_reason_admin': model.avatar_ban_reason_admin,
  258. 'signature': model.signature,
  259. 'signature_ban': model.signature_ban,
  260. 'signature_ban_reason_user': model.signature_ban_reason_user,
  261. 'signature_ban_reason_admin': model.signature_ban_reason_admin,
  262. }
  263. def submit_form(self, form, target):
  264. target.title = form.cleaned_data['title']
  265. target.rank = form.cleaned_data['rank']
  266. target.avatar_ban_reason_user = form.cleaned_data['avatar_ban_reason_user']
  267. target.avatar_ban_reason_admin = form.cleaned_data['avatar_ban_reason_admin']
  268. target.signature_ban = form.cleaned_data['signature_ban']
  269. target.signature_ban_reason_user = form.cleaned_data['signature_ban_reason_user']
  270. target.signature_ban_reason_admin = form.cleaned_data['signature_ban_reason_admin']
  271. # Sync username?
  272. if target.username != self.original_name:
  273. target.sync_username()
  274. # Change email?
  275. if form.cleaned_data.get('email'):
  276. target.set_email(form.cleaned_data.get('email'))
  277. # Change password?
  278. if form.cleaned_data.get('new_password'):
  279. target.set_password(form.cleaned_data.get('new_password'))
  280. # Do signature mumbo-jumbo
  281. if form.cleaned_data['signature']:
  282. target.signature = form.cleaned_data['signature']
  283. target.signature_preparsed = signature_markdown(target.acl(),
  284. form.cleaned_data['signature'])
  285. else:
  286. target.signature = None
  287. target.signature_preparsed = None
  288. # Do avatar ban mumbo-jumbo
  289. if target.avatar_ban != form.cleaned_data['avatar_ban']:
  290. if form.cleaned_data['avatar_ban']:
  291. target.lock_avatar()
  292. else:
  293. target.default_avatar()
  294. target.avatar_ban = form.cleaned_data['avatar_ban']
  295. # Set custom avatar
  296. if form.cleaned_data['avatar_custom']:
  297. target.delete_avatar()
  298. target.avatar_image = form.cleaned_data['avatar_custom']
  299. target.avatar_type = 'gallery'
  300. # Update user roles
  301. if self.request.user.is_god():
  302. target.roles.clear()
  303. else:
  304. target.roles.remove(*target.roles.filter(protected=False))
  305. for role in form.cleaned_data['roles']:
  306. target.roles.add(role)
  307. target.make_acl_key(True)
  308. target.save(force_update=True)
  309. return target, Message(_('Changes in user\'s "%(name)s" account have been saved.') % {'name': self.original_name}, messages.SUCCESS)
  310. class Delete(ButtonWidget):
  311. admin = site.get_action('users')
  312. id = 'delete'
  313. fallback = 'admin_users'
  314. notfound_message = _('Requested User account could not be found.')
  315. def action(self, target):
  316. if target.pk == self.request.user.id:
  317. return Message(_('You cannot delete yourself.'), messages.ERROR), False
  318. if target.is_protected():
  319. return Message(_('You cannot delete protected member.'), messages.ERROR), False
  320. target.delete()
  321. User.objects.resync_monitor()
  322. return Message(_('User "%(name)s" has been deleted.') % {'name': target.username}, messages.SUCCESS), False
  323. def inactive(request):
  324. token = 'list_filter_users.User'
  325. request.session[token] = {'activation': [1, 2, 3]}
  326. return redirect(reverse('admin_users'))