forked from ScoDoc/ScoDoc
457 lines
17 KiB
Python
457 lines
17 KiB
Python
##############################################################################
|
|
# Module "Avis de poursuite d'étude"
|
|
# conçu et développé par Cléo Baras (IUT de Grenoble)
|
|
##############################################################################
|
|
|
|
"""
|
|
Created on 01-2024
|
|
|
|
@author: barasc
|
|
"""
|
|
|
|
import app.pe.pe_comp as pe_comp
|
|
|
|
from app.models import FormSemestre
|
|
from app.pe import pe_sxtag, pe_affichage
|
|
from app.pe.pe_etudiant import EtudiantsJuryPE, get_dernier_semestre_en_date
|
|
|
|
TYPES_RCS = {
|
|
"S1": {
|
|
"aggregat": ["S1"],
|
|
"descr": "Semestre 1 (S1)",
|
|
},
|
|
"S2": {
|
|
"aggregat": ["S2"],
|
|
"descr": "Semestre 2 (S2)",
|
|
},
|
|
"1A": {
|
|
"aggregat": ["S1", "S2"],
|
|
"descr": "BUT1 (S1+S2)",
|
|
},
|
|
"S3": {
|
|
"aggregat": ["S3"],
|
|
"descr": "Semestre 3 (S3)",
|
|
},
|
|
"S4": {
|
|
"aggregat": ["S4"],
|
|
"descr": "Semestre 4 (S4)",
|
|
},
|
|
"2A": {
|
|
"aggregat": ["S3", "S4"],
|
|
"descr": "BUT2 (S3+S4)",
|
|
},
|
|
"3S": {
|
|
"aggregat": ["S1", "S2", "S3"],
|
|
"descr": "Moyenne du semestre 1 au semestre 3 (S1+S2+S3)",
|
|
},
|
|
"4S": {
|
|
"aggregat": ["S1", "S2", "S3", "S4"],
|
|
"descr": "Moyenne du semestre 1 au semestre 4 (S1+S2+S3+S4)",
|
|
},
|
|
"S5": {
|
|
"aggregat": ["S5"],
|
|
"descr": "Semestre 5 (S5)",
|
|
},
|
|
"S6": {
|
|
"aggregat": ["S6"],
|
|
"descr": "Semestre 6 (S6)",
|
|
},
|
|
"3A": {
|
|
"aggregat": ["S5", "S6"],
|
|
"descr": "3ème année (S5+S6)",
|
|
},
|
|
"5S": {
|
|
"aggregat": ["S1", "S2", "S3", "S4", "S5"],
|
|
"descr": "Moyenne du semestre 1 au semestre 5 (S1+S2+S3+S4+S5)",
|
|
},
|
|
"6S": {
|
|
"aggregat": ["S1", "S2", "S3", "S4", "S5", "S6"],
|
|
"descr": "Moyenne globale (S1+S2+S3+S4+S5+S6)",
|
|
},
|
|
}
|
|
"""Dictionnaire détaillant les différents regroupements cohérents
|
|
de semestres (RCS), en leur attribuant un nom et en détaillant
|
|
le nom des semestres qu'ils regroupent et l'affichage qui en sera fait
|
|
dans les tableurs de synthèse.
|
|
"""
|
|
|
|
TOUS_LES_RCS_AVEC_PLUSIEURS_SEM = [cle for cle in TYPES_RCS if not cle.startswith("S")]
|
|
TOUS_LES_RCS = list(TYPES_RCS.keys())
|
|
TOUS_LES_SEMESTRES = [cle for cle in TYPES_RCS if cle.startswith("S")]
|
|
|
|
|
|
def get_descr_rcs(nom_rcs: str) -> str:
|
|
"""Renvoie la description pour les tableurs de synthèse
|
|
Excel d'un nom de RCS"""
|
|
return TYPES_RCS[nom_rcs]["descr"]
|
|
|
|
|
|
class RCS:
|
|
"""Modélise un regroupement cohérent de semestres (formsemestre ou de Sx)"""
|
|
|
|
def __init__(self, nom_rcs: str, semestre_final: FormSemestre):
|
|
self.nom: str = nom_rcs
|
|
"""Nom du RCS"""
|
|
|
|
self.formsemestre_final: FormSemestre = semestre_final
|
|
"""FormSemestre terminal du RCS"""
|
|
|
|
self.rang_final = self.formsemestre_final.semestre_id
|
|
"""Le rang du formsemestre final"""
|
|
|
|
self.rcs_id: (str, int) = (nom_rcs, semestre_final.formsemestre_id)
|
|
"""Identifiant du RCS sous forme (nom_rcs, id du semestre_terminal)"""
|
|
|
|
def get_formsemestre_id_final(self) -> int:
|
|
"""Renvoie l'identifiant du formsemestre final du RCS
|
|
|
|
Returns:
|
|
L'id du formsemestre final (marquant la fin) du RCS
|
|
"""
|
|
return self.formsemestre_final.formsemestre_id
|
|
|
|
def __repr__(self):
|
|
"""Représentation textuelle d'un RCS"""
|
|
return f"{self.nom} ({self.formsemestre_final.formsemestre_id}) {self.formsemestre_final.date_fin.year}"
|
|
|
|
def get_repr(self):
|
|
return self.__repr__()
|
|
|
|
def __eq(self, other):
|
|
"""Egalité de RCS"""
|
|
return (
|
|
self.nom == other.nom
|
|
and self.formsemestre_final == other.formsemestre_final
|
|
)
|
|
|
|
|
|
class RCF(RCS):
|
|
"""Modélise un ensemble de (form)semestres d'étudiants
|
|
associé à un type de regroupement cohérent de semestres
|
|
donné (par ex: 'S2', '3S', '2A').
|
|
|
|
Si le RCF est un semestre de type Si, stocke les
|
|
formsemestres de numéro i qu'ont suivi l'étudiant pour atteindre le Si
|
|
(en général 1 si personnes n'a redoublé, mais 2 s'il y a des redoublants)
|
|
|
|
Pour le RCF de type iS ou iA (par ex, 3A=S1+S2+S3), identifie
|
|
les semestres que les étudiants ont suivis pour les amener jusqu'au semestre
|
|
terminal du RCS (par ex: ici un S3).
|
|
|
|
Ces semestres peuvent être :
|
|
|
|
* des S1+S2+S1+S2+S3 si redoublement de la 1ère année
|
|
* des S1+S2+(année de césure)+S3 si césure, ...
|
|
|
|
Args:
|
|
nom_rcs: Un nom du RCS (par ex: '5S')
|
|
semestre_final: Le formsemestre final du RCS
|
|
"""
|
|
|
|
def __init__(self, nom_rcs: str, semestre_final: FormSemestre):
|
|
RCS.__init__(self, nom_rcs, semestre_final)
|
|
|
|
self.semestres_aggreges: dict[int:FormSemestre] = {}
|
|
"""Formsemestres regroupés dans le RCS"""
|
|
|
|
def add_semestres_a_aggreger(self, semestres: dict[int:FormSemestre]):
|
|
"""Ajout de semestres aux semestres à regrouper
|
|
|
|
Args:
|
|
semestres: Dictionnaire ``{fid: Formsemestre)``
|
|
"""
|
|
self.semestres_aggreges = self.semestres_aggreges | semestres
|
|
|
|
def get_repr(self, verbose=True) -> str:
|
|
"""Représentation textuelle d'un RCS
|
|
basé sur ses semestres aggrégés"""
|
|
|
|
noms = []
|
|
for fid in self.semestres_aggreges:
|
|
semestre = self.semestres_aggreges[fid]
|
|
noms.append(f"S{semestre.semestre_id}({fid})")
|
|
noms = sorted(noms)
|
|
title = f"""{str(self)}"""
|
|
if verbose and noms:
|
|
title += " - " + "+".join(noms)
|
|
return title
|
|
|
|
|
|
|
|
def get_rcf_from_semestres_aggreges(
|
|
rcfs: dict[(str, int):RCF], semestres_a_aggreges: list[FormSemestre]
|
|
) -> (str, int):
|
|
"""Partant d'un dictionnaire de RCFs (de la forme
|
|
``{ (nom_rcs, fid): RCF }, et connaissant une liste
|
|
de (form)semestres à aggréger, renvoie l'identifiant
|
|
(nom_rcs, fid) du RCFs qui lui correspond (c'est à dire celui dont
|
|
les semestres_aggregés par le RCF sont les même que les
|
|
semestres_a_aggreger.
|
|
|
|
Returns:
|
|
rcf_id: L'identifiant du RCF trouvé
|
|
"""
|
|
fids_semestres_a_aggreger = set(
|
|
[frms.formsemestre_id for frms in semestres_a_aggreges]
|
|
)
|
|
for rcf_id, rcf in rcfs.items():
|
|
fids_rcf = set(rcf.semestres_aggreges)
|
|
if fids_rcf == fids_semestres_a_aggreger:
|
|
return rcf_id
|
|
return None
|
|
|
|
|
|
class RCRCF:
|
|
"""Modélise les RCF d'étudiants suivis par un étudiant dans
|
|
le cadre d'un RCS donné (par ex: 3S=S1+S2+S3).
|
|
|
|
Pour rappel : un RCF (par ex. S1) combine les semestres 1 qu'a suivi
|
|
l'étudiant pour valider son S1 (1 si étudiant standard, 2 si redoublant).
|
|
|
|
Le RCRCF 3S est donc le regroupement du RCF S1 + RCF S2 + RCF S3.
|
|
|
|
Il est identifié par le formsemestre de S3 marquant la fin du regroupement.
|
|
|
|
|
|
Args:
|
|
nom_rcs: Un nom du RCS (par ex: '5S')
|
|
semestre_final: Le semestre final du RCS
|
|
"""
|
|
|
|
def __init__(self, nom_rcs: str, semestre_final: FormSemestre):
|
|
RCS.__init__(self, nom_rcs, semestre_final)
|
|
|
|
self.rcfs_aggreges: dict[(str, int) : pe_sxtag.SxTag] = {}
|
|
"""Les RCFs à aggréger"""
|
|
|
|
def add_rcfs_a_aggreger(self, rcfs: dict[(str, int):RCF]):
|
|
"""Ajout des RCFs aux RCFS à regrouper
|
|
|
|
Args:
|
|
rcfs: Dictionnaire ``{(str,fid): RCF}`` à ajouter
|
|
"""
|
|
self.rcfs_aggreges = self.rcfs_aggreges | rcfs
|
|
|
|
def get_repr(self, verbose=True) -> str:
|
|
"""Représentation textuelle d'un RCSF
|
|
basé sur ses RCF aggrégés"""
|
|
|
|
noms = []
|
|
for rcf_id, rcf in self.rcfs_aggreges.items():
|
|
noms.append(rcf.get_repr())
|
|
title = f"""{str(self)}"""
|
|
if verbose and noms:
|
|
title += " : " + "+".join(noms)
|
|
return title
|
|
|
|
|
|
class RCSsJuryPE:
|
|
"""Classe centralisant tous les regroupements cohérents de
|
|
semestres (RCS) des étudiants à prendre en compte dans un jury PE
|
|
|
|
Args:
|
|
annee_diplome: L'année de diplomation
|
|
"""
|
|
|
|
def __init__(self, annee_diplome: int):
|
|
self.annee_diplome = annee_diplome
|
|
"""Année de diplômation"""
|
|
|
|
self.rcss: dict[tuple(int, str) : RCF] = {}
|
|
"""Ensemble des RCS recensés"""
|
|
|
|
self.rcss_suivis: dict[int:dict] = {}
|
|
"""Dictionnaire associant, pour chaque étudiant et pour chaque type de RCS,
|
|
son RCS : {etudid: {nom_RCS: RCS}}"""
|
|
|
|
self.rcfs: dict[tuple(int, str) : RCF] = {}
|
|
"""Ensemble des RCF recensés : {(nom_RCS, fid_terminal): RCF}"""
|
|
|
|
self.rcfs_suivis: dict[int:dict] = {}
|
|
"""Dictionnaire associant, pour chaque étudiant et pour chaque type de RCS,
|
|
son RCS : {etudid: {nom_RCS: RCF}}"""
|
|
|
|
self.rcrcfs: dict[tuple(int, str) : RCRCF] = {}
|
|
"""Ensemble des RCS recensés : {(nom_RCS, fid_terminal): RCRCF}"""
|
|
|
|
self.rcrcfs_suivis: dict[int:str] = {}
|
|
"""Dictionnaire associant, pour chaque étudiant et pour chaque type de RCS,
|
|
son RCRCF : {etudid: {nom_RCS: RCSx}}"""
|
|
|
|
def cree_rcss(self, etudiants: EtudiantsJuryPE):
|
|
"""Créé tous les RCS, au regard du cursus des étudiants
|
|
analysés + les mémorise dans les données de l'étudiant
|
|
|
|
Args:
|
|
etudiants: Les étudiants à prendre en compte dans le Jury PE
|
|
"""
|
|
|
|
for etudid in etudiants.trajectoires:
|
|
self.rcss_suivis[etudid] = {
|
|
aggregat: None
|
|
for aggregat in pe_comp.TOUS_LES_SEMESTRES
|
|
+ TOUS_LES_RCS_AVEC_PLUSIEURS_SEM
|
|
}
|
|
|
|
for nom_rcs in pe_comp.TOUS_LES_SEMESTRES + TOUS_LES_RCS_AVEC_PLUSIEURS_SEM:
|
|
# L'aggrégat considéré (par ex: 3S=S1+S2+S3), son nom de son semestre
|
|
# terminal (par ex: S3) et son numéro (par ex: 3)
|
|
noms_semestre_de_aggregat = TYPES_RCS[nom_rcs]["aggregat"]
|
|
nom_semestre_terminal = noms_semestre_de_aggregat[-1]
|
|
|
|
for etudid in etudiants.trajectoires:
|
|
# Le formsemestre terminal (dernier en date) associé au
|
|
# semestre marquant la fin de l'aggrégat
|
|
# (par ex: son dernier S3 en date)
|
|
trajectoire = etudiants.trajectoires[etudid][nom_semestre_terminal]
|
|
if trajectoire:
|
|
formsemestre_final = get_dernier_semestre_en_date(trajectoire)
|
|
|
|
# Ajout ou récupération du RCS associé
|
|
rcs_id = (nom_rcs, formsemestre_final.formsemestre_id)
|
|
if rcs_id not in self.rcss:
|
|
self.rcss[rcs_id] = RCF(nom_rcs, formsemestre_final)
|
|
rcs = self.rcss[rcs_id]
|
|
|
|
# La liste des semestres de l'étudiant à prendre en compte
|
|
# pour cette trajectoire
|
|
semestres_a_aggreger = get_rcs_etudiant(
|
|
etudiants.trajectoires[etudid], formsemestre_final, nom_rcs
|
|
)
|
|
|
|
# Ajout des semestres au RCS
|
|
rcs.add_semestres_a_aggreger(semestres_a_aggreger)
|
|
|
|
# Mémorise le RCS suivi par l'étudiant
|
|
self.rcss_suivis[etudid][nom_rcs] = rcs
|
|
|
|
# Affichage pour debug
|
|
jeunes = list(enumerate(self.rcss_suivis))
|
|
for no_etud, etudid in jeunes[:20]:
|
|
pe_affichage.pe_print(f"-> {etudiants.identites[etudid].nomprenom} :")
|
|
for nom_rcs, rcs in self.rcss_suivis[etudid].items():
|
|
if rcs:
|
|
pe_affichage.pe_print(f" > RCS {nom_rcs}: {rcs.get_repr()}")
|
|
|
|
def cree_rcfs(self, etudiants: EtudiantsJuryPE):
|
|
"""Créé les RCFs en ne conservant dans les RCS que les regroupements
|
|
de type Sx"""
|
|
self.rcfs = {}
|
|
for rcs_id, rcs in self.rcss.items():
|
|
if rcs and rcs.nom in pe_comp.TOUS_LES_SEMESTRES:
|
|
self.rcfs[rcs_id] = rcs
|
|
print(self.rcfs)
|
|
|
|
for etudid in self.rcss_suivis:
|
|
for nom_rcs, rcs in self.rcss_suivis[etudid].items():
|
|
if rcs and nom_rcs in pe_comp.TOUS_LES_SEMESTRES:
|
|
if etudid not in self.rcfs_suivis:
|
|
self.rcfs_suivis[etudid] = {}
|
|
self.rcfs_suivis[etudid][nom_rcs] = rcs
|
|
|
|
# Affichage pour debug
|
|
jeunes = list(enumerate(self.rcfs_suivis))
|
|
for no_etud, etudid in jeunes[:20]:
|
|
pe_affichage.pe_print(f"-> {etudiants.identites[etudid].nomprenom} :")
|
|
for nom_rcs, rcs in self.rcfs_suivis[etudid].items():
|
|
if rcs:
|
|
pe_affichage.pe_print(f" > RCF {nom_rcs}: {rcs.get_repr()}")
|
|
else:
|
|
pe_affichage.pe_print(f" > RCF {nom_rcs}: <vide> !!! ")
|
|
|
|
def cree_rcrcfs(self, etudiants: EtudiantsJuryPE):
|
|
"""Créé tous les RCRCF, au regard du cursus des étudiants
|
|
analysés (trajectoires traduisant son parcours dans les
|
|
différents semestres) + les mémorise dans les données de l'étudiant
|
|
"""
|
|
|
|
# Pour tous les étudiants du jury
|
|
for etudid in self.rcss_suivis:
|
|
self.rcrcfs_suivis[etudid] = {}
|
|
|
|
for nom_rcs, rcf in self.rcfs_suivis[etudid].items(): # Pour chaque RCS
|
|
semestres_a_aggreger = rcf.semestres_aggreges
|
|
|
|
# Tri des semestres par rang
|
|
semestres_tries = pe_comp.tri_semestres_par_rang(semestres_a_aggreger)
|
|
|
|
# Récupére les RCFs de type Sx traduisant sa trajectoire
|
|
rcfs_a_aggreger = {}
|
|
for semestres_du_rang in semestres_tries.values():
|
|
if semestres_du_rang:
|
|
rcf_id = get_rcf_from_semestres_aggreges(
|
|
self.rcfs, semestres_du_rang
|
|
)
|
|
if not rcf_id:
|
|
raise ValueError(
|
|
"Il manque un RCF pour créer les RCRCFs dans cree_rcrcfs"
|
|
)
|
|
rcfs_a_aggreger[rcf_id] = self.rcfs[rcf_id]
|
|
|
|
# Ajout du RCRCF
|
|
if rcf_id not in self.rcrcfs:
|
|
self.rcrcfs[rcf_id] = RCRCF(rcf_id, rcf.formsemestre_final)
|
|
rcrcf = self.rcrcfs[rcf_id]
|
|
|
|
# Ajout des RCFs au RCRCF
|
|
rcrcf.add_rcfs_a_aggreger(rcfs_a_aggreger)
|
|
|
|
# Mémoire la trajectoire RCRCF suivie par l'étudiant
|
|
nom_rcs = rcrcf.nom
|
|
self.rcrcfs_suivis[etudid][nom_rcs] = rcrcf
|
|
|
|
# Affichage pour debug
|
|
jeunes = list(enumerate(self.rcrcfs_suivis))
|
|
for no_etud, etudid in jeunes[:20]:
|
|
pe_affichage.pe_print(f"-> {etudiants.identites[etudid].nomprenom} :")
|
|
for nom_rcs, rcs in self.rcrcfs_suivis[etudid].items():
|
|
if rcs:
|
|
pe_affichage.pe_print(f" > RCRCF {nom_rcs}: {rcs.get_repr()}")
|
|
else:
|
|
pe_affichage.pe_print(f" > RCRCF {nom_rcs}: <vide> !!! ")
|
|
|
|
def get_rcs_etudiant(
|
|
semestres: dict[int:FormSemestre], formsemestre_final: FormSemestre, nom_rcs: str
|
|
) -> dict[int, FormSemestre]:
|
|
"""Ensemble des semestres parcourus (trajectoire)
|
|
par un étudiant dans le cadre
|
|
d'un RCS de type Sx, iA ou iS et ayant pour semestre terminal `formsemestre_final`.
|
|
|
|
Par ex: pour un RCS "3S", dont le formsemestre_terminal est un S3, regroupe
|
|
le ou les S1 qu'il a suivi (1 ou 2 si redoublement) + le ou les S2 + le ou les S3.
|
|
|
|
Les semestres parcourus sont antérieurs (en terme de date de fin)
|
|
au formsemestre_terminal.
|
|
|
|
Args:
|
|
cursus: Dictionnaire {fid: Formsemestre} donnant l'ensemble des semestres
|
|
dans lesquels l'étudiant a été inscrit
|
|
formsemestre_final: le semestre final visé
|
|
nom_rcs: Nom du RCS visé
|
|
"""
|
|
numero_semestre_terminal = formsemestre_final.semestre_id
|
|
# semestres_significatifs = self.get_semestres_significatifs(etudid)
|
|
semestres_significatifs = {}
|
|
for i in range(1, pe_comp.NBRE_SEMESTRES_DIPLOMANT + 1):
|
|
semestres_significatifs = semestres_significatifs | semestres[f"S{i}"]
|
|
|
|
if nom_rcs.startswith("S"): # les semestres
|
|
numero_semestres_possibles = [numero_semestre_terminal]
|
|
elif nom_rcs.endswith("A"): # les années
|
|
numero_semestres_possibles = [
|
|
int(sem[-1]) for sem in TYPES_RCS[nom_rcs]["aggregat"]
|
|
]
|
|
assert numero_semestre_terminal in numero_semestres_possibles
|
|
else: # les xS = tous les semestres jusqu'à Sx (eg S1, S2, S3 pour un S3 terminal)
|
|
numero_semestres_possibles = list(range(1, numero_semestre_terminal + 1))
|
|
|
|
semestres_aggreges = {}
|
|
for fid, semestre in semestres_significatifs.items():
|
|
# Semestres parmi ceux de n° possibles & qui lui sont antérieurs
|
|
if (
|
|
semestre.semestre_id in numero_semestres_possibles
|
|
and semestre.date_fin <= formsemestre_final.date_fin
|
|
):
|
|
semestres_aggreges[fid] = semestre
|
|
return semestres_aggreges
|