forms.py
245 lines
| 7.1 KiB
| text/x-python
|
PythonLexer
/ boards / forms.py
neko259
|
r69 | import re | |
Ilyas
|
r78 | from captcha.fields import CaptchaField | |
Ilyas
|
r14 | from django import forms | |
neko259
|
r76 | from django.forms.util import ErrorList | |
neko259
|
r205 | from django.utils.translation import ugettext_lazy as _ | |
neko259
|
r153 | import time | |
neko259
|
r144 | from boards.models import TITLE_MAX_LENGTH, User | |
neko259
|
r35 | from neboard import settings | |
wnc_21
|
r95 | from boards import utils | |
neko259
|
r76 | ||
neko259
|
r153 | LAST_POST_TIME = "last_post_time" | |
neko259
|
r211 | LAST_LOGIN_TIME = "last_login_time" | |
LOGIN_DELAY = 60 * 60 | |||
MAX_TEXT_LENGTH = 30000 | |||
MAX_IMAGE_SIZE = 8 * 1024 * 1024 | |||
neko259
|
r153 | ||
neko259
|
r76 | class PlainErrorList(ErrorList): | |
def __unicode__(self): | |||
return self.as_text() | |||
def as_text(self): | |||
return ''.join([u'(!) %s ' % e for e in self]) | |||
neko259
|
r16 | ||
neko259
|
r205 | class NeboardForm(forms.Form): | |
def as_p(self): | |||
"Returns this form rendered as HTML <p>s." | |||
return self._html_output( | |||
normal_row='<div class="form-row">' | |||
'<div class="form-label">' | |||
'%(label)s' | |||
'</div>' | |||
'<div class="form-input">' | |||
'%(field)s' | |||
'</div>' | |||
'%(help_text)s' | |||
'</div>', | |||
error_row='<div class="form-errors">%s</div>', | |||
row_ender='</p>', | |||
help_text_html=' <span class="helptext">%s</span>', | |||
errors_on_separate_row=True) | |||
class PostForm(NeboardForm): | |||
neko259
|
r76 | ||
title = forms.CharField(max_length=TITLE_MAX_LENGTH, required=False) | |||
neko259
|
r29 | text = forms.CharField(widget=forms.Textarea, required=False) | |
Ilyas
|
r18 | image = forms.ImageField(required=False) | |
neko259
|
r29 | ||
neko259
|
r207 | # This field is for spam prevention only | |
email = forms.CharField(max_length=100, required=False) | |||
neko259
|
r153 | session = None | |
neko259
|
r76 | def clean_title(self): | |
title = self.cleaned_data['title'] | |||
if title: | |||
if len(title) > TITLE_MAX_LENGTH: | |||
neko259
|
r211 | raise forms.ValidationError(_('Title must have less than %s ' | |
'characters') % | |||
str(TITLE_MAX_LENGTH)) | |||
neko259
|
r76 | return title | |
neko259
|
r29 | def clean_text(self): | |
text = self.cleaned_data['text'] | |||
if text: | |||
neko259
|
r211 | if len(text) > MAX_TEXT_LENGTH: | |
raise forms.ValidationError(_('Text must have less than %s ' | |||
'characters') % | |||
str(MAX_TEXT_LENGTH)) | |||
neko259
|
r29 | return text | |
def clean_image(self): | |||
image = self.cleaned_data['image'] | |||
if image: | |||
neko259
|
r211 | if image._size > MAX_IMAGE_SIZE: | |
raise forms.ValidationError(_('Image must be less than %s ' | |||
'bytes') % str(MAX_IMAGE_SIZE)) | |||
neko259
|
r29 | return image | |
def clean(self): | |||
cleaned_data = super(PostForm, self).clean() | |||
neko259
|
r211 | if not self.session: | |
raise forms.ValidationError('Humans have sessions') | |||
if cleaned_data['email']: | |||
raise forms.ValidationError('A human cannot enter a hidden field') | |||
neko259
|
r207 | ||
neko259
|
r151 | if not self.errors: | |
self._clean_text_image() | |||
neko259
|
r77 | ||
neko259
|
r153 | if not self.errors and self.session: | |
self._validate_posting_speed() | |||
neko259
|
r76 | return cleaned_data | |
def _clean_text_image(self): | |||
text = self.cleaned_data.get('text') | |||
image = self.cleaned_data.get('image') | |||
neko259
|
r29 | ||
if (not text) and (not image): | |||
neko259
|
r205 | error_message = _('Either text or image must be entered.') | |
neko259
|
r77 | self._errors['text'] = self.error_class([error_message]) | |
neko259
|
r153 | ||
def _validate_posting_speed(self): | |||
can_post = True | |||
if LAST_POST_TIME in self.session: | |||
now = time.time() | |||
last_post_time = self.session[LAST_POST_TIME] | |||
current_delay = int(now - last_post_time) | |||
if current_delay < settings.POSTING_DELAY: | |||
neko259
|
r211 | error_message = _('Wait %s seconds after last posting') % str( | |
settings.POSTING_DELAY - current_delay) | |||
neko259
|
r153 | self._errors['text'] = self.error_class([error_message]) | |
can_post = False | |||
if can_post: | |||
self.session[LAST_POST_TIME] = time.time() | |||
neko259
|
r29 | ||
class ThreadForm(PostForm): | |||
neko259
|
r71 | regex_tags = re.compile(ur'^[\w\s\d]+$', re.UNICODE) | |
Ilyas
|
r14 | tags = forms.CharField(max_length=100) | |
neko259
|
r31 | ||
def clean_tags(self): | |||
tags = self.cleaned_data['tags'] | |||
neko259
|
r69 | ||
neko259
|
r31 | if tags: | |
neko259
|
r69 | if not self.regex_tags.match(tags): | |
raise forms.ValidationError( | |||
neko259
|
r205 | _('Inappropriate characters in tags.')) | |
neko259
|
r31 | ||
return tags | |||
def clean(self): | |||
cleaned_data = super(ThreadForm, self).clean() | |||
neko259
|
r35 | return cleaned_data | |
wnc_21
|
r95 | class PostCaptchaForm(PostForm): | |
captcha = CaptchaField() | |||
def __init__(self, *args, **kwargs): | |||
self.request = kwargs['request'] | |||
del kwargs['request'] | |||
super(PostCaptchaForm, self).__init__(*args, **kwargs) | |||
def clean(self): | |||
cleaned_data = super(PostCaptchaForm, self).clean() | |||
success = self.is_valid() | |||
utils.update_captcha_access(self.request, success) | |||
if success: | |||
return cleaned_data | |||
else: | |||
neko259
|
r205 | raise forms.ValidationError(_("Captcha validation failed")) | |
wnc_21
|
r95 | ||
Ilyas
|
r78 | class ThreadCaptchaForm(ThreadForm): | |
captcha = CaptchaField() | |||
wnc_21
|
r95 | def __init__(self, *args, **kwargs): | |
self.request = kwargs['request'] | |||
del kwargs['request'] | |||
super(ThreadCaptchaForm, self).__init__(*args, **kwargs) | |||
def clean(self): | |||
cleaned_data = super(ThreadCaptchaForm, self).clean() | |||
success = self.is_valid() | |||
utils.update_captcha_access(self.request, success) | |||
if success: | |||
return cleaned_data | |||
else: | |||
neko259
|
r205 | raise forms.ValidationError(_("Captcha validation failed")) | |
wnc_21
|
r95 | ||
Ilyas
|
r78 | ||
neko259
|
r205 | class SettingsForm(NeboardForm): | |
theme = forms.ChoiceField(choices=settings.THEMES, | |||
label=_('Theme')) | |||
neko259
|
r144 | ||
neko259
|
r205 | class ModeratorSettingsForm(SettingsForm): | |
moderate = forms.BooleanField(required=False, label=_('Enable moderation ' | |||
'panel')) | |||
class LoginForm(NeboardForm): | |||
neko259
|
r211 | ||
neko259
|
r144 | user_id = forms.CharField() | |
neko259
|
r211 | session = None | |
neko259
|
r144 | def clean_user_id(self): | |
user_id = self.cleaned_data['user_id'] | |||
if user_id: | |||
users = User.objects.filter(user_id=user_id) | |||
if len(users) == 0: | |||
neko259
|
r205 | raise forms.ValidationError(_('No such user found')) | |
neko259
|
r144 | ||
return user_id | |||
neko259
|
r211 | def _validate_login_speed(self): | |
can_post = True | |||
if LAST_LOGIN_TIME in self.session: | |||
now = time.time() | |||
last_login_time = self.session[LAST_LOGIN_TIME] | |||
current_delay = int(now - last_login_time) | |||
if current_delay < LOGIN_DELAY: | |||
error_message = _('Wait %s minutes after last login') % str( | |||
(LOGIN_DELAY - current_delay) / 60) | |||
self._errors['user_id'] = self.error_class([error_message]) | |||
can_post = False | |||
if can_post: | |||
self.session[LAST_LOGIN_TIME] = time.time() | |||
neko259
|
r144 | def clean(self): | |
neko259
|
r211 | if not self.session: | |
raise forms.ValidationError('Humans have sessions') | |||
self._validate_login_speed() | |||
neko259
|
r144 | cleaned_data = super(LoginForm, self).clean() | |
neko259
|
r211 | return cleaned_data |