2020-12-27 11:49:54 +01:00
|
|
|
|
# Copyright (C) 2020 by Animath
|
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
|
2022-05-15 12:23:17 +02:00
|
|
|
|
import csv
|
|
|
|
|
from io import StringIO
|
2022-11-08 15:52:44 +01:00
|
|
|
|
import re
|
2022-05-15 12:23:17 +02:00
|
|
|
|
from typing import Iterable
|
2020-12-27 11:49:54 +01:00
|
|
|
|
|
2023-04-05 16:54:16 +02:00
|
|
|
|
from crispy_forms.bootstrap import InlineField
|
|
|
|
|
from crispy_forms.helper import FormHelper
|
2023-04-05 17:52:46 +02:00
|
|
|
|
from crispy_forms.layout import Div, Fieldset, Submit
|
2020-12-27 11:49:54 +01:00
|
|
|
|
from django import forms
|
2022-05-15 12:23:17 +02:00
|
|
|
|
from django.contrib.auth.models import User
|
2020-12-28 18:52:50 +01:00
|
|
|
|
from django.core.exceptions import ValidationError
|
2022-05-15 12:23:17 +02:00
|
|
|
|
from django.core.validators import FileExtensionValidator
|
2023-04-10 17:26:30 +02:00
|
|
|
|
from django.db.models import CharField, Value
|
|
|
|
|
from django.db.models.functions import Concat
|
2020-12-27 11:49:54 +01:00
|
|
|
|
from django.utils.translation import gettext_lazy as _
|
2023-03-29 18:34:55 +02:00
|
|
|
|
from pypdf import PdfReader
|
2022-11-08 15:52:44 +01:00
|
|
|
|
|
2021-01-17 16:23:48 +01:00
|
|
|
|
from .models import Note, Participation, Passage, Pool, Solution, Synthesis, Team, Tournament
|
2020-12-27 11:49:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TeamForm(forms.ModelForm):
|
|
|
|
|
"""
|
2021-01-12 15:42:32 +01:00
|
|
|
|
Form to create a team, with the name and the trigram,...
|
2020-12-27 11:49:54 +01:00
|
|
|
|
"""
|
2021-02-07 17:39:24 +01:00
|
|
|
|
def clean_name(self):
|
|
|
|
|
if "name" in self.cleaned_data:
|
2022-02-04 15:40:45 +01:00
|
|
|
|
name = self.cleaned_data["name"]
|
2023-04-10 09:56:16 +02:00
|
|
|
|
if Team.objects.filter(name=name).exclude(pk=self.instance.pk).exists():
|
2021-02-07 17:39:24 +01:00
|
|
|
|
raise ValidationError(_("This name is already used."))
|
|
|
|
|
return name
|
|
|
|
|
|
2020-12-27 11:49:54 +01:00
|
|
|
|
def clean_trigram(self):
|
2021-02-07 17:39:24 +01:00
|
|
|
|
if "trigram" in self.cleaned_data:
|
|
|
|
|
trigram = self.cleaned_data["trigram"].upper()
|
|
|
|
|
if not re.match("[A-Z]{3}", trigram):
|
|
|
|
|
raise ValidationError(_("The trigram must be composed of three uppercase letters."))
|
|
|
|
|
|
2023-04-10 09:56:16 +02:00
|
|
|
|
if Team.objects.filter(trigram=trigram).exclude(pk=self.instance.pk).exists():
|
2021-02-07 17:39:24 +01:00
|
|
|
|
raise ValidationError(_("This trigram is already used."))
|
|
|
|
|
return trigram
|
2020-12-27 11:49:54 +01:00
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Team
|
2020-12-28 18:52:50 +01:00
|
|
|
|
fields = ('name', 'trigram',)
|
2020-12-27 11:49:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class JoinTeamForm(forms.ModelForm):
|
|
|
|
|
"""
|
|
|
|
|
Form to join a team by the access code.
|
|
|
|
|
"""
|
|
|
|
|
def clean_access_code(self):
|
|
|
|
|
access_code = self.cleaned_data["access_code"]
|
|
|
|
|
if not Team.objects.filter(access_code=access_code).exists():
|
|
|
|
|
raise ValidationError(_("No team was found with this access code."))
|
|
|
|
|
return access_code
|
|
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
|
if "access_code" in cleaned_data:
|
|
|
|
|
team = Team.objects.get(access_code=cleaned_data["access_code"])
|
|
|
|
|
self.instance = team
|
|
|
|
|
return cleaned_data
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Team
|
|
|
|
|
fields = ('access_code',)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ParticipationForm(forms.ModelForm):
|
|
|
|
|
"""
|
|
|
|
|
Form to update the problem of a team participation.
|
|
|
|
|
"""
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Participation
|
2021-04-29 14:10:38 +02:00
|
|
|
|
fields = ('tournament', 'final',)
|
2020-12-27 11:49:54 +01:00
|
|
|
|
|
|
|
|
|
|
2021-01-22 09:40:28 +01:00
|
|
|
|
class MotivationLetterForm(forms.ModelForm):
|
|
|
|
|
def clean_file(self):
|
2021-01-23 14:26:15 +01:00
|
|
|
|
if "motivation_letter" in self.files:
|
2021-01-22 09:40:28 +01:00
|
|
|
|
file = self.files["motivation_letter"]
|
|
|
|
|
if file.size > 2e6:
|
|
|
|
|
raise ValidationError(_("The uploaded file size must be under 2 Mo."))
|
|
|
|
|
if file.content_type not in ["application/pdf", "image/png", "image/jpeg"]:
|
|
|
|
|
raise ValidationError(_("The uploaded file must be a PDF, PNG of JPEG file."))
|
|
|
|
|
return self.cleaned_data["motivation_letter"]
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Team
|
|
|
|
|
fields = ('motivation_letter',)
|
|
|
|
|
|
|
|
|
|
|
2020-12-27 11:49:54 +01:00
|
|
|
|
class RequestValidationForm(forms.Form):
|
|
|
|
|
"""
|
|
|
|
|
Form to ask about validation.
|
|
|
|
|
"""
|
|
|
|
|
_form_type = forms.CharField(
|
|
|
|
|
initial="RequestValidationForm",
|
|
|
|
|
widget=forms.HiddenInput(),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
engagement = forms.BooleanField(
|
2021-01-18 15:52:09 +01:00
|
|
|
|
label=_("I engage myself to participate to the whole TFJM²."),
|
2020-12-27 11:49:54 +01:00
|
|
|
|
required=True,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ValidateParticipationForm(forms.Form):
|
|
|
|
|
"""
|
|
|
|
|
Form to let administrators to accept or refuse a team.
|
|
|
|
|
"""
|
|
|
|
|
_form_type = forms.CharField(
|
|
|
|
|
initial="ValidateParticipationForm",
|
|
|
|
|
widget=forms.HiddenInput(),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
message = forms.CharField(
|
|
|
|
|
label=_("Message to address to the team:"),
|
|
|
|
|
widget=forms.Textarea(),
|
|
|
|
|
)
|
2020-12-31 12:13:42 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TournamentForm(forms.ModelForm):
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Tournament
|
|
|
|
|
fields = '__all__'
|
2023-02-20 17:23:12 +01:00
|
|
|
|
widgets = {
|
|
|
|
|
'date_start': forms.DateInput(attrs={'type': 'date'}, format='%Y-%m-%d'),
|
|
|
|
|
'date_end': forms.DateInput(attrs={'type': 'date'}, format='%Y-%m-%d'),
|
|
|
|
|
'inscription_limit': forms.DateTimeInput(attrs={'type': 'datetime-local'}, format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'solution_limit': forms.DateTimeInput(attrs={'type': 'datetime-local'}, format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'solutions_draw': forms.DateTimeInput(attrs={'type': 'datetime-local'}, format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'syntheses_first_phase_limit': forms.DateTimeInput(attrs={'type': 'datetime-local'},
|
|
|
|
|
format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'solutions_available_second_phase': forms.DateTimeInput(attrs={'type': 'datetime-local'},
|
|
|
|
|
format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'syntheses_second_phase_limit': forms.DateTimeInput(attrs={'type': 'datetime-local'},
|
|
|
|
|
format='%Y-%m-%d %H:%M'),
|
|
|
|
|
'organizers': forms.SelectMultiple(attrs={
|
|
|
|
|
'class': 'selectpicker',
|
|
|
|
|
'data-live-search': 'true',
|
|
|
|
|
'data-live-search-normalize': 'true',
|
|
|
|
|
'data-width': 'fit',
|
|
|
|
|
})
|
|
|
|
|
}
|
2021-01-12 17:24:46 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class SolutionForm(forms.ModelForm):
|
2021-01-19 00:32:34 +01:00
|
|
|
|
def clean_file(self):
|
|
|
|
|
if "file" in self.files:
|
|
|
|
|
file = self.files["file"]
|
2021-01-19 00:33:44 +01:00
|
|
|
|
if file.size > 5e6:
|
2021-01-19 00:32:34 +01:00
|
|
|
|
raise ValidationError(_("The uploaded file size must be under 5 Mo."))
|
|
|
|
|
if file.content_type != "application/pdf":
|
|
|
|
|
raise ValidationError(_("The uploaded file must be a PDF file."))
|
2023-03-29 18:34:55 +02:00
|
|
|
|
pdf_reader = PdfReader(file)
|
2021-01-19 00:32:34 +01:00
|
|
|
|
pages = len(pdf_reader.pages)
|
|
|
|
|
if pages > 30:
|
|
|
|
|
raise ValidationError(_("The PDF file must not have more than 30 pages."))
|
2021-04-03 22:02:53 +02:00
|
|
|
|
return self.cleaned_data["file"]
|
2021-01-19 00:32:34 +01:00
|
|
|
|
|
2021-01-12 17:24:46 +01:00
|
|
|
|
def save(self, commit=True):
|
|
|
|
|
"""
|
|
|
|
|
Don't save a solution with this way. Use a view instead
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Solution
|
|
|
|
|
fields = ('problem', 'file',)
|
2021-01-13 17:00:50 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PoolForm(forms.ModelForm):
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Pool
|
2023-03-31 17:23:40 +02:00
|
|
|
|
fields = ('tournament', 'round', 'letter', 'bbb_url', 'results_available', 'juries',)
|
2021-01-14 14:44:12 +01:00
|
|
|
|
widgets = {
|
2023-02-20 17:23:12 +01:00
|
|
|
|
"juries": forms.SelectMultiple(attrs={
|
|
|
|
|
'class': 'selectpicker',
|
|
|
|
|
'data-live-search': 'true',
|
|
|
|
|
'data-live-search-normalize': 'true',
|
|
|
|
|
}),
|
2021-01-14 14:44:12 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PoolTeamsForm(forms.ModelForm):
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
self.fields["participations"].queryset = self.instance.tournament.participations.all()
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Pool
|
|
|
|
|
fields = ('participations',)
|
|
|
|
|
widgets = {
|
2023-02-20 17:23:12 +01:00
|
|
|
|
"participations": forms.SelectMultiple(attrs={
|
|
|
|
|
'class': 'selectpicker',
|
|
|
|
|
'data-live-search': 'true',
|
|
|
|
|
'data-live-search-normalize': 'true',
|
|
|
|
|
'data-width': 'fit',
|
|
|
|
|
}),
|
2021-01-14 14:44:12 +01:00
|
|
|
|
}
|
2021-01-14 15:59:11 +01:00
|
|
|
|
|
2023-04-05 17:52:46 +02:00
|
|
|
|
|
2023-04-05 16:54:16 +02:00
|
|
|
|
class AddJuryForm(forms.ModelForm):
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
self.helper = FormHelper()
|
|
|
|
|
self.helper.form_class = 'form-inline'
|
|
|
|
|
self.helper.layout = Fieldset(
|
|
|
|
|
_("Add new jury"),
|
|
|
|
|
Div(
|
|
|
|
|
Div(
|
|
|
|
|
InlineField('first_name', autofocus="autofocus"),
|
|
|
|
|
css_class='col-xl-3',
|
|
|
|
|
),
|
|
|
|
|
Div(
|
|
|
|
|
InlineField('last_name'),
|
|
|
|
|
css_class='col-xl-3',
|
|
|
|
|
),
|
|
|
|
|
Div(
|
|
|
|
|
InlineField('email'),
|
|
|
|
|
css_class='col-xl-5',
|
|
|
|
|
),
|
|
|
|
|
Div(
|
|
|
|
|
Submit('submit', _("Add")),
|
|
|
|
|
css_class='col-xl-1',
|
|
|
|
|
),
|
|
|
|
|
css_class='row',
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def clean_email(self):
|
|
|
|
|
"""
|
|
|
|
|
Ensure that the email address is unique.
|
|
|
|
|
"""
|
|
|
|
|
email = self.data["email"]
|
|
|
|
|
if User.objects.filter(email=email).exists():
|
|
|
|
|
self.add_error("email", _("This email address is already used."))
|
|
|
|
|
return email
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = User
|
|
|
|
|
fields = ('first_name', 'last_name', 'email',)
|
|
|
|
|
|
|
|
|
|
|
2022-05-15 12:23:17 +02:00
|
|
|
|
class UploadNotesForm(forms.Form):
|
|
|
|
|
file = forms.FileField(
|
|
|
|
|
label=_("CSV file:"),
|
|
|
|
|
validators=[FileExtensionValidator(allowed_extensions=["csv"])],
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
|
self.fields['file'].widget.attrs['accept'] = 'text/csv'
|
|
|
|
|
|
|
|
|
|
def clean(self):
|
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
|
|
|
|
|
|
if 'file' in cleaned_data:
|
|
|
|
|
file = cleaned_data['file']
|
|
|
|
|
with file:
|
|
|
|
|
try:
|
2023-04-10 17:26:55 +02:00
|
|
|
|
data: bytes = file.read()
|
|
|
|
|
try:
|
|
|
|
|
content = data.decode()
|
|
|
|
|
except UnicodeDecodeError:
|
|
|
|
|
# This is not UTF-8, grrrr
|
|
|
|
|
content = data.decode('latin1')
|
|
|
|
|
csvfile = csv.reader(StringIO(content))
|
2023-04-08 17:33:01 +02:00
|
|
|
|
self.process(csvfile, cleaned_data)
|
2022-05-15 12:23:17 +02:00
|
|
|
|
except UnicodeDecodeError:
|
2023-04-10 17:26:55 +02:00
|
|
|
|
self.add_error('file', _("This file contains non-UTF-8 and non-ISO-8859-1 content. "
|
2022-05-15 12:23:17 +02:00
|
|
|
|
"Please send your sheet as a CSV file."))
|
|
|
|
|
|
|
|
|
|
return cleaned_data
|
|
|
|
|
|
|
|
|
|
def process(self, csvfile: Iterable[str], cleaned_data: dict):
|
|
|
|
|
parsed_notes = {}
|
2023-04-07 13:16:49 +02:00
|
|
|
|
valid_lengths = [1 + 6 * 3, 1 + 7 * 4, 1 + 6 * 5] # Per pool sizes
|
|
|
|
|
pool_size = 0
|
|
|
|
|
line_length = 0
|
2022-05-15 12:23:17 +02:00
|
|
|
|
for line in csvfile:
|
2023-04-10 17:26:55 +02:00
|
|
|
|
line = [s.strip() for s in line if s]
|
2023-04-07 13:16:49 +02:00
|
|
|
|
if line and line[0] == 'Problème':
|
|
|
|
|
pool_size = len(line) - 1
|
|
|
|
|
if pool_size < 3 or pool_size > 5:
|
|
|
|
|
self.add_error('file', _("Can't determine the pool size. Are you sure your file is correct?"))
|
|
|
|
|
return
|
|
|
|
|
line_length = valid_lengths[pool_size - 3]
|
2022-05-15 12:23:17 +02:00
|
|
|
|
continue
|
2023-04-07 13:16:49 +02:00
|
|
|
|
|
|
|
|
|
if pool_size == 0 or len(line) < line_length:
|
|
|
|
|
continue
|
|
|
|
|
|
2022-05-15 12:23:17 +02:00
|
|
|
|
name = line[0]
|
2023-04-07 21:47:06 +02:00
|
|
|
|
if name.lower() in ["rôle", "juré", "moyenne", "coefficient", "sous-total", "équipe", "equipe"]:
|
2023-03-31 18:28:23 +02:00
|
|
|
|
continue
|
2023-04-07 13:16:49 +02:00
|
|
|
|
notes = line[1:line_length]
|
|
|
|
|
if not all(s.isnumeric() or s[0] == '-' and s[1:].isnumeric() for s in notes):
|
2022-05-15 12:23:17 +02:00
|
|
|
|
continue
|
|
|
|
|
notes = list(map(int, notes))
|
|
|
|
|
|
2023-04-07 13:16:49 +02:00
|
|
|
|
max_notes = pool_size * ([20, 16, 9, 10, 9, 10] + ([4] if pool_size == 4 else []))
|
2022-05-15 12:23:17 +02:00
|
|
|
|
for n, max_n in zip(notes, max_notes):
|
|
|
|
|
if n > max_n:
|
|
|
|
|
self.add_error('file',
|
|
|
|
|
_("The following note is higher of the maximum expected value:")
|
|
|
|
|
+ str(n) + " > " + str(max_n))
|
|
|
|
|
|
2023-04-10 17:26:30 +02:00
|
|
|
|
# Search by "{first_name} {last_name}"
|
|
|
|
|
jury = User.objects.annotate(full_name=Concat('first_name', Value(' '), 'last_name',
|
|
|
|
|
output_field=CharField())) \
|
|
|
|
|
.filter(full_name=name.replace('’', '\''), registration__volunteerregistration__isnull=False)
|
2022-05-15 12:23:17 +02:00
|
|
|
|
if jury.count() != 1:
|
2022-05-15 16:16:41 +02:00
|
|
|
|
self.add_error('file', _("The following user was not found:") + " " + name)
|
2022-05-15 12:23:17 +02:00
|
|
|
|
continue
|
|
|
|
|
jury = jury.get()
|
|
|
|
|
|
|
|
|
|
vr = jury.registration
|
|
|
|
|
parsed_notes[vr] = notes
|
|
|
|
|
|
|
|
|
|
cleaned_data['parsed_notes'] = parsed_notes
|
|
|
|
|
|
|
|
|
|
return cleaned_data
|
|
|
|
|
|
|
|
|
|
|
2021-01-14 15:59:11 +01:00
|
|
|
|
class PassageForm(forms.ModelForm):
|
|
|
|
|
def clean(self):
|
|
|
|
|
cleaned_data = super().clean()
|
|
|
|
|
if "defender" in cleaned_data and "opponent" in cleaned_data and "reporter" in cleaned_data \
|
|
|
|
|
and len({cleaned_data["defender"], cleaned_data["opponent"], cleaned_data["reporter"]}) < 3:
|
|
|
|
|
self.add_error(None, _("The defender, the opponent and the reporter must be different."))
|
|
|
|
|
if "defender" in self.cleaned_data and "solution_number" in self.cleaned_data \
|
|
|
|
|
and not Solution.objects.filter(participation=cleaned_data["defender"],
|
|
|
|
|
problem=cleaned_data["solution_number"]).exists():
|
|
|
|
|
self.add_error("solution_number", _("This defender did not work on this problem."))
|
|
|
|
|
return cleaned_data
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Passage
|
2023-04-07 12:10:25 +02:00
|
|
|
|
fields = ('position', 'solution_number', 'defender', 'opponent', 'reporter', 'observer', 'defender_penalties',)
|
2021-01-14 17:26:08 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class SynthesisForm(forms.ModelForm):
|
2021-01-19 00:33:44 +01:00
|
|
|
|
def clean_file(self):
|
|
|
|
|
if "file" in self.files:
|
|
|
|
|
file = self.files["file"]
|
|
|
|
|
if file.size > 2e6:
|
|
|
|
|
raise ValidationError(_("The uploaded file size must be under 2 Mo."))
|
|
|
|
|
if file.content_type != "application/pdf":
|
|
|
|
|
raise ValidationError(_("The uploaded file must be a PDF file."))
|
2023-05-20 17:09:32 +02:00
|
|
|
|
pdf_reader = PdfReader(file)
|
|
|
|
|
pages = len(pdf_reader.pages)
|
|
|
|
|
if pages > 2:
|
|
|
|
|
raise ValidationError(_("The PDF file must not have more than 2 pages."))
|
2021-04-03 22:02:53 +02:00
|
|
|
|
return self.cleaned_data["file"]
|
2021-01-19 00:33:44 +01:00
|
|
|
|
|
2021-01-14 17:26:08 +01:00
|
|
|
|
def save(self, commit=True):
|
|
|
|
|
"""
|
|
|
|
|
Don't save a synthesis with this way. Use a view instead
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Synthesis
|
2021-04-04 18:13:30 +02:00
|
|
|
|
fields = ('file',)
|
2021-01-14 18:21:22 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class NoteForm(forms.ModelForm):
|
|
|
|
|
class Meta:
|
|
|
|
|
model = Note
|
|
|
|
|
fields = ('defender_writing', 'defender_oral', 'opponent_writing',
|
2023-04-07 12:10:25 +02:00
|
|
|
|
'opponent_oral', 'reporter_writing', 'reporter_oral', 'observer_oral', )
|