# -*- mode: python -*- # -*- coding: utf-8 -*- ############################################################################## # # Gestion scolarite IUT # # 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 # ############################################################################## """Exception handling """ from flask_login import current_user import app # --- Exceptions class ScoException(Exception): "super classe de toutes les exceptions ScoDoc." class InvalidNoteValue(ScoException): "Valeur note invalide. Usage interne saisie note." class ScoInvalidCSRF(ScoException): "Erreur validation token CSRF" class ScoValueError(ScoException): """Exception avec page d'erreur utilisateur - dest_url : url où aller après la page d'erreur - safe (default False): si vrai, affiche le message non html quoté. """ # mal nommée: super classe de toutes les exceptions avec page # d'erreur gentille. def __init__(self, msg, dest_url=None, safe=False): super().__init__(msg) self.dest_url = dest_url self.safe = safe # utilisé par template sco_value_error.j2 class ScoPermissionDenied(ScoValueError): """Permission non accordée (appli web)""" def __init__(self, msg=None, dest_url=None): if msg is None: msg = f"""Opération non autorisée pour { current_user.get_nomcomplet() if current_user else "?" }. Pas la permission, ou objet verrouillé.""" super().__init__(msg, dest_url=dest_url) class ScoBugCatcher(ScoException): "bug avec enquete en cours" class NoteProcessError(ScoValueError): "Valeurs notes invalides" class InvalidEtudId(NoteProcessError): pass class ScoFormatError(ScoValueError): "Erreur lecture d'un fichier fourni par l'utilisateur" def __init__(self, msg, filename="", dest_url=None): super().__init__(msg, dest_url=dest_url) self.filename = filename class ScoInvalidParamError(ScoValueError): """Paramètres requete invalides. Utilisée lorsqu'une route est appelée avec des paramètres invalides (id strings, ...) """ def __init__(self, msg=None, dest_url=None): msg = msg or "Adresse invalide. Vérifiez vos signets." super().__init__(msg, dest_url=dest_url) class ScoPDFFormatError(ScoValueError): "erreur génération PDF (templates platypus, ...)" def __init__(self, msg, dest_url=None): super().__init__( f"""Erreur dans un format pdf:
{msg}
Vérifiez les paramètres (polices de caractères, balisage, réglages bulletins...) dans les paramètres ou préférences.
""", dest_url=dest_url, ) class ScoInvalidDept(ScoValueError): """departement invalide""" pass class ScoConfigurationError(ScoValueError): """Configuration invalid""" pass class ScoLockedFormError(ScoValueError): "Modification d'une formation verrouillée" def __init__(self, msg="", dest_url=None): msg = ( "Cette formation est verrouillée (car il y a un semestre verrouillé qui s'y réfère). " + str(msg) ) super().__init__(msg=msg, dest_url=dest_url) class ScoLockedSemError(ScoValueError): "Modification d'un formsemestre verrouillé" def __init__(self, msg="", dest_url=None): msg = "Ce semestre est verrouillé ! " + str(msg) super().__init__(msg=msg, dest_url=dest_url) class ScoNonEmptyFormationObject(ScoValueError): """On ne peut pas supprimer un module/matiere ou UE si des formsemestre s'y réfèrent""" def __init__(self, type_objet="objet'", msg="", dest_url=None): msg = f"""Il faut d'abord supprimer le semestre (ou en retirer ce {type_objet}). Mais il est peut-être préférable de laisser ce programme intact et d'en créer une nouvelle version pour la modifier sans affecter les semestres déjà en place.
""" super().__init__(msg=msg, dest_url=dest_url) class ScoInvalidIdType(ScoValueError): """Pour les clients qui s'obstinent à utiliser des bookmarks ou historiques anciens avec des ID ScoDoc7. """ def __init__(self, msg=""): import app.scodoc.sco_utils as scu msg = f"""
Vous utilisez un lien invalide, qui correspond probablement
à une ancienne version du logiciel.
Au besoin, mettre à jour vos marque-pages.
Si le problème persiste, merci de contacter l'assistance via la liste de diffusion Notes ou de préférence le salon Discord.
Message serveur: {msg}
""" super().__init__(msg) class ScoNoReferentielCompetences(ScoValueError): """Formation APC (BUT) non associée à référentiel de compétences""" def __init__(self, msg: str = "", formation: "Formation" = None): formation_title = ( f"{formation.titre} version {formation.version}" if formation else "" ) msg = f"""Pas de référentiel de compétences associé à la formation {formation_title}!
Pour associer un référentiel, passer par le menu Semestre / Voir la formation... et suivre le lien "associer à un référentiel de compétences" """ super().__init__(msg) class ScoGenError(ScoException): "exception avec affichage d'une page explicative ad-hoc" def __init__(self, msg=""): super().__init__(msg) class AccessDenied(ScoGenError): pass class ScoInvalidDateError(ScoValueError): pass class APIInvalidParams(Exception): """Exception pour les API JSON""" status_code = 400 def __init__(self, message, status_code=None, payload=None): super().__init__() self.message = message if status_code is not None: self.status_code = status_code self.payload = payload def to_dict(self): "dict" rv = dict(self.payload or ()) rv["message"] = self.message return rv class ScoFormationConflict(Exception): """Conflit cohérence formation (APC)""" class ScoTemporaryError(ScoValueError): """Erreurs temporaires rarissimes (caches ?)""" def __init__(self, msg: str = ""): msg = """
Erreur temporaire
Veuillez ré-essayer. Si le problème persiste (ou s'il venait à se produire fréquemment), merci de contacter l'assistance ScoDoc (voir les informations de contact).
""" app.clear_scodoc_cache() super().__init__(msg)