forked from ScoDoc/ScoDoc
209 lines
7.8 KiB
Python
209 lines
7.8 KiB
Python
# -*- mode: python -*-
|
|
# -*- coding: utf-8 -*-
|
|
|
|
##############################################################################
|
|
#
|
|
# ScoDoc
|
|
#
|
|
# Copyright (c) 1999 - 2024 Emmanuel Viennet. All rights reserved.
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
# Emmanuel Viennet emmanuel.viennet@viennet.net
|
|
#
|
|
##############################################################################
|
|
|
|
"""
|
|
Formulaire configuration Module Assiduités
|
|
"""
|
|
import datetime
|
|
import re
|
|
|
|
from flask_wtf import FlaskForm
|
|
from wtforms import DecimalField, SubmitField, ValidationError
|
|
from wtforms.fields.simple import StringField
|
|
from wtforms.validators import Optional
|
|
|
|
from wtforms.widgets import TimeInput
|
|
|
|
|
|
class TimeField(StringField):
|
|
"""HTML5 time input.
|
|
tiré de : https://gist.github.com/tachyondecay/6016d32f65a996d0d94f
|
|
"""
|
|
|
|
widget = TimeInput()
|
|
|
|
def __init__(self, label=None, validators=None, fmt="%H:%M:%S", **kwargs):
|
|
super(TimeField, self).__init__(label, validators, **kwargs)
|
|
self.fmt = fmt
|
|
self.data = None
|
|
|
|
def _value(self):
|
|
if self.raw_data:
|
|
return " ".join(self.raw_data)
|
|
if self.data and isinstance(self.data, str):
|
|
self.data = datetime.time(*map(int, self.data.split(":")))
|
|
return self.data and self.data.strftime(self.fmt) or ""
|
|
|
|
def process_formdata(self, valuelist):
|
|
if valuelist:
|
|
time_str = " ".join(valuelist)
|
|
try:
|
|
components = time_str.split(":")
|
|
hour = 0
|
|
minutes = 0
|
|
seconds = 0
|
|
if len(components) in range(2, 4):
|
|
hour = int(components[0])
|
|
minutes = int(components[1])
|
|
|
|
if len(components) == 3:
|
|
seconds = int(components[2])
|
|
else:
|
|
raise ValueError
|
|
self.data = datetime.time(hour, minutes, seconds)
|
|
except ValueError as exc:
|
|
self.data = None
|
|
raise ValueError(self.gettext("Not a valid time string")) from exc
|
|
|
|
|
|
def check_tick_time(form, field):
|
|
"""Le tick_time doit être entre 0 et 60 minutes"""
|
|
if field.data < 1 or field.data > 59:
|
|
raise ValidationError("Valeur de granularité invalide (entre 1 et 59)")
|
|
|
|
|
|
def check_ics_path(form, field):
|
|
"""Vérifie que le chemin est bien un chemin absolu
|
|
et qu'il contient edt_id
|
|
"""
|
|
data = field.data.strip()
|
|
if not data:
|
|
return
|
|
if not data.startswith("/"):
|
|
raise ValidationError("Le chemin vers les ics doit commencer par /")
|
|
if not "{edt_id}" in data:
|
|
raise ValidationError("Le chemin vers les ics doit utiliser {edt_id}")
|
|
|
|
|
|
def check_ics_field(form, field):
|
|
"""Vérifie que c'est un nom de champ crédible: un mot alphanumérique"""
|
|
if not re.match(r"^[a-zA-Z\-_0-9]+$", field.data):
|
|
raise ValidationError("nom de champ ics invalide")
|
|
|
|
|
|
def check_ics_regexp(form, field):
|
|
"""Vérifie que field est une expresssion régulière"""
|
|
value = field.data.strip()
|
|
# check that it compiles
|
|
try:
|
|
_ = re.compile(value)
|
|
except re.error as exc:
|
|
raise ValidationError("expression invalide") from exc
|
|
return True
|
|
|
|
|
|
class ConfigAssiduitesForm(FlaskForm):
|
|
"Formulaire paramétrage Module Assiduité"
|
|
|
|
assi_morning_time = TimeField(
|
|
"Début de la journée"
|
|
) # TODO utiliser TextField + timepicker voir AjoutAssiOrJustForm
|
|
assi_lunch_time = TimeField(
|
|
"Heure de midi (date pivot entre matin et après-midi)"
|
|
) # TODO
|
|
assi_afternoon_time = TimeField("Fin de la journée") # TODO
|
|
|
|
assi_tick_time = DecimalField(
|
|
"Granularité de la timeline (temps en minutes)",
|
|
places=0,
|
|
validators=[check_tick_time],
|
|
)
|
|
|
|
edt_ics_path = StringField(
|
|
label="Chemin vers les ics",
|
|
description="""Chemin absolu unix sur le serveur vers le fichier ics donnant l'emploi
|
|
du temps d'un semestre. La balise <tt>{edt_id}</tt> sera remplacée par l'edt_id du
|
|
semestre (par défaut, son code étape Apogée).
|
|
Si ce champ n'est pas renseigné, les emplois du temps ne seront pas utilisés.""",
|
|
validators=[Optional(), check_ics_path],
|
|
)
|
|
edt_ics_user_path = StringField(
|
|
label="Chemin vers les ics des utilisateurs (enseignants)",
|
|
description="""Optionnel. Chemin absolu unix sur le serveur vers le fichier ics donnant l'emploi
|
|
du temps d'un enseignant. La balise <tt>{edt_id}</tt> sera remplacée par l'edt_id du
|
|
de l'utilisateur.
|
|
Dans certains cas (XXX), ScoDoc peut générer ces fichiers et les écrira suivant
|
|
ce chemin (avec edt_id).
|
|
""",
|
|
validators=[Optional(), check_ics_path],
|
|
)
|
|
|
|
edt_ics_title_field = StringField(
|
|
label="Champ contenant le titre",
|
|
description="""champ de l'évènement calendrier: DESCRIPTION, SUMMARY, ...""",
|
|
validators=[Optional(), check_ics_field],
|
|
)
|
|
edt_ics_title_regexp = StringField(
|
|
label="Extraction du titre",
|
|
description=r"""expression régulière python dont le premier groupe doit
|
|
sera le titre de l'évènement affcihé dans le calendrier ScoDoc.
|
|
Exemple: <tt>Matière : \w+ - ([\w\.\s']+)</tt>
|
|
""",
|
|
validators=[Optional(), check_ics_regexp],
|
|
)
|
|
edt_ics_group_field = StringField(
|
|
label="Champ contenant le groupe",
|
|
description="""champ de l'évènement calendrier: DESCRIPTION, SUMMARY, ...""",
|
|
validators=[Optional(), check_ics_field],
|
|
)
|
|
edt_ics_group_regexp = StringField(
|
|
label="Extraction du groupe",
|
|
description=r"""expression régulière python dont le premier groupe doit
|
|
correspondre à l'identifiant de groupe de l'emploi du temps.
|
|
Exemple: <tt>.*- ([\w\s]+)$</tt>
|
|
""",
|
|
validators=[Optional(), check_ics_regexp],
|
|
)
|
|
edt_ics_mod_field = StringField(
|
|
label="Champ contenant le module",
|
|
description="""champ de l'évènement calendrier: DESCRIPTION, SUMMARY, ...""",
|
|
validators=[Optional(), check_ics_field],
|
|
)
|
|
edt_ics_mod_regexp = StringField(
|
|
label="Extraction du module",
|
|
description=r"""expression régulière python dont le premier groupe doit
|
|
correspondre à l'identifiant (code) du module de l'emploi du temps.
|
|
Exemple: <tt>Matière : ([A-Z][A-Z0-9]+)</tt>
|
|
""",
|
|
validators=[Optional(), check_ics_regexp],
|
|
)
|
|
edt_ics_uid_field = StringField(
|
|
label="Champ contenant l'enseignant",
|
|
description="""champ de l'évènement calendrier: DESCRIPTION, SUMMARY, ...""",
|
|
validators=[Optional(), check_ics_field],
|
|
)
|
|
edt_ics_uid_regexp = StringField(
|
|
label="Extraction de l'enseignant",
|
|
description=r"""expression régulière python dont le premier groupe doit
|
|
correspondre à l'identifiant (edt_id) de l'enseignant associé à l'évènement.
|
|
Exemple: <tt>Enseignant : ([0-9]+)</tt>
|
|
""",
|
|
validators=[Optional(), check_ics_regexp],
|
|
)
|
|
submit = SubmitField("Valider")
|
|
cancel = SubmitField("Annuler", render_kw={"formnovalidate": True})
|