PageRenderTime 1948ms CodeModel.GetById 29ms RepoModel.GetById 1ms app.codeStats 0ms

/apps/users/forms.py

https://github.com/oremj/kitsune
Python | 246 lines | 216 code | 19 blank | 11 comment | 5 complexity | 8868c6d28ceccbad17145fbce2889f21 MD5 | raw file
  1. import re
  2. from django import forms
  3. from django.conf import settings
  4. from django.contrib.auth import authenticate, forms as auth_forms
  5. from django.contrib.auth.models import User
  6. from tower import ugettext as _, ugettext_lazy as _lazy
  7. from sumo.widgets import ImageWidget
  8. from upload.forms import clean_image_extension
  9. from upload.utils import check_file_size, FileTooLargeError
  10. from users.models import Profile
  11. from users.passwords import password_allowed
  12. from users.widgets import FacebookURLWidget, TwitterURLWidget
  13. USERNAME_INVALID = _lazy(u'Username may contain only letters, '
  14. 'numbers and ./+/-/_ characters.')
  15. USERNAME_REQUIRED = _lazy(u'Username is required.')
  16. USERNAME_SHORT = _lazy(u'Username is too short (%(show_value)s characters). '
  17. 'It must be at least %(limit_value)s characters.')
  18. USERNAME_LONG = _lazy(u'Username is too long (%(show_value)s characters). '
  19. 'It must be %(limit_value)s characters or less.')
  20. EMAIL_REQUIRED = _lazy(u'Email address is required.')
  21. EMAIL_SHORT = _lazy(u'Email address is too short (%(show_value)s characters). '
  22. 'It must be at least %(limit_value)s characters.')
  23. EMAIL_LONG = _lazy(u'Email address is too long (%(show_value)s characters). '
  24. 'It must be %(limit_value)s characters or less.')
  25. PASSWD_REQUIRED = _lazy(u'Password is required.')
  26. PASSWD2_REQUIRED = _lazy(u'Please enter your password twice.')
  27. class SettingsForm(forms.Form):
  28. forums_watch_new_thread = forms.BooleanField(
  29. required=False, initial=True,
  30. label=_lazy(u'Watch forum threads I start'))
  31. forums_watch_after_reply = forms.BooleanField(
  32. required=False, initial=True,
  33. label=_lazy(u'Watch forum threads I comment in'))
  34. kbforums_watch_new_thread = forms.BooleanField(
  35. required=False, initial=True,
  36. label=_lazy(u'Watch KB discussion threads I start'))
  37. kbforums_watch_after_reply = forms.BooleanField(
  38. required=False, initial=True,
  39. label=_lazy(u'Watch KB discussion threads I comment in'))
  40. def save_for_user(self, user):
  41. for field in self.fields.keys():
  42. value = str(self.cleaned_data[field])
  43. setting = user.settings.filter(name=field)
  44. update_count = setting.update(value=value)
  45. if update_count == 0:
  46. # This user didn't have this setting so create it.
  47. user.settings.create(name=field, value=value)
  48. class RegisterForm(forms.ModelForm):
  49. """A user registration form that requires unique email addresses.
  50. The default Django user creation form does not require an email address,
  51. let alone that it be unique. This form does, and sets a minimum length
  52. for usernames.
  53. """
  54. username = forms.RegexField(
  55. label=_lazy(u'Username:'), max_length=30, min_length=4,
  56. regex=r'^[\w.+-]+$',
  57. help_text=_lazy(u'Required. 30 characters or fewer. Letters, digits '
  58. 'and ./+/-/_ only.'),
  59. error_messages={'invalid': USERNAME_INVALID,
  60. 'required': USERNAME_REQUIRED,
  61. 'min_length': USERNAME_SHORT,
  62. 'max_length': USERNAME_LONG})
  63. email = forms.EmailField(label=_lazy(u'Email address:'),
  64. error_messages={'required': EMAIL_REQUIRED,
  65. 'min_length': EMAIL_SHORT,
  66. 'max_length': EMAIL_LONG})
  67. password = forms.CharField(label=_lazy(u'Password:'),
  68. widget=forms.PasswordInput(
  69. render_value=False),
  70. error_messages={'required': PASSWD_REQUIRED})
  71. password2 = forms.CharField(label=_lazy(u'Repeat password:'),
  72. widget=forms.PasswordInput(
  73. render_value=False),
  74. error_messages={'required': PASSWD2_REQUIRED},
  75. help_text=_lazy(u'Enter the same password as '
  76. 'above, for verification.'))
  77. class Meta(object):
  78. model = User
  79. fields = ('username', 'password', 'password2', 'email')
  80. def clean(self):
  81. super(RegisterForm, self).clean()
  82. password = self.cleaned_data.get('password')
  83. password2 = self.cleaned_data.get('password2')
  84. if not password == password2:
  85. raise forms.ValidationError(_('Passwords must match.'))
  86. _check_password(password)
  87. return self.cleaned_data
  88. def clean_email(self):
  89. email = self.cleaned_data['email']
  90. if User.objects.filter(email=email).exists():
  91. raise forms.ValidationError(_('A user with that email address '
  92. 'already exists.'))
  93. return email
  94. def __init__(self, request=None, *args, **kwargs):
  95. super(RegisterForm, self).__init__(request, auto_id='id_for_%s',
  96. *args, **kwargs)
  97. class AuthenticationForm(auth_forms.AuthenticationForm):
  98. """Overrides the default django form.
  99. * Doesn't prefill password on validation error.
  100. * Allows logging in inactive users (initialize with `only_active=False`).
  101. """
  102. password = forms.CharField(label=_lazy(u"Password"),
  103. widget=forms.PasswordInput(render_value=False))
  104. def __init__(self, request=None, only_active=True, *args, **kwargs):
  105. self.only_active = only_active
  106. super(AuthenticationForm, self).__init__(request, *args, **kwargs)
  107. def clean(self):
  108. username = self.cleaned_data.get('username')
  109. password = self.cleaned_data.get('password')
  110. if username and password:
  111. self.user_cache = authenticate(username=username,
  112. password=password)
  113. if self.user_cache is None:
  114. raise forms.ValidationError(
  115. _('Please enter a correct username and password. Note '
  116. 'that both fields are case-sensitive.'))
  117. elif self.only_active and not self.user_cache.is_active:
  118. raise forms.ValidationError(_('This account is inactive.'))
  119. if self.request:
  120. if not self.request.session.test_cookie_worked():
  121. raise forms.ValidationError(
  122. _("Your Web browser doesn't appear to have cookies "
  123. "enabled. Cookies are required for logging in."))
  124. return self.cleaned_data
  125. class ProfileForm(forms.ModelForm):
  126. """The form for editing the user's profile."""
  127. class Meta(object):
  128. model = Profile
  129. exclude = ('user', 'livechat_id', 'avatar')
  130. widgets = {
  131. 'twitter': TwitterURLWidget,
  132. 'facebook': FacebookURLWidget,
  133. }
  134. def clean_twitter(self):
  135. twitter = self.cleaned_data['twitter']
  136. if twitter and not re.match(TwitterURLWidget.pattern, twitter):
  137. raise forms.ValidationError(_(u'Please enter a twitter.com URL.'))
  138. return twitter
  139. def clean_facebook(self):
  140. facebook = self.cleaned_data['facebook']
  141. if facebook and not re.match(FacebookURLWidget.pattern, facebook):
  142. raise forms.ValidationError(_(u'Please enter a facebook.com URL.'))
  143. return facebook
  144. class AvatarForm(forms.ModelForm):
  145. """The form for editing the user's avatar."""
  146. avatar = forms.ImageField(required=True, widget=ImageWidget)
  147. def __init__(self, *args, **kwargs):
  148. super(AvatarForm, self).__init__(*args, **kwargs)
  149. self.fields['avatar'].help_text = (
  150. u'Your avatar will be resized to {size}x{size}'.format(
  151. size=settings.AVATAR_SIZE))
  152. class Meta(object):
  153. model = Profile
  154. fields = ('avatar',)
  155. def clean_avatar(self):
  156. if not ('avatar' in self.cleaned_data and self.cleaned_data['avatar']):
  157. return self.cleaned_data['avatar']
  158. try:
  159. check_file_size(self.cleaned_data['avatar'],
  160. settings.MAX_AVATAR_FILE_SIZE)
  161. except FileTooLargeError as e:
  162. raise forms.ValidationError(e.args[0])
  163. clean_image_extension(self.cleaned_data.get('avatar'))
  164. return self.cleaned_data['avatar']
  165. class EmailConfirmationForm(forms.Form):
  166. """A simple form that requires an email address."""
  167. email = forms.EmailField(label=_lazy(u'Email address:'))
  168. class EmailChangeForm(forms.Form):
  169. """A simple form that requires an email address and validates that it is
  170. not the current user's email."""
  171. email = forms.EmailField(label=_lazy(u'Email address:'))
  172. def __init__(self, user, *args, **kwargs):
  173. super(EmailChangeForm, self).__init__(*args, **kwargs)
  174. self.user = user
  175. def clean_email(self):
  176. email = self.cleaned_data['email']
  177. if self.user.email == email:
  178. raise forms.ValidationError(_('This is your current email.'))
  179. if User.objects.filter(email=email).exists():
  180. raise forms.ValidationError(_('A user with that email address '
  181. 'already exists.'))
  182. return self.cleaned_data['email']
  183. class SetPasswordForm(auth_forms.SetPasswordForm):
  184. def clean(self):
  185. super(SetPasswordForm, self).clean()
  186. _check_password(self.cleaned_data.get('new_password1'))
  187. return self.cleaned_data
  188. class PasswordChangeForm(auth_forms.PasswordChangeForm):
  189. def clean(self):
  190. super(PasswordChangeForm, self).clean()
  191. _check_password(self.cleaned_data.get('new_password1'))
  192. return self.cleaned_data
  193. def _check_password(password):
  194. if not password_allowed(password):
  195. msg = _('The password entered is known to be commonly used and '
  196. 'is not allowed.')
  197. raise forms.ValidationError(msg)