NDOJ/judge/views/register.py
2022-05-30 01:59:53 -05:00

157 lines
5.6 KiB
Python

# coding=utf-8
import re
from django import forms
from django.conf import settings
from django.contrib.auth.models import User
from django.contrib.auth.password_validation import get_default_password_validators
from django.forms import ChoiceField, ModelChoiceField
from django.shortcuts import render
from django.utils.translation import gettext, gettext_lazy as _
from registration.backends.default.views import (
ActivationView as OldActivationView,
RegistrationView as OldRegistrationView,
)
from registration.forms import RegistrationForm
from sortedm2m.forms import SortedMultipleChoiceField
from judge.models import Language, Organization, Profile, TIMEZONE
from judge.utils.recaptcha import ReCaptchaField, ReCaptchaWidget
from judge.utils.subscription import Subscription, newsletter_id
from judge.widgets import Select2MultipleWidget, Select2Widget
valid_id = re.compile(r"^\w+$")
bad_mail_regex = list(map(re.compile, settings.BAD_MAIL_PROVIDER_REGEX))
class CustomRegistrationForm(RegistrationForm):
username = forms.RegexField(
regex=r"^\w+$",
max_length=30,
label=_("Username"),
error_messages={
"invalid": _("A username must contain letters, " "numbers, or underscores")
},
)
timezone = ChoiceField(
label=_("Timezone"),
choices=TIMEZONE,
widget=Select2Widget(attrs={"style": "width:100%"}),
)
language = ModelChoiceField(
queryset=Language.objects.all(),
label=_("Preferred language"),
empty_label=None,
widget=Select2Widget(attrs={"style": "width:100%"}),
)
organizations = SortedMultipleChoiceField(
queryset=Organization.objects.filter(is_open=True),
label=_("Groups"),
required=False,
widget=Select2MultipleWidget(attrs={"style": "width:100%"}),
)
if newsletter_id is not None:
newsletter = forms.BooleanField(
label=_("Subscribe to newsletter?"), initial=True, required=False
)
if ReCaptchaField is not None:
captcha = ReCaptchaField(widget=ReCaptchaWidget())
def clean_organizations(self):
organizations = self.cleaned_data.get("organizations") or []
max_orgs = settings.DMOJ_USER_MAX_ORGANIZATION_COUNT
if sum(org.is_open for org in organizations) > max_orgs:
raise forms.ValidationError(
_("You may not be part of more than {count} public groups.").format(
count=max_orgs
)
)
return self.cleaned_data["organizations"]
def clean_email(self):
if User.objects.filter(email=self.cleaned_data["email"]).exists():
raise forms.ValidationError(
gettext(
'The email address "%s" is already taken. Only one registration '
"is allowed per address."
)
% self.cleaned_data["email"]
)
if "@" in self.cleaned_data["email"]:
domain = self.cleaned_data["email"].split("@")[-1].lower()
if domain in settings.BAD_MAIL_PROVIDERS or any(
regex.match(domain) for regex in bad_mail_regex
):
raise forms.ValidationError(
gettext(
"Your email provider is not allowed due to history of abuse. "
"Please use a reputable email provider."
)
)
return self.cleaned_data["email"]
class RegistrationView(OldRegistrationView):
title = _("Registration")
form_class = CustomRegistrationForm
template_name = "registration/registration_form.html"
def get_context_data(self, **kwargs):
if "title" not in kwargs:
kwargs["title"] = self.title
tzmap = settings.TIMEZONE_MAP
kwargs["TIMEZONE_MAP"] = tzmap or "http://momentjs.com/static/img/world.png"
kwargs["TIMEZONE_BG"] = settings.TIMEZONE_BG if tzmap else "#4E7CAD"
kwargs["password_validators"] = get_default_password_validators()
kwargs["tos_url"] = settings.TERMS_OF_SERVICE_URL
return super(RegistrationView, self).get_context_data(**kwargs)
def register(self, form):
user = super(RegistrationView, self).register(form)
profile, _ = Profile.objects.get_or_create(
user=user,
defaults={
"language": Language.get_default_language(),
},
)
cleaned_data = form.cleaned_data
profile.timezone = cleaned_data["timezone"]
profile.language = cleaned_data["language"]
profile.organizations.add(*cleaned_data["organizations"])
profile.save()
if newsletter_id is not None and cleaned_data["newsletter"]:
Subscription(user=user, newsletter_id=newsletter_id, subscribed=True).save()
return user
def get_initial(self, *args, **kwargs):
initial = super(RegistrationView, self).get_initial(*args, **kwargs)
initial["timezone"] = settings.DEFAULT_USER_TIME_ZONE
initial["language"] = Language.objects.get(key=settings.DEFAULT_USER_LANGUAGE)
return initial
class ActivationView(OldActivationView):
title = _("Registration")
template_name = "registration/activate.html"
def get_context_data(self, **kwargs):
if "title" not in kwargs:
kwargs["title"] = self.title
return super(ActivationView, self).get_context_data(**kwargs)
def social_auth_error(request):
return render(
request,
"generic-message.html",
{
"title": gettext("Authentication failure"),
"message": request.GET.get("message"),
},
)