2020-09-26 16:19:37 +02:00
# -*- mode: python -*-
# -*- coding: utf-8 -*-
##############################################################################
#
# Gestion scolarite IUT
#
2023-01-02 13:16:27 +01:00
# Copyright (c) 1999 - 2023 Emmanuel Viennet. All rights reserved.
2020-09-26 16:19:37 +02:00
#
# 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
#
##############################################################################
##############################################################################
# Module "Avis de poursuite d'étude"
# conçu et développé par Cléo Baras (IUT de Grenoble)
##############################################################################
"""
Created on Fri Sep 9 09 : 15 : 05 2016
@author : barasc
"""
# ----------------------------------------------------------
# Ensemble des fonctions et des classes
# permettant les calculs preliminaires (hors affichage)
# a l'edition d'un jury de poursuites d'etudes
# ----------------------------------------------------------
2021-08-31 20:18:50 +02:00
import io
2021-02-01 23:54:46 +01:00
import os
2021-08-31 20:18:50 +02:00
from zipfile import ZipFile
2020-09-26 16:19:37 +02:00
2022-02-11 09:11:07 +01:00
from app . comp import res_sem
2022-03-27 22:25:00 +02:00
from app . comp . res_compat import NotesTableCompat
2023-02-18 00:13:00 +01:00
from app . models import Formation , FormSemestre
2022-02-11 09:11:07 +01:00
2021-06-19 23:21:37 +02:00
from app . scodoc . gen_tables import GenTable , SeqGenTable
import app . scodoc . sco_utils as scu
2023-02-12 13:36:47 +01:00
from app . scodoc import codes_cursus # codes_cursus.NEXT -> sem suivant
2021-06-21 10:17:16 +02:00
from app . scodoc import sco_etud
2021-06-21 11:22:55 +02:00
from app . scodoc import sco_formsemestre
2021-09-26 10:01:20 +02:00
from app . pe import pe_tagtable
from app . pe import pe_tools
from app . pe import pe_semestretag
from app . pe import pe_settag
2020-09-26 16:19:37 +02:00
# ----------------------------------------------------------------------------------------
2021-08-21 00:24:51 +02:00
def comp_nom_semestre_dans_parcours ( sem ) :
2020-09-26 16:19:37 +02:00
""" Le nom a afficher pour titrer un semestre
par exemple : " semestre 2 FI 2015 "
"""
2023-02-18 00:13:00 +01:00
formation : Formation = Formation . query . get_or_404 ( sem [ " formation_id " ] )
parcours = codes_cursus . get_cursus_from_code ( formation . type_parcours )
2020-09-26 16:19:37 +02:00
return " %s %s %s %s " % (
parcours . SESSION_NAME , # eg "semestre"
sem [ " semestre_id " ] , # eg 2
sem . get ( " modalite " , " " ) , # eg FI ou FC
sem [ " annee_debut " ] , # eg 2015
)
# ----------------------------------------------------------------------------------------
2021-07-09 23:31:16 +02:00
class JuryPE ( object ) :
2020-09-26 16:19:37 +02:00
""" Classe memorisant toutes les informations necessaires pour etablir un jury de PE. Modele
base sur NotesTable
Attributs : - diplome : l ' annee d ' obtention du diplome DUT et du jury de PE ( generalement fevrier XXXX )
- juryEtudDict : dictionnaire récapitulant les étudiants participant au jury PE ( données administratives +
celles des semestres valides à prendre en compte permettant le calcul des moyennes . . .
2021-02-13 17:28:55 +01:00
{ ' etudid : { ' nom ' , ' prenom ' , ' civilite ' , ' diplome ' , ' ' , }}
2020-09-26 16:19:37 +02:00
Rq : il contient à la fois les étudiants qui vont être diplomés à la date prévue
et ceux qui sont éliminés ( abandon , redoublement , . . . ) pour affichage alternatif
2021-01-01 18:40:47 +01:00
2020-09-26 16:19:37 +02:00
"""
# Variables de classe décrivant les aggrégats, leur ordre d'apparition temporelle et
# leur affichage dans les avis latex
PARCOURS = {
" S1 " : {
" aggregat " : [ " S1 " ] ,
" ordre " : 1 ,
" affichage_court " : " S1 " ,
" affichage_long " : " Semestre 1 " ,
} ,
" S2 " : {
" aggregat " : [ " S2 " ] ,
" ordre " : 2 ,
" affichage_court " : " S2 " ,
" affichage_long " : " Semestre 2 " ,
} ,
" S3 " : {
" aggregat " : [ " S3 " ] ,
" ordre " : 4 ,
" affichage_court " : " S3 " ,
" affichage_long " : " Semestre 3 " ,
} ,
" S4 " : {
" aggregat " : [ " S4 " ] ,
" ordre " : 5 ,
" affichage_court " : " S4 " ,
" affichage_long " : " Semestre 4 " ,
} ,
" 1A " : {
" aggregat " : [ " S1 " , " S2 " ] ,
" ordre " : 3 ,
" affichage_court " : " 1A " ,
" affichage_long " : " 1ère année " ,
} ,
" 2A " : {
" aggregat " : [ " S3 " , " S4 " ] ,
" ordre " : 6 ,
" affichage_court " : " 2A " ,
" affichage_long " : " 2ème année " ,
} ,
" 3S " : {
" aggregat " : [ " S1 " , " S2 " , " S3 " ] ,
" ordre " : 7 ,
" affichage_court " : " S1+S2+S3 " ,
" affichage_long " : " DUT du semestre 1 au semestre 3 " ,
} ,
" 4S " : {
" aggregat " : [ " S1 " , " S2 " , " S3 " , " S4 " ] ,
" ordre " : 8 ,
" affichage_court " : " DUT " ,
" affichage_long " : " DUT (tout semestre inclus) " ,
} ,
}
# ------------------------------------------------------------------------------------------------------------------
2021-08-21 00:24:51 +02:00
def __init__ ( self , semBase ) :
2020-09-26 16:19:37 +02:00
"""
Création d ' une table PE sur la base d ' un semestre selectionné . De ce semestre est déduit :
1. l ' année d ' obtention du DUT ,
2. tous les étudiants susceptibles à ce stade ( au regard de leur parcours ) d ' être diplomés.
Args :
semBase : le dictionnaire sem donnant la base du jury
meme_programme : si True , impose un même programme pour tous les étudiants participant au jury ,
si False , permet des programmes differents
"""
self . semTagDict = (
{ }
) # Les semestres taggués à la base des calculs de moyenne par tag
self . setTagDict = (
{ }
) # dictionnaire récapitulant les semTag impliqués dans le jury de la forme { 'formsemestre_id' : object Semestre_tag
self . promoTagDict = { }
# L'année du diplome
self . diplome = get_annee_diplome_semestre ( semBase )
# Un zip où ranger les fichiers générés:
self . NOM_EXPORT_ZIP = " Jury_PE_ %s " % self . diplome
2021-08-31 20:18:50 +02:00
self . zipdata = io . BytesIO ( )
2020-09-26 16:19:37 +02:00
self . zipfile = ZipFile ( self . zipdata , " w " )
#
self . ETUDINFO_DICT = { } # Les infos sur les étudiants
self . PARCOURSINFO_DICT = { } # Les parcours des étudiants
self . syntheseJury = { } # Le jury de synthèse
2022-02-11 23:22:07 +01:00
self . semestresDeScoDoc = sco_formsemestre . do_formsemestre_list ( )
2020-09-26 16:19:37 +02:00
# Calcul du jury PE
self . exe_calculs_juryPE ( semBase )
self . synthetise_juryPE ( )
# Export des données => mode 1 seule feuille -> supprimé
# filename = self.NOM_EXPORT_ZIP + "jurySyntheseDict_" + str(self.diplome) + '.xls'
# self.xls = self.table_syntheseJury(mode="singlesheet")
# self.add_file_to_zip(filename, self.xls.excel())
# Fabrique 1 fichier excel résultat avec 1 seule feuille => trop gros
2021-08-12 14:49:53 +02:00
filename = self . NOM_EXPORT_ZIP + " _jurySyntheseDict " + scu . XLSX_SUFFIX
2020-09-26 16:19:37 +02:00
self . xlsV2 = self . table_syntheseJury ( mode = " multiplesheet " )
if self . xlsV2 :
self . add_file_to_zip ( filename , self . xlsV2 . excel ( ) )
# Pour debug
# self.syntheseJury = pe_tools.JURY_SYNTHESE_POUR_DEBUG #Un dictionnaire fictif pour debug
# ------------------------------------------------------------------------------------------------------------------
def add_file_to_zip ( self , filename , data , path = " " ) :
""" Add a file to our zip
All files under NOM_EXPORT_ZIP /
path may specify a subdirectory
"""
path_in_zip = os . path . join ( self . NOM_EXPORT_ZIP , path , filename )
self . zipfile . writestr ( path_in_zip , data )
# ------------------------------------------------------------------------------------------------------------------
def get_zipped_data ( self ) :
2021-08-31 20:18:50 +02:00
""" returns file-like data with a zip of all generated (CSV) files.
Reset file cursor at the beginning !
"""
2020-09-26 16:19:37 +02:00
if self . zipfile :
self . zipfile . close ( )
self . zipfile = None
2021-08-31 20:18:50 +02:00
self . zipdata . seek ( 0 )
return self . zipdata
2020-09-26 16:19:37 +02:00
# **************************************************************************************************************** #
# Lancement des différentes actions permettant le calcul du jury PE
# **************************************************************************************************************** #
def exe_calculs_juryPE ( self , semBase ) :
# Liste des étudiants à traiter pour identifier ceux qui seront diplômés
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
" *** Recherche et chargement des étudiants diplômés en %d "
% ( self . diplome )
)
self . get_etudiants_in_jury (
semBase , avec_meme_formation = False
) # calcul des coSemestres
# Les semestres impliqués (ceux valides pour les étudiants à traiter)
# -------------------------------------------------------------------
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " *** Création des semestres taggués " )
self . get_semtags_in_jury ( )
if pe_tools . PE_DEBUG :
for semtag in self . semTagDict . values ( ) : # Export
filename = self . NOM_EXPORT_ZIP + semtag . nom + " .csv "
self . zipfile . writestr ( filename , semtag . str_tagtable ( ) )
# self.export_juryPEDict()
# Les moyennes sur toute la scolarité
# -----------------------------------
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
" *** Création des moyennes sur différentes combinaisons de semestres et différents groupes d ' étudiant "
)
self . get_settags_in_jury ( )
if pe_tools . PE_DEBUG :
for settagdict in self . setTagDict . values ( ) : # Export
for settag in settagdict . values ( ) :
filename = self . NOM_EXPORT_ZIP + semtag . nom + " .csv "
self . zipfile . writestr ( filename , semtag . str_tagtable ( ) )
# self.export_juryPEDict()
# Les interclassements
# --------------------
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
" *** Création des interclassements au sein de la promo sur différentes combinaisons de semestres "
)
self . get_promotags_in_jury ( )
# **************************************************************************************************************** #
# Fonctions relatives à la liste des étudiants à prendre en compte dans le jury
# **************************************************************************************************************** #
# ------------------------------------------------------------------------------------------------------------------
def get_etudiants_in_jury ( self , semBase , avec_meme_formation = False ) :
"""
Calcule la liste des étudiants à prendre en compte dans le jury et la renvoie sous la forme
"""
# Les cosemestres donnant lieu à meme année de diplome
coSems = get_cosemestres_diplomants (
2021-08-21 00:24:51 +02:00
semBase , avec_meme_formation = avec_meme_formation
2020-09-26 16:19:37 +02:00
) # calcul des coSemestres
if p e_tools . PE_DEBUG :
pe_tools . pe_print (
" 1) Recherche des coSemestres -> %d trouvés " % len ( coSems )
)
# Les étudiants inscrits dans les cosemestres
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " 2) Liste des étudiants dans les différents co-semestres " )
listEtudId = self . get_etudiants_dans_semestres (
coSems
) # étudiants faisant parti des cosemestres
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " => %d étudiants trouvés " % len ( listEtudId ) )
# L'analyse des parcours étudiants pour déterminer leur année effective de diplome avec prise en compte des redoublements, des abandons, ....
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " 3) Analyse des parcours individuels des étudiants " )
for ( no_etud , etudid ) in enumerate ( listEtudId ) :
self . add_etudiants ( etudid )
if pe_tools . PE_DEBUG :
if ( no_etud + 1 ) % 10 == 0 :
pe_tools . pe_print ( ( no_etud + 1 ) , " " , end = " " )
pe_tools . pe_print ( )
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
" => %d étudiants à diplômer en %d "
% ( len ( self . get_etudids_du_jury ( ) ) , self . diplome )
)
pe_tools . pe_print (
" => %d étudiants éliminer pour abandon "
% ( len ( listEtudId ) - len ( self . get_etudids_du_jury ( ) ) )
)
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
def get_etudiants_dans_semestres ( self , semsListe ) :
""" Renvoie la liste des etudid des etudiants inscrits à l ' un des semestres de la liste fournie en paramètre
en supprimant les doublons ( i . e . un même étudiant qui apparaîtra 2 fois ) """
etudiants = [ ]
for sem in semsListe : # pour chacun des semestres de la liste
2021-08-21 00:24:51 +02:00
nt = self . get_cache_notes_d_un_semestre ( sem [ " formsemestre_id " ] )
2020-09-26 16:19:37 +02:00
etudiantsDuSemestre = (
nt . get_etudids ( )
2022-02-13 23:53:11 +01:00
) # identification des etudiants du semestre
2020-09-26 16:19:37 +02:00
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
" --> chargement du semestre %s : %d etudiants "
% ( sem [ " formsemestre_id " ] , len ( etudiantsDuSemestre ) )
)
etudiants . extend ( etudiantsDuSemestre )
return list ( set ( etudiants ) ) # suppression des doublons
# ------------------------------------------------------------------------------------------------------------------
def get_etudids_du_jury ( self , ordre = " aucun " ) :
""" Renvoie la liste de tous les étudiants (concrètement leur etudid)
participant au jury c ' est à dire, ceux dont la date du ' jury ' est self.diplome
et n ' ayant pas abandonné.
Si l ' ordre est précisé, donne une liste etudid dont le nom, prenom trié par ordre alphabétique
"""
etudids = [
etudid
for ( etudid , donnees ) in self . PARCOURSINFO_DICT . items ( )
if donnees [ " diplome " ] == self . diplome and donnees [ " abandon " ] == False
]
if ordre == " alphabetique " : # Tri alphabétique
etudidsAvecNom = [
( etudid , etud [ " nom " ] + " / " + etud [ " prenom " ] )
for ( etudid , etud ) in self . PARCOURSINFO_DICT . items ( )
if etudid in etudids
]
etudidsAvecNomTrie = sorted ( etudidsAvecNom , key = lambda col : col [ 1 ] )
etudids = [ etud [ 0 ] for etud in etudidsAvecNomTrie ]
return etudids
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
def add_etudiants ( self , etudid ) :
2021-01-01 18:40:47 +01:00
""" Ajoute un étudiant (via son etudid) au dictionnaire de synthèse jurydict.
2020-09-26 16:19:37 +02:00
L ' ajout consiste à :
> insérer une entrée pour l ' étudiant en mémorisant ses infos (get_etudInfo),
avec son nom , prénom , etc . . .
> à analyser son parcours , pour vérifier s ' il n ' a pas abandonné l ' IUT en cours de route => clé abandon
> à chercher ses semestres valides ( formsemestre_id ) et ses années valides ( formannee_id ) ,
c ' est à dire ceux pour lesquels il faudra prendre en compte ses notes dans les calculs de moyenne (type 1A=S1+S2/2)
"""
2021-07-09 17:47:06 +02:00
if etudid not in self . PARCOURSINFO_DICT :
2020-09-26 16:19:37 +02:00
etud = self . get_cache_etudInfo_d_un_etudiant (
2021-08-21 00:24:51 +02:00
etudid
2020-09-26 16:19:37 +02:00
) # On charge les données de l'étudiant
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
pe_tools . pe_print ( etud [ " nom " ] + " " + etud [ " prenom " ] , end = " " )
self . PARCOURSINFO_DICT [ etudid ] = {
" etudid " : etudid , # les infos sur l'étudiant
" nom " : etud [ " nom " ] , # Ajout à la table jury
}
# Analyse du parcours de l'étudiant
# Sa date prévisionnelle de diplome
self . PARCOURSINFO_DICT [ etudid ] [
" diplome "
] = self . calcul_anneePromoDUT_d_un_etudiant ( etudid )
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
pe_tools . pe_print (
" promo= " + str ( self . PARCOURSINFO_DICT [ etudid ] [ " diplome " ] ) , end = " "
)
# Est-il réorienté ou démissionnaire ?
self . PARCOURSINFO_DICT [ etudid ] [
" abandon "
] = self . est_un_etudiant_reoriente_ou_demissionnaire ( etudid )
# A-t-il arrêté de lui-même sa formation avant la fin ?
etatD = self . est_un_etudiant_disparu ( etudid )
if etatD == True :
self . PARCOURSINFO_DICT [ etudid ] [ " abandon " ] = True
# dans le jury ne seront traités que les étudiants ayant la date attendue de diplome et n'ayant pas abandonné
# Quels sont ses semestres validant (i.e ceux dont les notes doivent être prises en compte pour le jury)
# et s'ils existent quelles sont ses notes utiles ?
sesFormsemestre_idValidants = [
self . get_Fid_d_un_Si_valide_d_un_etudiant ( etudid , nom_sem )
for nom_sem in JuryPE . PARCOURS [ " 4S " ] [
" aggregat "
] # Recherche du formsemestre_id de son Si valide (ou a défaut en cours)
]
for ( i , nom_sem ) in enumerate ( JuryPE . PARCOURS [ " 4S " ] [ " aggregat " ] ) :
fid = sesFormsemestre_idValidants [ i ]
self . PARCOURSINFO_DICT [ etudid ] [ nom_sem ] = fid # ['formsemestre_id']
if fid != None and pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
2021-08-10 17:12:10 +02:00
pe_tools . pe_print ( nom_sem + " = " + str ( fid ) , end = " " )
2020-09-26 16:19:37 +02:00
# self.get_moyennesEtClassements_par_semestre_d_un_etudiant( etudid, fid )
# Quelles sont ses années validantes ('1A', '2A') et ses parcours (3S, 4S) validants ?
for parcours in [ " 1A " , " 2A " , " 3S " , " 4S " ] :
lesSemsDuParcours = JuryPE . PARCOURS [ parcours ] [
" aggregat "
] # les semestres du parcours : par ex. ['S1', 'S2', 'S3']
lesFidsValidantDuParcours = [
sesFormsemestre_idValidants [
JuryPE . PARCOURS [ " 4S " ] [ " aggregat " ] . index ( nom_sem )
]
for nom_sem in lesSemsDuParcours # par ex. ['SEM4532', 'SEM567', ...]
]
parcours_incomplet = (
sum ( [ fid == None for fid in lesFidsValidantDuParcours ] ) > 0
)
if not parcours_incomplet :
self . PARCOURSINFO_DICT [ etudid ] [
parcours
] = lesFidsValidantDuParcours [ - 1 ]
else :
self . PARCOURSINFO_DICT [ etudid ] [ parcours ] = None
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
pe_tools . pe_print (
parcours + " = " + str ( self . PARCOURSINFO_DICT [ etudid ] [ parcours ] ) ,
end = " " ,
)
2021-02-01 23:54:46 +01:00
# if pe_tools.PE_DEBUG and pe_tools.PE_DEBUG >= 2:
# print
2020-09-26 16:19:37 +02:00
# ------------------------------------------------------------------------------------------------------------------
def est_un_etudiant_reoriente_ou_demissionnaire ( self , etudid ) :
""" Renvoie True si l ' étudiant est réorienté (NAR) ou démissionnaire (DEM) """
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_report
2020-09-26 16:19:37 +02:00
reponse = False
2021-08-21 00:24:51 +02:00
etud = self . get_cache_etudInfo_d_un_etudiant ( etudid )
2023-02-12 13:36:47 +01:00
( _ , parcours ) = sco_report . get_code_cursus_etud ( etud )
2020-09-26 16:19:37 +02:00
if (
2023-02-12 13:36:47 +01:00
len ( codes_cursus . CODES_SEM_REO & set ( parcours . values ( ) ) ) > 0
2020-09-26 16:19:37 +02:00
) : # Eliminé car NAR apparait dans le parcours
reponse = True
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
pe_tools . pe_print ( " -> à éliminer car réorienté (NAR) " )
2021-07-09 17:47:06 +02:00
if " DEM " in list ( parcours . values ( ) ) : # Eliminé car DEM
2020-09-26 16:19:37 +02:00
reponse = True
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 2 :
pe_tools . pe_print ( " -> à éliminer car DEM " )
return reponse
# ------------------------------------------------------------------------------------------------------------------
def est_un_etudiant_disparu ( self , etudid ) :
""" Renvoie True si l ' étudiant n ' a pas achevé la formation à l ' IUT et a disparu des listes, sans
pour autant avoir été indiqué NAR ou DEM ; recherche son dernier semestre validé et regarde s ' il
n ' existe pas parmi les semestres existants dans scodoc un semestre postérieur (en terme de date de
début ) de n ° au moins égal à celui de son dernier semestre valide dans lequel il aurait pu
2021-01-01 18:40:47 +01:00
s ' inscrire mais ne l ' a pas fait . """
2020-09-26 16:19:37 +02:00
sessems = self . get_semestresDUT_d_un_etudiant (
etudid
) # les semestres de l'étudiant
sonDernierSidValide = self . get_dernier_semestre_id_valide_d_un_etudiant ( etudid )
sesdates = [
pe_tagtable . conversionDate_StrToDate ( sem [ " date_fin " ] ) for sem in sessems
] # association 1 date -> 1 semestrePE pour les semestres de l'étudiant
2022-04-20 12:23:43 +02:00
if sesdates :
lastdate = max ( sesdates ) # date de fin de l'inscription la plus récente
else :
return False
2020-09-26 16:19:37 +02:00
# if PETable.AFFICHAGE_DEBUG_PE == True : pe_tools.pe_print(" derniere inscription = ", lastDateSem)
2022-02-11 23:22:07 +01:00
2021-08-29 23:04:38 +02:00
if sonDernierSidValide is None :
# si l'étudiant n'a validé aucun semestre, les prend tous ? (à vérifier)
2022-02-11 23:22:07 +01:00
semestresSuperieurs = self . semestresDeScoDoc
2021-08-29 23:04:38 +02:00
else :
semestresSuperieurs = [
sem
2022-02-11 23:22:07 +01:00
for sem in self . semestresDeScoDoc
2021-08-29 23:04:38 +02:00
if sem [ " semestre_id " ] > sonDernierSidValide
] # Semestre de rang plus élevé que son dernier sem valide
2020-09-26 16:19:37 +02:00
datesDesSemestresSuperieurs = [
pe_tagtable . conversionDate_StrToDate ( sem [ " date_debut " ] )
for sem in semestresSuperieurs
]
datesDesSemestresPossibles = [
date_deb for date_deb in datesDesSemestresSuperieurs if date_deb > = lastdate
] # date de debut des semestres possibles postérieur au dernier semestre de l'étudiant et de niveau plus élevé que le dernier semestre valide de l'étudiant
if (
len ( datesDesSemestresPossibles ) > 0
) : # etudiant ayant disparu de la circulation
# if PETable.AFFICHAGE_DEBUG_PE == True :
# pe_tools.pe_print(" -> à éliminer car des semestres où il aurait pu s'inscrire existent ")
# pe_tools.pe_print(pe_tools.print_semestres_description( datesDesSemestresPossibles.values() ))
return True
else :
return False
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
def get_dernier_semestre_id_valide_d_un_etudiant ( self , etudid ) :
""" Renvoie le n° (semestre_id) du dernier semestre validé par un étudiant fourni par son etudid
et None si aucun semestre n ' a été validé
"""
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_report
2021-08-21 00:24:51 +02:00
etud = self . get_cache_etudInfo_d_un_etudiant ( etudid )
2023-02-12 13:36:47 +01:00
( code , parcours ) = sco_report . get_code_cursus_etud (
2021-08-21 00:24:51 +02:00
etud
2020-09-26 16:19:37 +02:00
) # description = '1234:A', parcours = {1:ADM, 2:NAR, ...}
sonDernierSemestreValide = max (
[
int ( cle )
for ( cle , code ) in parcours . items ( )
2023-02-12 13:36:47 +01:00
if code in codes_cursus . CODES_SEM_VALIDES
2020-09-26 16:19:37 +02:00
]
+ [ 0 ]
) # n° du dernier semestre valide, 0 sinon
return sonDernierSemestreValide if sonDernierSemestreValide > 0 else None
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
def get_Fid_d_un_Si_valide_d_un_etudiant ( self , etudid , nom_semestre ) :
""" Récupère le formsemestre_id valide d ' un étudiant fourni son etudid à un semestre DUT de n° semestre_id
donné . Si le semestre est en cours ( pas encore de jury ) , renvoie le formsemestre_id actuel . """
semestre_id = JuryPE . PARCOURS [ " 4S " ] [ " aggregat " ] . index ( nom_semestre ) + 1
sesSi = self . get_semestresDUT_d_un_etudiant (
etudid , semestre_id
) # extrait uniquement les Si par ordre temporel décroissant
if len ( sesSi ) > 0 : # S'il a obtenu au moins une note
# mT = sesMoyennes[0]
leFid = sesSi [ 0 ] [ " formsemestre_id " ]
for ( i , sem ) in enumerate (
sesSi
) : # Parcours des éventuels semestres précédents
2021-08-21 00:24:51 +02:00
nt = self . get_cache_notes_d_un_semestre ( sem [ " formsemestre_id " ] )
2020-09-26 16:19:37 +02:00
dec = nt . get_etud_decision_sem (
etudid
) # quelle est la décision du jury ?
2023-02-12 13:36:47 +01:00
if dec and ( dec [ " code " ] in codes_cursus . CODES_SEM_VALIDES ) :
2023-01-23 11:38:47 +01:00
# isinstance( sesMoyennes[i+1], float) and
2020-09-26 16:19:37 +02:00
# mT = sesMoyennes[i+1] # substitue la moyenne si le semestre suivant est "valide"
leFid = sem [ " formsemestre_id " ]
else :
leFid = None
return leFid
# **************************************************************************************************************** #
# Traitements des semestres impliqués dans le jury
# **************************************************************************************************************** #
# ------------------------------------------------------------------------------------------------------------------
def get_semtags_in_jury ( self ) :
"""
Créé les semestres tagués relatifs aux résultats des étudiants à prendre en compte dans le jury .
Calcule les moyennes et les classements de chaque semestre par tag et les statistiques de ces semestres .
"""
lesFids = self . get_formsemestreids_du_jury (
self . get_etudids_du_jury ( ) , liste_semestres = [ " S1 " , " S2 " , " S3 " , " S4 " ]
)
for ( i , fid ) in enumerate ( lesFids ) :
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" %d ) Semestre taggué %s (avec classement dans groupe) "
2020-09-26 16:19:37 +02:00
% ( i + 1 , fid )
)
self . add_semtags_in_jury ( fid )
# ------------------------------------------------------------------------------------------------------------------
def add_semtags_in_jury ( self , fid ) :
""" Crée si nécessaire un semtag et le mémorise dans self.semTag ;
charge également les données des nouveaux étudiants qui en font partis .
"""
# Semestre taggué avec classement dans le groupe
2021-07-09 17:47:06 +02:00
if fid not in self . semTagDict :
2021-08-21 00:24:51 +02:00
nt = self . get_cache_notes_d_un_semestre ( fid )
2020-09-26 16:19:37 +02:00
# Création du semestres
self . semTagDict [ fid ] = pe_semestretag . SemestreTag (
2021-08-21 00:24:51 +02:00
nt , nt . sem
2020-09-26 16:19:37 +02:00
) # Création du pesemestre associé
self . semTagDict [ fid ] . comp_data_semtag ( )
lesEtudids = self . semTagDict [ fid ] . get_etudids ( )
lesEtudidsManquants = [ ]
for etudid in lesEtudids :
if (
etudid not in self . PARCOURSINFO_DICT
) : # Si l'étudiant n'a pas été pris en compte dans le jury car déjà diplômé ou redoublant
lesEtudidsManquants . append ( etudid )
2021-08-21 00:24:51 +02:00
# self.get_cache_etudInfo_d_un_etudiant(etudid)
2020-09-26 16:19:37 +02:00
self . add_etudiants (
etudid
) # Ajoute les élements de parcours de l'étudiant
nbinscrit = self . semTagDict [ fid ] . get_nbinscrits ( )
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" - %d étudiants classés " % ( nbinscrit )
2020-09-26 16:19:37 +02:00
+ " : "
+ " , " . join (
[ etudid for etudid in self . semTagDict [ fid ] . get_etudids ( ) ]
)
)
if lesEtudidsManquants :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" - dont %d étudiants manquants ajoutés aux données du jury "
2020-09-26 16:19:37 +02:00
% ( len ( lesEtudidsManquants ) )
+ " : "
+ " , " . join ( lesEtudidsManquants )
)
2022-04-20 12:23:43 +02:00
pe_tools . pe_print ( " - Export csv " )
2020-09-26 16:19:37 +02:00
filename = self . NOM_EXPORT_ZIP + self . semTagDict [ fid ] . nom + " .csv "
self . zipfile . writestr ( filename , self . semTagDict [ fid ] . str_tagtable ( ) )
# ----------------------------------------------------------------------------------------------------------------
def get_formsemestreids_du_jury ( self , etudids , liste_semestres = " 4S " ) :
""" Renvoie la liste des formsemestre_id validants des étudiants en parcourant les semestres valides des étudiants mémorisés dans
self . PARCOURSINFO_DICT .
Les étudiants sont identifiés par leur etudic donnés dans la liste etudids ( généralement self . get_etudids_in_jury ( ) ) .
La liste_semestres peut être une liste ou une chaine de caractères parmi :
* None = > tous les Fids validant
* ' Si ' = > le ième 1 semestre
* ' iA ' = > l ' année i = [ ' S1 , ' S2 ' ] ou [ ' S3 ' , ' S4 ' ]
* ' 3S ' , ' 4S ' = > fusion des semestres
* [ ' Si ' , ' iA ' , . . . ] = > une liste combinant les formats précédents
"""
2021-07-09 17:47:06 +02:00
champs_possibles = list ( JuryPE . PARCOURS . keys ( ) )
2020-09-26 16:19:37 +02:00
if (
not isinstance ( liste_semestres , list )
and not isinstance ( liste_semestres , str )
and liste_semestres not in champs_possibles
) :
raise ValueError (
" Probleme de paramètres d ' appel dans pe_jurype.JuryPE.get_formsemestreids_du_jury "
)
if isinstance ( liste_semestres , list ) :
res = [ ]
for elmt in liste_semestres :
res . extend ( self . get_formsemestreids_du_jury ( etudids , elmt ) )
return list ( set ( res ) )
# si liste_sem est un nom de parcours
nom_sem = liste_semestres
# if nom_sem in ['1A', '2A', '3S', '4S'] :
# return self.get_formsemestreids_du_jury(etudids, JuryPE.PARCOURS[nom_sem] )
# else :
fids = {
self . PARCOURSINFO_DICT [ etudid ] [ nom_sem ]
for etudid in etudids
if self . PARCOURSINFO_DICT [ etudid ] [ nom_sem ] != None
}
return list ( fids )
# **************************************************************************************************************** #
# Traitements des parcours impliquées dans le jury
# **************************************************************************************************************** #
# # ----------------------------------------------------------------------------------------------------------------
# def get_antags_in_jury(self, avec_affichage_debug=True ):
# """Construit les settag associés aux années 1A et 2A du jury"""
# lesAnnees = {'1A' : ['S1', 'S2'], '2A' : ['S3', 'S4'] }
# for nom_annee in lesAnnees:
# lesAidDesAnnees = self.get_anneeids_du_jury(annee= nom_annee) # les annee_ids des étudiants du jury
# for aid in lesAidDesAnnees:
# fidSemTagFinal = JuryPE.convert_aid_en_fid( aid )
# lesEtudisDelAnnee = self.semTagDict[ fidSemTagFinal ].get_etudids() # les etudiants sont ceux inscrits dans le semestre final de l'année
# parcoursDesEtudiants = { etudid : self.PARCOURSINFO_DICT[etudid] for etudid in lesEtudisDelAnnee } # les parcours des etudid aka quels semestres sont à prendre en compte
#
# lesFidsDesEtudiants = self.get_formsemestreids_du_jury(lesEtudisDelAnnee, nom_annee) # les formsemestres_id à prendre en compte pour les moyennes
# # Manque-t-il des semtag associés ; si oui, les créé
# pe_tools.pe_print(aid, lesFidsDesEtudiants)
# for fid in lesFidsDesEtudiants:
# self.add_semtags_in_jury(fid, avec_affichage_debug=avec_affichage_debug)
# lesSemTagDesEtudiants = { fid: self.semTagDict[fid] for fid in lesFidsDesEtudiants }
#
# # Tous les semtag nécessaires pour ses étudiants avec ajout éventuel s'ils n'ont pas été chargés
# pe_tools.pe_print(" -> Création de l'année tagguée " + str( aid ))
# #settag_id, short_name, listeEtudId, groupe, listeSemAAggreger, ParcoursEtudDict, SemTagDict, with_comp_moy=True)
# self.anTagDict[ aid ] = pe_settag.SetTag( aid, "Annee " + self.semTagDict[fidSemTagFinal].short_name, \
# lesEtudisDelAnnee, 'groupe', lesAnnees[ nom_annee ], parcoursDesEtudiants, lesSemTagDesEtudiants )
# self.anTagDict[ aid ].comp_data_settag() # calcul les moyennes
# **************************************************************************************************************** #
# Traitements des moyennes sur différentes combinaisons de parcours 1A, 2A, 3S et 4S,
# impliquées dans le jury
# **************************************************************************************************************** #
def get_settags_in_jury ( self ) :
""" Calcule les moyennes sur la totalité du parcours (S1 jusqu ' à S3 ou S4)
en classant les étudiants au sein du semestre final du parcours ( même S3 , même S4 , . . . ) """
# Par groupe :
# combinaisons = { 'S1' : ['S1'], 'S2' : ['S2'], 'S3' : ['S3'], 'S4' : ['S4'], \
# '1A' : ['S1', 'S2'], '2A' : ['S3', 'S4'],
# '3S' : ['S1', 'S2', 'S3'], '4S' : ['S1', 'S2', 'S3', 'S4'] }
# ---> sur 2 parcours DUT (cas S3 fini, cas S4 fini)
combinaisons = [ " 1A " , " 2A " , " 3S " , " 4S " ]
for ( i , nom ) in enumerate ( combinaisons ) :
parcours = JuryPE . PARCOURS [ nom ] [
" aggregat "
] # La liste des noms de semestres (S1, S2, ...) impliqués dans l'aggrégat
# Recherche des parcours possibles par le biais de leur Fid final
fids_finaux = self . get_formsemestreids_du_jury (
self . get_etudids_du_jury ( ) , nom
) # les formsemestre_ids validant finaux des étudiants du jury
if len ( fids_finaux ) > 0 : # S'il existe des parcours validant
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 1 :
pe_tools . pe_print ( " %d ) Fusion %s avec " % ( i + 1 , nom ) )
if nom not in self . setTagDict :
self . setTagDict [ nom ] = { }
for fid in fids_finaux :
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 1 :
2022-04-20 12:23:43 +02:00
pe_tools . pe_print ( " - semestre final %s " % ( fid ) )
2020-09-26 16:19:37 +02:00
settag = pe_settag . SetTag (
nom , parcours = parcours
) # Le set tag fusionnant les données
etudiants = self . semTagDict [
fid
] . get_etudids ( ) # Les étudiants du sem final
# ajoute les étudiants au semestre
settag . set_Etudiants (
etudiants ,
self . PARCOURSINFO_DICT ,
self . ETUDINFO_DICT ,
nom_sem_final = self . semTagDict [ fid ] . nom ,
)
# manque-t-il des semestres ? Si oui, les ajoute au jurype puis au settag
for ffid in settag . get_Fids_in_settag ( ) :
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 1 :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" -> ajout du semestre tagué %s " % ( ffid )
2020-09-26 16:19:37 +02:00
)
self . add_semtags_in_jury ( ffid )
settag . set_SemTagDict (
self . semTagDict
) # ajoute les semestres au settag
settag . comp_data_settag ( ) # Calcul les moyennes, les rangs, ..
self . setTagDict [ nom ] [ fid ] = settag # Mémorise le résultat
else :
if pe_tools . PE_DEBUG and pe_tools . PE_DEBUG > = 1 :
pe_tools . pe_print ( " %d ) Pas de fusion %s possible " % ( i + 1 , nom ) )
def get_promotags_in_jury ( self ) :
""" Calcule les aggrégats en interclassant les étudiants du jury (les moyennes ont déjà été calculées en amont) """
lesEtudids = self . get_etudids_du_jury ( )
for ( i , nom ) in enumerate ( JuryPE . PARCOURS . keys ( ) ) :
settag = pe_settag . SetTagInterClasse ( nom , diplome = self . diplome )
nbreEtudInscrits = settag . set_Etudiants (
lesEtudids , self . PARCOURSINFO_DICT , self . ETUDINFO_DICT
)
if nbreEtudInscrits > 0 :
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" %d ) %s avec interclassement sur la promo " % ( i + 1 , nom )
2020-09-26 16:19:37 +02:00
)
if nom in [ " S1 " , " S2 " , " S3 " , " S4 " ] :
settag . set_SetTagDict ( self . semTagDict )
else : # cas des aggrégats
settag . set_SetTagDict ( self . setTagDict [ nom ] )
settag . comp_data_settag ( )
self . promoTagDict [ nom ] = settag
else :
if pe_tools . PE_DEBUG :
pe_tools . pe_print (
2022-04-20 12:23:43 +02:00
" %d ) Pas d ' interclassement %s sur la promo faute de notes "
2020-09-26 16:19:37 +02:00
% ( i + 1 , nom )
)
# **************************************************************************************************************** #
# Méthodes pour la synthèse du juryPE
# *****************************************************************************************************************
def synthetise_juryPE ( self ) :
""" Synthétise tous les résultats du jury PE dans un dictionnaire """
self . syntheseJury = { }
for etudid in self . get_etudids_du_jury ( ) :
etudinfo = self . ETUDINFO_DICT [ etudid ]
self . syntheseJury [ etudid ] = {
" nom " : etudinfo [ " nom " ] ,
" prenom " : etudinfo [ " prenom " ] ,
2021-02-13 17:28:55 +01:00
" civilite " : etudinfo [ " civilite " ] ,
2021-02-16 15:18:06 +01:00
" civilite_str " : etudinfo [ " civilite_str " ] ,
2020-09-26 16:19:37 +02:00
" age " : str ( pe_tools . calcul_age ( etudinfo [ " date_naissance " ] ) ) ,
" lycee " : etudinfo [ " nomlycee " ]
+ (
" ( " + etudinfo [ " villelycee " ] + " ) "
if etudinfo [ " villelycee " ] != " "
else " "
) ,
" bac " : etudinfo [ " bac " ] ,
2022-07-29 16:19:40 +02:00
" code_nip " : etudinfo [ " code_nip " ] , # pour la photo
2020-09-26 16:19:37 +02:00
" entree " : self . get_dateEntree ( etudid ) ,
" promo " : self . diplome ,
}
# Le parcours
self . syntheseJury [ etudid ] [ " parcours " ] = self . get_parcoursIUT (
etudid
) # liste des semestres
self . syntheseJury [ etudid ] [ " nbSemestres " ] = len (
self . syntheseJury [ etudid ] [ " parcours " ]
) # nombre de semestres
# Ses résultats
for nom in JuryPE . PARCOURS : # S1, puis S2, puis 1A
# dans le groupe : la table tagguée dans les semtag ou les settag si aggrégat
self . syntheseJury [ etudid ] [ nom ] = { " groupe " : { } , " promo " : { } }
if (
self . PARCOURSINFO_DICT [ etudid ] [ nom ] != None
) : # Un parcours valide existe
if nom in [ " S1 " , " S2 " , " S3 " , " S4 " ] :
tagtable = self . semTagDict [ self . PARCOURSINFO_DICT [ etudid ] [ nom ] ]
else :
tagtable = self . setTagDict [ nom ] [
self . PARCOURSINFO_DICT [ etudid ] [ nom ]
]
for tag in tagtable . get_all_tags ( ) :
self . syntheseJury [ etudid ] [ nom ] [ " groupe " ] [
tag
] = tagtable . get_resultatsEtud (
tag , etudid
) # Le tuple des résultats
# interclassé dans la promo
tagtable = self . promoTagDict [ nom ]
for tag in tagtable . get_all_tags ( ) :
self . syntheseJury [ etudid ] [ nom ] [ " promo " ] [
tag
] = tagtable . get_resultatsEtud ( tag , etudid )
def get_dateEntree ( self , etudid ) :
""" Renvoie l ' année d ' entrée de l ' étudiant à l ' IUT """
2021-02-01 23:54:46 +01:00
# etudinfo = self.ETUDINFO_DICT[etudid]
2022-08-30 11:09:13 +02:00
semestres = self . get_semestresDUT_d_un_etudiant ( etudid )
if semestres :
# le 1er sem à l'IUT
2022-09-08 09:48:19 +02:00
return semestres [ 0 ] [ " annee_debut " ]
2022-08-30 11:09:13 +02:00
else :
return " "
2020-09-26 16:19:37 +02:00
def get_parcoursIUT ( self , etudid ) :
2021-01-01 18:40:47 +01:00
""" Renvoie une liste d ' infos sur les semestres du parcours d ' un étudiant """
2021-02-01 23:54:46 +01:00
# etudinfo = self.ETUDINFO_DICT[etudid]
2020-09-26 16:19:37 +02:00
sems = self . get_semestresDUT_d_un_etudiant ( etudid )
infos = [ ]
for sem in sems :
2021-08-21 00:24:51 +02:00
nomsem = comp_nom_semestre_dans_parcours ( sem )
2020-09-26 16:19:37 +02:00
infos . append (
{
" nom_semestre_dans_parcours " : nomsem ,
" titreannee " : sem [ " titreannee " ] ,
" formsemestre_id " : sem [ " formsemestre_id " ] , # utile dans le futur ?
}
)
return infos
# **************************************************************************************************************** #
# Méthodes d'affichage pour debug
# **************************************************************************************************************** #
def str_etudiants_in_jury ( self , delim = " ; " ) :
# En tete:
entete = [ " Id " , " Nom " , " Abandon " , " Diplome " ]
for nom_sem in [ " S1 " , " S2 " , " S3 " , " S4 " , " 1A " , " 2A " , " 3S " , " 4S " ] :
entete + = [ nom_sem , " descr " ]
chaine = delim . join ( entete ) + " \n "
for etudid in self . PARCOURSINFO_DICT :
donnees = self . PARCOURSINFO_DICT [ etudid ]
# pe_tools.pe_print(etudid, donnees)
# les infos générales
descr = [
etudid ,
donnees [ " nom " ] ,
str ( donnees [ " abandon " ] ) ,
str ( donnees [ " diplome " ] ) ,
]
# les semestres
for nom_sem in [ " S1 " , " S2 " , " S3 " , " S4 " , " 1A " , " 2A " , " 3S " , " 4S " ] :
table = (
self . semTagDict [ donnees [ nom_sem ] ] . nom
if donnees [ nom_sem ] in self . semTagDict
else " manquant "
)
descr + = [
donnees [ nom_sem ] if donnees [ nom_sem ] != None else " manquant " ,
table ,
]
chaine + = delim . join ( descr ) + " \n "
return chaine
#
def export_juryPEDict ( self ) :
""" Export csv de self.PARCOURSINFO_DICT """
fichier = " juryParcoursDict_ " + str ( self . diplome )
pe_tools . pe_print ( " -> Export de " + fichier )
filename = self . NOM_EXPORT_ZIP + fichier + " .csv "
self . zipfile . writestr ( filename , self . str_etudiants_in_jury ( ) )
def get_allTagForAggregat ( self , nom_aggregat ) :
""" Extrait du dictionnaire syntheseJury la liste des tags d ' un semestre ou
2021-01-01 18:40:47 +01:00
d ' un aggrégat donné par son nom (S1, S2, S3 ou S4, 1A, ...). Renvoie [] si aucun tag. " " "
2020-09-26 16:19:37 +02:00
taglist = set ( )
for etudid in self . get_etudids_du_jury ( ) :
taglist = taglist . union (
set ( self . syntheseJury [ etudid ] [ nom_aggregat ] [ " groupe " ] . keys ( ) )
)
taglist = taglist . union (
set ( self . syntheseJury [ etudid ] [ nom_aggregat ] [ " promo " ] . keys ( ) )
)
return list ( taglist )
def get_allTagInSyntheseJury ( self ) :
2021-09-27 07:46:35 +02:00
""" Extrait tous les tags du dictionnaire syntheseJury trié par ordre alphabétique. [] si aucun tag """
2020-09-26 16:19:37 +02:00
allTags = set ( )
for nom in JuryPE . PARCOURS . keys ( ) :
allTags = allTags . union ( set ( self . get_allTagForAggregat ( nom ) ) )
return sorted ( list ( allTags ) ) if len ( allTags ) > 0 else [ ]
2021-07-09 17:47:06 +02:00
def table_syntheseJury ( self , mode = " singlesheet " ) : # was str_syntheseJury
2020-09-26 16:19:37 +02:00
""" Table(s) du jury
mode : singlesheet ou multiplesheet pour export excel
"""
sT = SeqGenTable ( ) # le fichier excel à générer
# Les etudids des étudiants à afficher, triés par ordre alphabétiques de nom+prénom
donnees_tries = sorted (
[
(
etudid ,
self . syntheseJury [ etudid ] [ " nom " ]
+ " "
+ self . syntheseJury [ etudid ] [ " prenom " ] ,
)
for etudid in self . syntheseJury . keys ( )
] ,
key = lambda c : c [ 1 ] ,
)
etudids = [ e [ 0 ] for e in donnees_tries ]
if not etudids : # Si pas d'étudiants
T = GenTable (
columns_ids = [ " pas d ' étudiants " ] ,
rows = [ ] ,
titles = { " pas d ' étudiants " : " pas d ' étudiants " } ,
html_sortable = True ,
xls_sheet_name = " dut " ,
)
sT . add_genTable ( " dut " , T )
return sT
# Si des étudiants
maxParcours = max (
[ self . syntheseJury [ etudid ] [ " nbSemestres " ] for etudid in etudids ]
)
2021-02-13 17:28:55 +01:00
infos = [ " civilite " , " nom " , " prenom " , " age " , " nbSemestres " ]
2020-09-26 16:19:37 +02:00
entete = [ " etudid " ]
entete . extend ( infos )
entete . extend ( [ " P %d " % i for i in range ( 1 , maxParcours + 1 ) ] )
champs = [
" note " ,
" class groupe " ,
" class promo " ,
" min/moy/max groupe " ,
" min/moy/max promo " ,
]
# Les aggrégats à afficher par ordre tel que indiqué dans le dictionnaire parcours
2021-07-09 17:47:06 +02:00
aggregats = list ( JuryPE . PARCOURS . keys ( ) ) # ['S1', 'S2', ..., '1A', '4S']
2020-09-26 16:19:37 +02:00
aggregats = sorted (
aggregats , key = lambda t : JuryPE . PARCOURS [ t ] [ " ordre " ]
) # Tri des aggrégats
if mode == " multiplesheet " :
allSheets = (
self . get_allTagInSyntheseJury ( )
) # tous les tags de syntheseJuryDict
allSheets = sorted ( allSheets ) # Tri des tags par ordre alphabétique
for (
sem
) in aggregats : # JuryPE.PARCOURS.keys() -> ['S1', 'S2', ..., '1A', '4S']
entete . extend ( [ " %s %s " % ( sem , champ ) for champ in champs ] )
else : # "singlesheet"
allSheets = [ " singlesheet " ]
for (
sem
) in aggregats : # JuryPE.PARCOURS.keys() -> ['S1', 'S2', ..., '1A', '4S']
tags = self . get_allTagForAggregat ( sem )
entete . extend (
[ " %s %s %s " % ( sem , tag , champ ) for tag in tags for champ in champs ]
)
columns_ids = entete # les id et les titres de colonnes sont ici identiques
titles = { i : i for i in columns_ids }
for (
sheet
) in (
allSheets
) : # Pour tous les sheets à générer (1 si singlesheet, autant que de tags si multiplesheet)
rows = [ ]
for etudid in etudids :
e = self . syntheseJury [ etudid ]
# Les info générales:
row = {
" etudid " : etudid ,
2021-02-13 17:28:55 +01:00
" civilite " : e [ " civilite " ] ,
2020-09-26 16:19:37 +02:00
" nom " : e [ " nom " ] ,
" prenom " : e [ " prenom " ] ,
" age " : e [ " age " ] ,
" nbSemestres " : e [ " nbSemestres " ] ,
}
# Les parcours: P1, P2, ...
n = 1
for p in e [ " parcours " ] :
row [ " P %d " % n ] = p [ " titreannee " ]
n + = 1
# if self.syntheseJury[etudid]['nbSemestres'] < maxParcours:
# descr += delim.join( ['']*( maxParcours -self.syntheseJury[etudid]['nbSemestres']) ) + delim
for sem in aggregats : # JuryPE.PARCOURS.keys():
listeTags = (
self . get_allTagForAggregat ( sem )
if mode == " singlesheet "
else [ sheet ]
)
for tag in listeTags :
if tag in self . syntheseJury [ etudid ] [ sem ] [ " groupe " ] :
resgroupe = self . syntheseJury [ etudid ] [ sem ] [ " groupe " ] [
tag
] # tuple
else :
resgroupe = ( None , None , None , None , None , None , None )
if tag in self . syntheseJury [ etudid ] [ sem ] [ " promo " ] :
respromo = self . syntheseJury [ etudid ] [ sem ] [ " promo " ] [ tag ]
else :
respromo = ( None , None , None , None , None , None , None )
# note = "%2.2f" % resgroupe[0] if isinstance(resgroupe[0], float) else str(resgroupe[0])
champ = (
" %s %s " % ( sem , tag )
if mode == " singlesheet "
else " %s " % ( sem )
)
2021-02-01 23:54:46 +01:00
row [ champ + " note " ] = scu . fmt_note ( resgroupe [ 0 ] )
2020-09-26 16:19:37 +02:00
row [ champ + " class groupe " ] = " %s / %s " % (
resgroupe [ 2 ] ,
resgroupe [ 3 ] ,
)
row [ champ + " class promo " ] = " %s / %s " % (
respromo [ 2 ] ,
respromo [ 3 ] ,
)
row [ champ + " min/moy/max groupe " ] = " %s / %s / %s " % tuple (
2021-02-01 23:54:46 +01:00
scu . fmt_note ( x )
2020-09-26 16:19:37 +02:00
for x in ( resgroupe [ 6 ] , resgroupe [ 4 ] , resgroupe [ 5 ] )
)
row [ champ + " min/moy/max promo " ] = " %s / %s / %s " % tuple (
2021-02-01 23:54:46 +01:00
scu . fmt_note ( x )
for x in ( respromo [ 6 ] , respromo [ 4 ] , respromo [ 5 ] )
2020-09-26 16:19:37 +02:00
)
rows . append ( row )
T = GenTable (
columns_ids = columns_ids ,
rows = rows ,
titles = titles ,
html_sortable = True ,
xls_sheet_name = sheet ,
)
sT . add_genTable ( sheet , T )
if mode == " singlesheet " :
return sT . get_genTable ( " singlesheet " )
else :
return sT
# **************************************************************************************************************** #
# Méthodes de classe pour gestion d'un cache de données accélérant les calculs / intérêt à débattre
# **************************************************************************************************************** #
# ------------------------------------------------------------------------------------------------------------------
2021-08-21 00:24:51 +02:00
def get_cache_etudInfo_d_un_etudiant ( self , etudid ) :
2020-09-26 16:19:37 +02:00
""" Renvoie les informations sur le parcours d ' un étudiant soit en les relisant depuis
ETUDINFO_DICT si mémorisée soit en les chargeant et en les mémorisant
"""
if etudid not in self . ETUDINFO_DICT :
2021-06-19 23:21:37 +02:00
self . ETUDINFO_DICT [ etudid ] = sco_etud . get_etud_info (
2020-09-26 16:19:37 +02:00
etudid = etudid , filled = True
) [ 0 ]
return self . ETUDINFO_DICT [ etudid ]
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
2022-02-11 09:11:07 +01:00
def get_cache_notes_d_un_semestre ( self , formsemestre_id : int ) - > NotesTableCompat :
2021-01-01 18:40:47 +01:00
""" Charge la table des notes d ' un formsemestre """
2023-03-20 11:17:38 +01:00
formsemestre = FormSemestre . get_formsemestre ( formsemestre_id )
2022-02-11 23:12:40 +01:00
return res_sem . load_formsemestre_results ( formsemestre )
2020-09-26 16:19:37 +02:00
# ------------------------------------------------------------------------------------------------------------------
# ------------------------------------------------------------------------------------------------------------------
def get_semestresDUT_d_un_etudiant ( self , etudid , semestre_id = None ) :
""" Renvoie la liste des semestres DUT d ' un étudiant
pour un semestre_id ( parmi 1 , 2 , 3 , 4 ) donné
2021-06-19 23:21:37 +02:00
en fonction de ses infos d ' etud (cf. sco_etud.get_etud_info( etudid=etudid, filled=True)[0]),
2020-09-26 16:19:37 +02:00
les semestres étant triés par ordre décroissant .
Si semestre_id == None renvoie tous les semestres """
2021-08-21 00:24:51 +02:00
etud = self . get_cache_etudInfo_d_un_etudiant ( etudid )
2020-09-26 16:19:37 +02:00
if semestre_id == None :
sesSems = [ sem for sem in etud [ " sems " ] if 1 < = sem [ " semestre_id " ] < = 4 ]
else :
sesSems = [ sem for sem in etud [ " sems " ] if sem [ " semestre_id " ] == semestre_id ]
return sesSems
# **********************************************
2022-04-20 12:23:43 +02:00
def calcul_anneePromoDUT_d_un_etudiant ( self , etudid ) - > int :
2021-09-26 09:52:55 +02:00
""" Calcule et renvoie la date de diplome prévue pour un étudiant fourni avec son etudid
2022-04-20 12:23:43 +02:00
en fonction de ses semestres de scolarisation """
semestres = self . get_semestresDUT_d_un_etudiant ( etudid )
if semestres :
return max ( [ get_annee_diplome_semestre ( sem ) for sem in semestres ] )
else :
return None
2020-09-26 16:19:37 +02:00
# *********************************************
# Fonctions d'affichage pour debug
def get_resultat_d_un_etudiant ( self , etudid ) :
chaine = " "
for nom_sem in [ " S1 " , " S2 " , " S3 " , " S4 " ] :
semtagid = self . PARCOURSINFO_DICT [ etudid ] [
nom_sem
] # le formsemestre_id du semestre taggué de l'étudiant
semtag = self . semTagDict [ semtagid ]
2021-08-10 17:12:10 +02:00
chaine + = " Semestre " + nom_sem + str ( semtagid ) + " \n "
2020-09-26 16:19:37 +02:00
# le détail du calcul tag par tag
# chaine += "Détail du calcul du tag\n"
# chaine += "-----------------------\n"
# for tag in semtag.taglist:
# chaine += "Tag=" + tag + "\n"
# chaine += semtag.str_detail_resultat_d_un_tag(tag, etudid=etudid) + "\n"
# le bilan des tags
chaine + = " Bilan des tags \n "
chaine + = " -------------- \n "
for tag in semtag . taglist :
chaine + = (
tag + " ; " + semtag . str_resTag_d_un_etudiant ( tag , etudid ) + " \n "
)
chaine + = " \n "
return chaine
2022-04-20 12:23:43 +02:00
def get_date_entree_etudiant ( self , etudid ) - > str :
""" Renvoie la date d ' entree d ' un étudiant: " 1996 " """
annees_debut = [
int ( sem [ " annee_debut " ] ) for sem in self . ETUDINFO_DICT [ etudid ] [ " sems " ]
]
if annees_debut :
return str ( min ( annees_debut ) )
return " "
2020-09-26 16:19:37 +02:00
# ----------------------------------------------------------------------------------------
# Fonctions
# ----------------------------------------------------------------------------------------
2022-04-20 12:23:43 +02:00
def get_annee_diplome_semestre ( sem ) - > int :
2021-01-01 18:40:47 +01:00
""" Pour un semestre donne, décrit par le biais du dictionnaire sem usuel :
2020-09-26 16:19:37 +02:00
sem = { ' formestre_id ' : . . . , ' semestre_id ' : . . . , ' annee_debut ' : . . . } ,
à condition qu ' il soit un semestre de formation DUT,
predit l ' annee à laquelle sera remis le diplome DUT des etudiants scolarisés dans le semestre
( en supposant qu ' il n ' y ait plus de redoublement ) et la renvoie sous la forme d ' un int.
Hypothese : les semestres de 1 ere partie d ' annee universitaire (comme des S1 ou des S3) s ' etalent
sur deux annees civiles - contrairement au semestre de seconde partie d ' annee universitaire (comme
des S2 ou des S4 ) .
Par exemple :
> S4 debutant en 2016 finissant en 2016 = > diplome en 2016
> S3 debutant en 2015 et finissant en 2016 = > diplome en 2016
> S3 ( decale ) debutant en 2015 et finissant en 2015 = > diplome en 2016
La regle de calcul utilise l ' annee_fin du semestre sur le principe suivant :
nbreSemRestant = nombre de semestres restant avant diplome
nbreAnneeRestant = nombre d ' annees restant avant diplome
1 - delta = 0 si semestre de 1 ere partie d ' annee / 1 sinon
decalage = active ou desactive un increment a prendre en compte en cas de semestre decale
"""
if (
1 < = sem [ " semestre_id " ] < = 4
) : # Si le semestre est un semestre DUT => problème si formation DUT en 1 an ??
nbreSemRestant = 4 - sem [ " semestre_id " ]
nbreAnRestant = nbreSemRestant / / 2
delta = int ( sem [ " annee_fin " ] ) - int ( sem [ " annee_debut " ] )
decalage = nbreSemRestant % 2 # 0 si S4, 1 si S3, 0 si S2, 1 si S1
increment = decalage * ( 1 - delta )
return int ( sem [ " annee_fin " ] ) + nbreAnRestant + increment
# ----------------------------------------------------------------------------------------
# ----------------------------------------------------------------------------------
2021-08-21 00:24:51 +02:00
def get_cosemestres_diplomants ( semBase , avec_meme_formation = False ) :
2021-01-01 18:40:47 +01:00
""" Partant d ' un semestre de Base = { ' formsemestre_id ' : ..., ' semestre_id ' : ..., ' annee_debut ' : ...},
renvoie la liste de tous ses co - semestres ( lui - meme inclus )
Par co - semestre , s ' entend les semestres :
> dont l ' annee predite pour la remise du diplome DUT est la meme
> dont la formation est la même ( optionnel )
> ne prenant en compte que les etudiants sans redoublement
2020-09-26 16:19:37 +02:00
"""
2021-08-19 10:28:35 +02:00
tousLesSems = (
sco_formsemestre . do_formsemestre_list ( )
2021-07-29 10:19:00 +02:00
) # tous les semestres memorisés dans scodoc
2020-09-26 16:19:37 +02:00
diplome = get_annee_diplome_semestre ( semBase )
if avec_meme_formation : # si une formation est imposee
2021-08-09 10:09:04 +02:00
nom_formation = str ( semBase [ " formation_id " ] )
2020-09-26 16:19:37 +02:00
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " - avec formation imposée : " , nom_formation )
coSems = [
sem
for sem in tousLesSems
if get_annee_diplome_semestre ( sem ) == diplome
and sem [ " formation_id " ] == semBase [ " formation_id " ]
]
else :
if pe_tools . PE_DEBUG :
pe_tools . pe_print ( " - toutes formations confondues " )
coSems = [
sem for sem in tousLesSems if get_annee_diplome_semestre ( sem ) == diplome
]
return coSems