forked from ScoDoc/DocScoDoc
248 lines
8.6 KiB
Python
248 lines
8.6 KiB
Python
# -*- mode: python -*-
|
|
# -*- coding: utf-8 -*-
|
|
|
|
"""Test de base de ScoDoc
|
|
|
|
|
|
Utiliser comme:
|
|
pytest tests/unit/test_sco_basic.py
|
|
|
|
Au besoin, créer un base de test neuve:
|
|
./tools/create_database.sh SCODOC_TEST
|
|
|
|
"""
|
|
from app.models import FormSemestreInscription, Identite
|
|
|
|
from config import TestConfig
|
|
from tests.unit import sco_fake_gen
|
|
from tests.unit.setup import NOTES_T
|
|
|
|
import app
|
|
from app import db
|
|
from app.comp import res_sem
|
|
from app.comp.res_compat import NotesTableCompat
|
|
from app.models import FormSemestre
|
|
from app.scodoc import sco_formsemestre
|
|
from app.scodoc import sco_abs
|
|
from app.scodoc import sco_abs_views
|
|
from app.scodoc import sco_bulletins
|
|
from app.scodoc import sco_codes_parcours
|
|
from app.scodoc import sco_evaluations
|
|
from app.scodoc import sco_evaluation_db
|
|
from app.scodoc import sco_formsemestre_validation
|
|
from app.scodoc import sco_cursus_dut
|
|
from app.scodoc import sco_saisie_notes
|
|
|
|
DEPT = TestConfig.DEPT_TEST
|
|
|
|
|
|
def test_sco_basic(test_client):
|
|
"""Test quelques opérations élémentaires de ScoDoc
|
|
Création 10 étudiants, formation, semestre, inscription etudiant,
|
|
creation 1 evaluation, saisie 10 notes.
|
|
"""
|
|
app.set_sco_dept(DEPT)
|
|
run_sco_basic()
|
|
|
|
|
|
def run_sco_basic(verbose=False) -> FormSemestre:
|
|
"""Scénario de base: création formation, semestre, étudiants, notes,
|
|
décisions jury
|
|
Renvoie le formsemestre créé.
|
|
"""
|
|
G = sco_fake_gen.ScoFake(verbose=verbose)
|
|
|
|
# --- Création d'étudiants
|
|
etuds = [G.create_etud(code_nip=None) for _ in range(10)]
|
|
|
|
# --- Création d'une formation
|
|
formation_id = G.create_formation(acronyme="")
|
|
ue_id = G.create_ue(formation_id=formation_id, acronyme="TST1", titre="ue test")
|
|
matiere_id = G.create_matiere(ue_id=ue_id, titre="matière test")
|
|
module_id = G.create_module(
|
|
matiere_id=matiere_id,
|
|
code="TSM1",
|
|
coefficient=1.0,
|
|
titre="module test",
|
|
ue_id=ue_id,
|
|
formation_id=formation_id,
|
|
)
|
|
|
|
# --- Mise place d'un semestre
|
|
formsemestre_id = G.create_formsemestre(
|
|
formation_id=formation_id,
|
|
semestre_id=1,
|
|
date_debut="01/01/2020",
|
|
date_fin="30/06/2020",
|
|
)
|
|
sem = sco_formsemestre.get_formsemestre(formsemestre_id)
|
|
moduleimpl_id = G.create_moduleimpl(
|
|
module_id=module_id,
|
|
formsemestre_id=formsemestre_id,
|
|
)
|
|
|
|
# --- Inscription des étudiants
|
|
for etud in etuds:
|
|
G.inscrit_etudiant(formsemestre_id, etud)
|
|
# Vérification incription semestre:
|
|
q = FormSemestreInscription.query.filter_by(
|
|
etudid=etuds[0]["id"], formsemestre_id=formsemestre_id
|
|
)
|
|
assert q.count() == 1
|
|
ins = q.first()
|
|
assert ins.etape is None
|
|
assert ins.etat == "I"
|
|
assert ins.parcour is None
|
|
|
|
# --- Création évaluation
|
|
e = G.create_evaluation(
|
|
moduleimpl_id=moduleimpl_id,
|
|
jour="01/01/2020",
|
|
description="evaluation test",
|
|
coefficient=1.0,
|
|
)
|
|
|
|
# --- Saisie toutes les notes de l'évaluation
|
|
for idx, etud in enumerate(etuds):
|
|
nb_changed, nb_suppress, existing_decisions = G.create_note(
|
|
evaluation_id=e["id"], etudid=etud["id"], note=NOTES_T[idx % len(NOTES_T)]
|
|
)
|
|
assert not existing_decisions
|
|
assert nb_suppress == 0
|
|
assert nb_changed == 1
|
|
|
|
# --- Vérifie que les notes sont prises en compte:
|
|
b = sco_bulletins.formsemestre_bulletinetud_dict(formsemestre_id, etud["etudid"])
|
|
# Toute les notes sont saisies, donc eval complète
|
|
etat = sco_evaluations.do_evaluation_etat(e["evaluation_id"])
|
|
assert etat["evalcomplete"]
|
|
assert etat["nb_inscrits"] == len(etuds)
|
|
assert etat["nb_notes"] == len(etuds)
|
|
# Un seul module, donc moy gen == note module
|
|
assert b["ues"][0]["cur_moy_ue_txt"] == b["ues"][0]["modules"][0]["mod_moy_txt"]
|
|
# Note au module égale à celle de l'éval
|
|
assert (
|
|
b["ues"][0]["modules"][0]["mod_moy_txt"]
|
|
== b["ues"][0]["modules"][0]["evaluations"][0]["note_txt"]
|
|
)
|
|
|
|
# --- Une autre évaluation
|
|
e2 = G.create_evaluation(
|
|
moduleimpl_id=moduleimpl_id,
|
|
jour="02/01/2020",
|
|
description="evaluation test 2",
|
|
coefficient=1.0,
|
|
)
|
|
# Saisie les notes des 5 premiers étudiants:
|
|
for idx, etud in enumerate(etuds[:5]):
|
|
nb_changed, nb_suppress, existing_decisions = G.create_note(
|
|
evaluation_id=e2["id"], etudid=etud["id"], note=NOTES_T[idx % len(NOTES_T)]
|
|
)
|
|
# Cette éval n'est pas complète
|
|
etat = sco_evaluations.do_evaluation_etat(e2["evaluation_id"])
|
|
assert etat["evalcomplete"] is False
|
|
# la première éval est toujours complète:
|
|
etat = sco_evaluations.do_evaluation_etat(e["evaluation_id"])
|
|
assert etat["evalcomplete"]
|
|
|
|
# Modifie l'évaluation 2 pour "prise en compte immédiate"
|
|
e2["publish_incomplete"] = True
|
|
sco_evaluation_db.do_evaluation_edit(e2)
|
|
etat = sco_evaluations.do_evaluation_etat(e2["evaluation_id"])
|
|
assert etat["evalcomplete"] is False
|
|
assert etat["nb_att"] == 0 # il n'y a pas de notes (explicitement) en attente
|
|
assert etat["evalattente"] # mais l'eval est en attente (prise en compte immédiate)
|
|
|
|
# Saisie des notes qui manquent:
|
|
for idx, etud in enumerate(etuds[5:]):
|
|
nb_changed, nb_suppress, existing_decisions = G.create_note(
|
|
evaluation_id=e2["id"], etudid=etud["id"], note=NOTES_T[idx % len(NOTES_T)]
|
|
)
|
|
etat = sco_evaluations.do_evaluation_etat(e2["evaluation_id"])
|
|
assert etat["evalcomplete"]
|
|
assert etat["nb_att"] == 0
|
|
assert not etat["evalattente"] # toutes les notes sont présentes
|
|
|
|
# --- Suppression des notes
|
|
sco_saisie_notes.evaluation_suppress_alln(e["evaluation_id"], dialog_confirmed=True)
|
|
etat = sco_evaluations.do_evaluation_etat(e["evaluation_id"])
|
|
assert etat["nb_notes"] == 0
|
|
assert not etat["evalcomplete"]
|
|
# --- Saisie des notes manquantes
|
|
ans = sco_saisie_notes.do_evaluation_set_missing(
|
|
e["evaluation_id"], 12.34, dialog_confirmed=True
|
|
)
|
|
assert f'{etat["nb_inscrits"]} notes changées' in ans
|
|
etat = sco_evaluations.do_evaluation_etat(e["evaluation_id"])
|
|
assert etat["evalcomplete"]
|
|
|
|
# -----------------------
|
|
# --- Saisie absences ---
|
|
# -----------------------
|
|
etudid = etuds[0]["etudid"]
|
|
|
|
_ = sco_abs_views.doSignaleAbsence(
|
|
"15/01/2020", "18/01/2020", demijournee=2, etudid=etudid
|
|
)
|
|
|
|
_ = sco_abs_views.doJustifAbsence(
|
|
"17/01/2020",
|
|
"18/01/2020",
|
|
demijournee=2,
|
|
etudid=etudid,
|
|
)
|
|
|
|
nbabs, nbabsjust = sco_abs.get_abs_count(etudid, sem)
|
|
assert nbabs == 6, f"incorrect nbabs ({nbabs})"
|
|
assert nbabsjust == 2, f"incorrect nbabsjust ({nbabsjust})"
|
|
|
|
# --- Permission saisie notes et décisions de jury, avec ou sans démission ou défaillance
|
|
# on n'a pas encore saisi de décisions
|
|
assert not sco_cursus_dut.formsemestre_has_decisions(formsemestre_id)
|
|
# Saisie d'un décision AJ, non assidu
|
|
etudid = etuds[-1]["etudid"]
|
|
sco_cursus_dut.formsemestre_validate_ues(
|
|
formsemestre_id, etudid, sco_codes_parcours.AJ, False
|
|
)
|
|
assert sco_cursus_dut.formsemestre_has_decisions(
|
|
formsemestre_id
|
|
), "décisions manquantes"
|
|
# Suppression de la décision
|
|
sco_formsemestre_validation.formsemestre_validation_suppress_etud(
|
|
formsemestre_id, etudid
|
|
)
|
|
assert not sco_cursus_dut.formsemestre_has_decisions(
|
|
formsemestre_id
|
|
), "décisions non effacées"
|
|
|
|
# --- Décision de jury et validations des ECTS d'UE
|
|
for etud in etuds[:5]: # les etudiants notés
|
|
sco_formsemestre_validation.formsemestre_validation_etud_manu(
|
|
formsemestre_id,
|
|
etud["etudid"],
|
|
code_etat=sco_codes_parcours.ADJ,
|
|
assidu=True,
|
|
redirect=False,
|
|
)
|
|
# Vérifie que toutes les UE des étudiants notés ont été acquises:
|
|
formsemestre: FormSemestre = FormSemestre.query.get_or_404(formsemestre_id)
|
|
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
|
|
for etud in etuds[:5]:
|
|
dec_ues = nt.get_etud_decision_ues(etud["etudid"])
|
|
for ue_id in dec_ues:
|
|
assert dec_ues[ue_id]["code"] in {"ADM", "CMP"}
|
|
|
|
# ---- Suppression d'un étudiant, vérification inscription
|
|
# (permet de tester les cascades)
|
|
etud = Identite.query.get(etuds[0]["id"])
|
|
assert etud is not None
|
|
etudid = etud.id
|
|
db.session.delete(etud)
|
|
db.session.commit()
|
|
# Vérification incription semestre:
|
|
q = FormSemestreInscription.query.filter_by(
|
|
etudid=etudid, formsemestre_id=formsemestre_id
|
|
)
|
|
assert q.count() == 0
|
|
return formsemestre
|