2023-01-09 22:46:27 +01:00
|
|
|
##############################################################################
|
|
|
|
# ScoDoc
|
2023-12-31 23:04:06 +01:00
|
|
|
# Copyright (c) 1999 - 2024 Emmanuel Viennet. All rights reserved.
|
2023-01-09 22:46:27 +01:00
|
|
|
# See LICENSE
|
|
|
|
##############################################################################
|
|
|
|
|
2022-12-09 04:24:32 +01:00
|
|
|
"""
|
2023-01-11 19:09:03 +01:00
|
|
|
Met en place une base pour les tests unitaires, à partir d'une description
|
2023-01-09 22:46:27 +01:00
|
|
|
YAML qui peut donner la formation, son ref. compétences, les formsemestres,
|
|
|
|
les étudiants et leurs notes et décisions de jury.
|
|
|
|
|
2023-01-11 19:09:03 +01:00
|
|
|
Le traitement est effectué dans l'ordre suivant:
|
2023-01-09 22:46:27 +01:00
|
|
|
|
|
|
|
setup_from_yaml()
|
|
|
|
|
2023-02-11 05:04:10 +01:00
|
|
|
- setup_formation():
|
2023-01-09 22:46:27 +01:00
|
|
|
- import de la formation (le test utilise une seule formation)
|
|
|
|
- associe_ues_et_parcours():
|
|
|
|
- crée les associations formation <-> référentiel de compétence
|
2023-06-29 23:24:36 +02:00
|
|
|
- setup_formsemestre()
|
2023-01-09 22:46:27 +01:00
|
|
|
- crée les formsemestres décrits dans le YAML
|
|
|
|
avec tous les modules du semestre ou des parcours si indiqués
|
|
|
|
et une évaluation dans chaque moduleimpl.
|
|
|
|
- inscrit_les_etudiants()
|
|
|
|
- inscrit et place dans les groupes de parcours
|
|
|
|
- note_les_modules()
|
|
|
|
- saisie de toutes les notes indiquées dans le YAML
|
|
|
|
|
|
|
|
check_deca_fields()
|
|
|
|
- vérifie les champs du deca (nb UE, compétences, ...) mais pas les décisions de jury.
|
|
|
|
|
|
|
|
formsemestre_validation_auto_but(only_adm=False)
|
|
|
|
- enregistre toutes les décisions "par défaut" proposées (pas seulement les ADM)
|
|
|
|
|
|
|
|
test_but_jury()
|
|
|
|
- compare décisions attendues indiquées dans le YAML avec celles de ScoDoc
|
2023-01-11 13:37:02 +01:00
|
|
|
et enregistre immédiatement APRES la décision manuelle indiquée par `decision_jury`
|
2023-01-09 22:46:27 +01:00
|
|
|
dans le YAML.
|
|
|
|
|
|
|
|
|
|
|
|
Les tests unitaires associés sont généralement lents (construction de la base),
|
|
|
|
et donc marqués par `@pytest.mark.slow`.
|
2022-12-09 04:24:32 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
|
|
|
import yaml
|
2023-02-11 05:04:10 +01:00
|
|
|
from flask import g
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
from app import db
|
|
|
|
|
|
|
|
from app.auth.models import User
|
2023-02-11 05:04:10 +01:00
|
|
|
|
2022-12-09 04:24:32 +01:00
|
|
|
from app.models import (
|
|
|
|
ApcParcours,
|
2023-03-27 23:57:10 +02:00
|
|
|
DispenseUE,
|
2022-12-09 04:24:32 +01:00
|
|
|
Evaluation,
|
|
|
|
Formation,
|
|
|
|
FormSemestre,
|
|
|
|
Identite,
|
|
|
|
Module,
|
|
|
|
ModuleImpl,
|
2023-03-27 23:57:10 +02:00
|
|
|
UniteEns,
|
2022-12-09 04:24:32 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
from app.scodoc import sco_formations
|
|
|
|
from app.scodoc import sco_formsemestre_inscriptions
|
|
|
|
from app.scodoc import sco_groups
|
|
|
|
from app.scodoc import sco_saisie_notes
|
|
|
|
from app.scodoc import sco_utils as scu
|
|
|
|
|
|
|
|
from tests.conftest import RESOURCES_DIR
|
2023-02-11 05:04:10 +01:00
|
|
|
from tests.unit import yaml_setup_but
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
|
2023-02-11 05:04:10 +01:00
|
|
|
def setup_formation(formation_infos: dict) -> Formation:
|
2022-12-09 04:24:32 +01:00
|
|
|
"""Importe la formation, qui est lue à partir du fichier XML
|
|
|
|
formation_infos["filename"].
|
|
|
|
La formation peut être quelconque, on vérifie juste qu'elle est bien créée.
|
|
|
|
"""
|
|
|
|
# Lecture fichier XML local:
|
|
|
|
with open(
|
|
|
|
os.path.join(RESOURCES_DIR, "formations", formation_infos["filename"]),
|
|
|
|
encoding="utf-8",
|
|
|
|
) as f:
|
|
|
|
doc = f.read()
|
|
|
|
|
|
|
|
# --- Création de la formation
|
|
|
|
formation_id, _, _ = sco_formations.formation_import_xml(doc)
|
2023-07-11 06:57:38 +02:00
|
|
|
formation: Formation = db.session.get(Formation, formation_id)
|
2022-12-09 04:24:32 +01:00
|
|
|
assert formation
|
|
|
|
return formation
|
|
|
|
|
|
|
|
|
2023-02-11 05:04:10 +01:00
|
|
|
def create_formsemestre(
|
2022-12-09 04:24:32 +01:00
|
|
|
formation: Formation,
|
|
|
|
parcours: list[ApcParcours],
|
|
|
|
semestre_id: int,
|
|
|
|
titre: str,
|
|
|
|
date_debut: str,
|
|
|
|
date_fin: str,
|
2023-07-01 17:42:04 +02:00
|
|
|
elt_sem_apo: str = None,
|
|
|
|
elt_annee_apo: str = None,
|
|
|
|
etape_apo: str = None,
|
2022-12-09 04:24:32 +01:00
|
|
|
) -> FormSemestre:
|
2023-02-11 05:04:10 +01:00
|
|
|
"Création d'un formsemestre, avec ses modimpls et évaluations"
|
|
|
|
assert formation.is_apc() or not parcours # parcours seulement si APC
|
2022-12-09 04:24:32 +01:00
|
|
|
formsemestre = FormSemestre(
|
|
|
|
formation=formation,
|
|
|
|
parcours=parcours,
|
|
|
|
dept_id=g.scodoc_dept_id,
|
|
|
|
titre=titre,
|
|
|
|
semestre_id=semestre_id,
|
|
|
|
date_debut=date_debut,
|
|
|
|
date_fin=date_fin,
|
2023-07-01 17:42:04 +02:00
|
|
|
elt_sem_apo=elt_sem_apo,
|
|
|
|
elt_annee_apo=elt_annee_apo,
|
2022-12-09 04:24:32 +01:00
|
|
|
)
|
2023-07-11 06:57:38 +02:00
|
|
|
db.session.add(formsemestre)
|
|
|
|
db.session.flush()
|
2022-12-09 04:24:32 +01:00
|
|
|
# set responsable (list)
|
|
|
|
a_user = User.query.first()
|
|
|
|
formsemestre.responsables = [a_user]
|
2023-07-01 17:42:04 +02:00
|
|
|
formsemestre.add_etape(etape_apo)
|
2022-12-17 03:26:22 +01:00
|
|
|
# Ajoute tous les modules du semestre sans parcours OU avec l'un des parcours indiqués
|
|
|
|
sem_parcours_ids = {p.id for p in parcours}
|
|
|
|
modules = [
|
|
|
|
m
|
|
|
|
for m in formsemestre.formation.modules.filter_by(semestre_id=semestre_id)
|
2023-10-19 23:47:04 +02:00
|
|
|
if not m.parcours # module de tronc commun
|
2023-05-14 19:29:25 +02:00
|
|
|
or (not sem_parcours_ids) # semestre sans parcours => tous
|
|
|
|
or ({p.id for p in m.parcours} & sem_parcours_ids)
|
2022-12-17 03:26:22 +01:00
|
|
|
]
|
|
|
|
for module in modules:
|
|
|
|
modimpl = ModuleImpl(module=module, responsable_id=a_user.id)
|
|
|
|
db.session.add(modimpl)
|
|
|
|
formsemestre.modimpls.append(modimpl)
|
|
|
|
|
2022-12-09 04:24:32 +01:00
|
|
|
# Crée une évaluation dans chaque module
|
|
|
|
create_evaluations(formsemestre)
|
|
|
|
|
|
|
|
# Partition par défaut:
|
|
|
|
db.session.commit()
|
|
|
|
partition_id = sco_groups.partition_create(
|
|
|
|
formsemestre.id, default=True, redirect=False
|
|
|
|
)
|
|
|
|
_ = sco_groups.create_group(partition_id, default=True)
|
2023-02-11 05:04:10 +01:00
|
|
|
|
2022-12-09 04:24:32 +01:00
|
|
|
# Partition de parcours:
|
2023-02-11 05:04:10 +01:00
|
|
|
if formsemestre.formation.is_apc():
|
|
|
|
formsemestre.setup_parcours_groups()
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
return formsemestre
|
|
|
|
|
|
|
|
|
2023-06-15 21:53:05 +02:00
|
|
|
def create_evaluations(formsemestre: FormSemestre, publish_incomplete=True):
|
2022-12-09 04:24:32 +01:00
|
|
|
"""Crée une évaluation dans chaque module du semestre"""
|
|
|
|
for modimpl in formsemestre.modimpls:
|
|
|
|
evaluation = Evaluation(
|
|
|
|
moduleimpl=modimpl,
|
2023-08-25 17:58:57 +02:00
|
|
|
date_debut=formsemestre.date_debut,
|
2022-12-09 04:24:32 +01:00
|
|
|
description=f"Exam {modimpl.module.titre}",
|
|
|
|
coefficient=1.0,
|
|
|
|
note_max=20.0,
|
|
|
|
numero=1,
|
2023-06-15 21:53:05 +02:00
|
|
|
publish_incomplete=publish_incomplete,
|
2022-12-09 04:24:32 +01:00
|
|
|
)
|
|
|
|
db.session.add(evaluation)
|
2023-02-11 05:04:10 +01:00
|
|
|
|
|
|
|
if formsemestre.formation.is_apc():
|
|
|
|
# Affecte les mêmes poids que les coefs APC:
|
|
|
|
ue_coef_dict = modimpl.module.get_ue_coef_dict() # { ue_id : coef }
|
|
|
|
evaluation.set_ue_poids_dict(ue_coef_dict)
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
|
2023-06-29 23:24:36 +02:00
|
|
|
def note_les_modules(doc: dict, formsemestre_titre: str = ""):
|
|
|
|
"""Saisie les notes des étudiants pour ce formsemestre
|
2023-02-11 05:04:10 +01:00
|
|
|
doc : données YAML
|
|
|
|
"""
|
2022-12-09 04:24:32 +01:00
|
|
|
a_user = User.query.first()
|
2023-06-29 23:24:36 +02:00
|
|
|
formsemestre: FormSemestre = FormSemestre.query.filter_by(
|
|
|
|
titre=formsemestre_titre
|
|
|
|
).first()
|
|
|
|
assert formsemestre is not None
|
2022-12-09 04:24:32 +01:00
|
|
|
for nom, infos in doc["Etudiants"].items():
|
|
|
|
etud: Identite = Identite.query.filter_by(nom=nom).first()
|
|
|
|
assert etud is not None
|
2023-06-29 23:24:36 +02:00
|
|
|
sem_infos = infos["formsemestres"].get(formsemestre_titre)
|
|
|
|
if sem_infos:
|
2022-12-09 04:24:32 +01:00
|
|
|
for code_module, note in sem_infos.get("notes_modules", {}).items():
|
|
|
|
modimpl = (
|
|
|
|
formsemestre.modimpls.join(Module)
|
|
|
|
.filter_by(code=code_module)
|
|
|
|
.first()
|
|
|
|
)
|
|
|
|
# le sem. doit avoir un module du code indiqué:
|
2022-12-18 03:58:25 +01:00
|
|
|
if modimpl is None:
|
|
|
|
raise ValueError(f"module de code '{code_module}' introuvable")
|
2022-12-09 04:24:32 +01:00
|
|
|
for evaluation in modimpl.evaluations:
|
|
|
|
# s'il y a plusieurs evals, affecte la même note à chacune
|
2023-01-04 15:32:28 +01:00
|
|
|
note_value, invalid = sco_saisie_notes.convert_note_from_string(
|
|
|
|
str(note),
|
|
|
|
20.0,
|
|
|
|
note_min=scu.NOTES_MIN,
|
|
|
|
etudid=etud.id,
|
|
|
|
absents=[],
|
|
|
|
invalids=[],
|
|
|
|
)
|
|
|
|
assert not invalid # valeur note invalide
|
|
|
|
assert isinstance(note_value, float)
|
2022-12-09 04:24:32 +01:00
|
|
|
sco_saisie_notes.notes_add(
|
|
|
|
a_user,
|
|
|
|
evaluation.id,
|
2023-01-04 15:32:28 +01:00
|
|
|
[(etud.id, note_value)],
|
2022-12-09 04:24:32 +01:00
|
|
|
comment="note_les_modules",
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-06-29 23:24:36 +02:00
|
|
|
def setup_formsemestre(
|
|
|
|
formation: Formation, doc: str, formsemestre_titre: str = ""
|
|
|
|
) -> FormSemestre:
|
|
|
|
"""Création du formsemestre de titre indiqué.
|
|
|
|
Le cas échéant associé à son parcours.
|
2023-02-11 05:04:10 +01:00
|
|
|
"""
|
2023-06-29 23:24:36 +02:00
|
|
|
infos = doc["FormSemestres"][formsemestre_titre]
|
|
|
|
|
|
|
|
codes_parcours = infos.get("codes_parcours", [])
|
|
|
|
assert formation.is_apc() or not codes_parcours # parcours seulement en APC
|
|
|
|
parcours = []
|
|
|
|
for code_parcour in codes_parcours:
|
|
|
|
parcour = formation.referentiel_competence.parcours.filter_by(
|
|
|
|
code=code_parcour
|
|
|
|
).first()
|
|
|
|
assert parcour is not None
|
|
|
|
parcours.append(parcour)
|
|
|
|
|
2023-07-01 17:42:04 +02:00
|
|
|
elt_sem_apo = infos.get("elt_sem_apo")
|
|
|
|
elt_annee_apo = infos.get("elt_annee_apo")
|
|
|
|
etape_apo = infos.get("etape_apo")
|
|
|
|
|
2023-06-29 23:24:36 +02:00
|
|
|
formsemestre = create_formsemestre(
|
|
|
|
formation,
|
|
|
|
parcours,
|
|
|
|
infos["idx"],
|
|
|
|
formsemestre_titre,
|
|
|
|
infos["date_debut"],
|
|
|
|
infos["date_fin"],
|
2023-07-01 17:42:04 +02:00
|
|
|
elt_sem_apo=elt_sem_apo,
|
|
|
|
elt_annee_apo=elt_annee_apo,
|
|
|
|
etape_apo=etape_apo,
|
2023-06-29 23:24:36 +02:00
|
|
|
)
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
db.session.flush()
|
2023-06-29 23:24:36 +02:00
|
|
|
return formsemestre
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
|
2023-06-29 23:24:36 +02:00
|
|
|
def inscrit_les_etudiants(doc: dict, formsemestre_titre: str = ""):
|
2022-12-09 04:24:32 +01:00
|
|
|
"""Inscrit les étudiants dans chacun de leurs formsemestres"""
|
2023-02-21 15:28:42 +01:00
|
|
|
etudiants = doc.get("Etudiants")
|
|
|
|
if not etudiants:
|
|
|
|
return
|
2023-06-29 23:24:36 +02:00
|
|
|
formsemestre: FormSemestre = FormSemestre.query.filter_by(
|
|
|
|
titre=formsemestre_titre
|
|
|
|
).first()
|
|
|
|
assert formsemestre is not None
|
|
|
|
partition_parcours = formsemestre.partitions.filter_by(
|
|
|
|
partition_name=scu.PARTITION_PARCOURS
|
|
|
|
).first()
|
2023-02-21 15:28:42 +01:00
|
|
|
for nom, infos in etudiants.items():
|
2023-06-29 23:24:36 +02:00
|
|
|
# Création des étudiants (sauf si déjà existants)
|
|
|
|
prenom = infos.get("prenom", "prénom")
|
|
|
|
civilite = infos.get("civilite", "X")
|
2023-07-01 17:42:04 +02:00
|
|
|
code_nip = infos.get("code_nip", None)
|
2023-06-29 23:24:36 +02:00
|
|
|
etud = Identite.query.filter_by(
|
|
|
|
nom=nom, prenom=prenom, civilite=civilite
|
|
|
|
).first()
|
|
|
|
if etud is None:
|
|
|
|
etud = Identite.create_etud(
|
|
|
|
dept_id=g.scodoc_dept_id,
|
|
|
|
nom=nom,
|
|
|
|
prenom=prenom,
|
|
|
|
civilite=civilite,
|
2023-07-01 17:42:04 +02:00
|
|
|
code_nip=code_nip,
|
2023-06-29 23:24:36 +02:00
|
|
|
)
|
|
|
|
db.session.add(etud)
|
|
|
|
db.session.commit()
|
|
|
|
|
|
|
|
# L'inscrire au formsemestre
|
|
|
|
sem_infos = infos["formsemestres"].get(formsemestre_titre)
|
|
|
|
if sem_infos:
|
2022-12-20 04:16:38 +01:00
|
|
|
if partition_parcours is not None and "parcours" in sem_infos:
|
2022-12-09 04:24:32 +01:00
|
|
|
group = partition_parcours.groups.filter_by(
|
|
|
|
group_name=sem_infos["parcours"]
|
|
|
|
).first()
|
|
|
|
assert group is not None # le groupe de parcours doit exister
|
|
|
|
group_ids = [group.id]
|
2022-12-20 04:16:38 +01:00
|
|
|
else:
|
|
|
|
group_ids = []
|
2023-03-27 23:57:10 +02:00
|
|
|
# Génère des dispenses d'UEs
|
|
|
|
if "dispense_ues" in sem_infos:
|
|
|
|
etud_dispense_ues(formsemestre, etud, sem_infos["dispense_ues"])
|
2022-12-09 04:24:32 +01:00
|
|
|
sco_formsemestre_inscriptions.do_formsemestre_inscription_with_modules(
|
|
|
|
formsemestre.id,
|
|
|
|
etud.id,
|
|
|
|
group_ids=group_ids,
|
|
|
|
etat=scu.INSCRIT,
|
|
|
|
method="tests/unit/inscrit_les_etudiants",
|
|
|
|
)
|
2023-06-29 23:24:36 +02:00
|
|
|
# Met à jour les inscriptions:
|
|
|
|
formsemestre.update_inscriptions_parcours_from_groups()
|
2022-12-09 04:24:32 +01:00
|
|
|
|
|
|
|
|
2023-03-27 23:57:10 +02:00
|
|
|
def etud_dispense_ues(
|
|
|
|
formsemestre: FormSemestre, etud: Identite, ue_acronymes: list[str]
|
|
|
|
):
|
|
|
|
"""Génère des dispenses d'UE"""
|
|
|
|
for ue_acronyme in set(ue_acronymes):
|
|
|
|
ue: UniteEns = formsemestre.formation.ues.filter_by(
|
|
|
|
acronyme=ue_acronyme
|
|
|
|
).first()
|
|
|
|
assert ue
|
|
|
|
disp = DispenseUE(formsemestre_id=formsemestre.id, ue_id=ue.id, etudid=etud.id)
|
|
|
|
db.session.add(disp)
|
|
|
|
|
|
|
|
|
2023-06-29 23:24:36 +02:00
|
|
|
def setup_from_yaml(filename: str) -> tuple[dict, Formation, list[str]]:
|
2022-12-09 04:24:32 +01:00
|
|
|
"""Lit le fichier yaml et construit l'ensemble des objets"""
|
|
|
|
with open(filename, encoding="utf-8") as f:
|
|
|
|
doc = yaml.safe_load(f.read())
|
|
|
|
|
2023-10-19 23:47:04 +02:00
|
|
|
# Passe tous les noms en majuscules (car stockés ainsi en base)
|
|
|
|
if "Etudiants" in doc:
|
|
|
|
doc["Etudiants"] = {k.upper(): v for (k, v) in doc["Etudiants"].items()}
|
|
|
|
for e in doc["Etudiants"].values():
|
|
|
|
if "prenom" in e:
|
|
|
|
e["prenom"] = e["prenom"].upper()
|
|
|
|
|
2023-05-14 17:36:06 +02:00
|
|
|
# Charge de ref. comp. avant la formation, de façon à pouvoir
|
|
|
|
# re-créer les associations UE/Niveaux
|
|
|
|
yaml_setup_but.setup_formation_referentiel(doc.get("ReferentielCompetences", {}))
|
2023-02-11 05:04:10 +01:00
|
|
|
formation = setup_formation(doc["Formation"])
|
2023-05-14 17:36:06 +02:00
|
|
|
|
2023-02-11 05:04:10 +01:00
|
|
|
yaml_setup_but.associe_ues_et_parcours(formation, doc["Formation"])
|
2023-06-29 23:24:36 +02:00
|
|
|
|
|
|
|
formsemestre_titres = list(doc["FormSemestres"].keys())
|
|
|
|
return doc, formation, formsemestre_titres
|
|
|
|
|
|
|
|
|
|
|
|
def create_formsemestre_with_etuds(
|
|
|
|
doc: dict, formation: Formation, formsemestre_titre: str
|
|
|
|
) -> FormSemestre:
|
|
|
|
"""Création formsemestre de titre indiqué, puis inscrit ses étudianst et les note"""
|
|
|
|
formsemestre = setup_formsemestre(
|
|
|
|
formation, doc, formsemestre_titre=formsemestre_titre
|
|
|
|
)
|
2023-02-21 15:28:42 +01:00
|
|
|
etudiants = doc.get("Etudiants")
|
|
|
|
if etudiants:
|
2023-06-29 23:24:36 +02:00
|
|
|
inscrit_les_etudiants(doc, formsemestre_titre=formsemestre_titre)
|
|
|
|
note_les_modules(doc, formsemestre_titre=formsemestre_titre)
|
|
|
|
return formsemestre
|