changepassword.py 3.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. from django.contrib.auth import get_user_model
  2. from django.core.exceptions import ValidationError
  3. from django.core.urlresolvers import reverse
  4. from django.shortcuts import get_object_or_404
  5. from django.utils.translation import ugettext as _
  6. from django.views.decorators.csrf import csrf_protect
  7. from rest_framework import status
  8. from rest_framework.decorators import api_view, permission_classes
  9. from rest_framework.response import Response
  10. from misago.conf import settings
  11. from misago.core.mail import mail_user
  12. from misago.users.forms.auth import ResetPasswordForm
  13. from misago.users.rest_permissions import UnbannedAnonOnly
  14. from misago.users.tokens import (make_password_change_token,
  15. is_password_change_token_valid)
  16. from misago.users.validators import validate_password
  17. def password_api_view(f):
  18. @api_view(['POST'])
  19. @permission_classes((UnbannedAnonOnly,))
  20. @csrf_protect
  21. def decorator(request, *args, **kwargs):
  22. if 'user_id' in kwargs:
  23. User = get_user_model()
  24. user = get_object_or_404(User.objects, pk=kwargs.pop('user_id'))
  25. kwargs['user'] = user
  26. if not is_password_change_token_valid(user, kwargs['token']):
  27. message = _("Your link is invalid. Please try again.")
  28. return Response({'detail': message},
  29. status=status.HTTP_404_NOT_FOUND)
  30. try:
  31. form = ResetPasswordForm()
  32. form.confirm_allowed(user)
  33. except ValidationError:
  34. message = _("Your link has expired. Please request new one.")
  35. return Response({'detail': message},
  36. status=status.HTTP_404_NOT_FOUND)
  37. return f(request, *args, **kwargs)
  38. return decorator
  39. @password_api_view
  40. def send_link(request):
  41. form = ResetPasswordForm(request.DATA)
  42. if form.is_valid():
  43. requesting_user = form.user_cache
  44. mail_subject = _("Change %(user)s password "
  45. "on %(forum_title)s forums")
  46. subject_formats = {'user': requesting_user.username,
  47. 'forum_title': settings.forum_name}
  48. mail_subject = mail_subject % subject_formats
  49. confirmation_token = make_password_change_token(requesting_user)
  50. mail_user(request, requesting_user, mail_subject,
  51. 'misago/emails/change_password_form_link',
  52. {'confirmation_token': confirmation_token})
  53. return Response({
  54. 'username': form.user_cache.username,
  55. 'email': form.user_cache.email
  56. })
  57. else:
  58. return Response(form.get_errors_dict(),
  59. status=status.HTTP_400_BAD_REQUEST)
  60. @password_api_view
  61. def validate_token(request, user, token):
  62. return Response({
  63. 'user_id': user.id,
  64. 'token': token,
  65. 'username': user.username
  66. })
  67. @password_api_view
  68. def change_password(request, user, token):
  69. new_password = request.DATA.get('password', '').strip()
  70. try:
  71. validate_password(new_password)
  72. user.set_password(new_password)
  73. user.save()
  74. except ValidationError as e:
  75. return Response({'detail': e.messages[0]},
  76. status=status.HTTP_400_BAD_REQUEST)
  77. return Response({'detail': 'ok'})