changepassword.py 3.4 KB

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