From 52d0499c9b73cfd4ef0eda4d6f6fe1d0d033000a Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 20:29:30 +0200
Subject: [PATCH 1/9] Fix: scodoc_table_results
---
app/scodoc/sco_export_results.py | 14 +++++++++-----
1 file changed, 9 insertions(+), 5 deletions(-)
diff --git a/app/scodoc/sco_export_results.py b/app/scodoc/sco_export_results.py
index 30750e3e..6c005306 100644
--- a/app/scodoc/sco_export_results.py
+++ b/app/scodoc/sco_export_results.py
@@ -208,25 +208,29 @@ def _build_results_list(dpv_by_sem, etuds_infos):
return rows, titles, columns_ids
-def get_set_formsemestre_id_dates(start_date, end_date):
+def get_set_formsemestre_id_dates(start_date, end_date) -> set:
"""Ensemble des formsemestre_id entre ces dates"""
s = ndb.SimpleDictFetch(
"""SELECT id
FROM notes_formsemestre
- WHERE date_debut >= %(start_date)s AND date_fin <= %(end_date)s
+ WHERE date_debut >= %(start_date)s
+ AND date_fin <= %(end_date)s
+ AND dept_id = %(dept_id)s
""",
- {"start_date": start_date, "end_date": end_date},
+ {"start_date": start_date, "end_date": end_date, "dept_id": g.scodoc_dept_id},
)
return {x["id"] for x in s}
-def scodoc_table_results(start_date="", end_date="", types_parcours=[], format="html"):
+def scodoc_table_results(
+ start_date="", end_date="", types_parcours: list = None, format="html"
+):
"""Page affichant la table des résultats
Les dates sont en dd/mm/yyyy (datepicker javascript)
types_parcours est la liste des types de parcours à afficher
(liste de chaines, eg ['100', '210'] )
"""
- log("scodoc_table_results: start_date=%s" % (start_date,)) # XXX
+ log(f"scodoc_table_results: start_date={start_date!r}")
if not types_parcours:
types_parcours = []
if not isinstance(types_parcours, list):
From 9e494d39cb5ef0aaed04b4d713efcc3ef43b09b0 Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 20:32:25 +0200
Subject: [PATCH 2/9] typo
---
app/auth/models.py | 2 +-
app/scodoc/sco_recapcomplet.py | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/app/auth/models.py b/app/auth/models.py
index 4ed1e41d..ad56c506 100644
--- a/app/auth/models.py
+++ b/app/auth/models.py
@@ -348,7 +348,7 @@ class User(UserMixin, db.Model):
return None
def get_nom_fmt(self):
- """Nom formatté: "Martin" """
+ """Nom formaté: "Martin" """
if self.nom:
return sco_etud.format_nom(self.nom, uppercase=False)
else:
diff --git a/app/scodoc/sco_recapcomplet.py b/app/scodoc/sco_recapcomplet.py
index 96d576bf..bbb1aee0 100644
--- a/app/scodoc/sco_recapcomplet.py
+++ b/app/scodoc/sco_recapcomplet.py
@@ -131,7 +131,7 @@ def formsemestre_recapcomplet(
for (format, label) in (
("html", "Tableau"),
("evals", "Avec toutes les évaluations"),
- ("xlsx", "Excel (non formatté)"),
+ ("xlsx", "Excel (non formaté)"),
("xlsall", "Excel avec évaluations"),
("xml", "Bulletins XML (obsolète)"),
("json", "Bulletins JSON"),
From 01af698c6c6b7c741858d08ec7e5282fb71a2bfe Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 21:16:07 +0200
Subject: [PATCH 3/9] Import XML: remove useless route and set default module
type
---
app/scodoc/sco_formations.py | 5 +++++
app/views/notes.py | 11 -----------
2 files changed, 5 insertions(+), 11 deletions(-)
diff --git a/app/scodoc/sco_formations.py b/app/scodoc/sco_formations.py
index 178138b2..6c6e7019 100644
--- a/app/scodoc/sco_formations.py
+++ b/app/scodoc/sco_formations.py
@@ -256,6 +256,11 @@ def formation_import_xml(doc: str, import_tags=True):
mod_info[1]["formation_id"] = formation_id
mod_info[1]["matiere_id"] = mat_id
mod_info[1]["ue_id"] = ue_id
+ mod_info[1]["module_type"] = (
+ scu.ModuleType.STANDARD
+ if mod_info[1]["module_type"] is None
+ else mod_info[1]["module_type"]
+ )
mod_id = sco_edit_module.do_module_create(mod_info[1])
if xml_module_id:
modules_old2new[int(xml_module_id)] = mod_id
diff --git a/app/views/notes.py b/app/views/notes.py
index e66a42f7..86c4f6f7 100644
--- a/app/views/notes.py
+++ b/app/views/notes.py
@@ -648,17 +648,6 @@ def formation_export(formation_id, export_ids=False, format=None):
)
-@bp.route("/formation_import_xml")
-@scodoc
-@permission_required(Permission.ScoChangeFormation)
-@scodoc7func
-def formation_import_xml(file):
- "import d'une formation en XML"
- log("formation_import_xml")
- doc = file.read()
- return sco_formations.formation_import_xml(doc)
-
-
@bp.route("/formation_import_xml_form", methods=["GET", "POST"])
@scodoc
@permission_required(Permission.ScoChangeFormation)
From 29bdfb5cb0e44e4e39a2ac6ada13a0a31e27696c Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 21:19:12 +0200
Subject: [PATCH 4/9] oups
---
app/scodoc/sco_formations.py | 7 ++-----
1 file changed, 2 insertions(+), 5 deletions(-)
diff --git a/app/scodoc/sco_formations.py b/app/scodoc/sco_formations.py
index 6c6e7019..ae6ecc0b 100644
--- a/app/scodoc/sco_formations.py
+++ b/app/scodoc/sco_formations.py
@@ -256,11 +256,8 @@ def formation_import_xml(doc: str, import_tags=True):
mod_info[1]["formation_id"] = formation_id
mod_info[1]["matiere_id"] = mat_id
mod_info[1]["ue_id"] = ue_id
- mod_info[1]["module_type"] = (
- scu.ModuleType.STANDARD
- if mod_info[1]["module_type"] is None
- else mod_info[1]["module_type"]
- )
+ if not "module_type" in mod_info[1]:
+ mod_info[1]["module_type"] = scu.ModuleType.STANDARD
mod_id = sco_edit_module.do_module_create(mod_info[1])
if xml_module_id:
modules_old2new[int(xml_module_id)] = mod_id
From 7b4451918279950f9e8695525e34ce27db8d0df9 Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 21:41:46 +0200
Subject: [PATCH 5/9] Fix: pe_settag
---
app/pe/pe_settag.py | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/app/pe/pe_settag.py b/app/pe/pe_settag.py
index f4ada213..be5028f3 100644
--- a/app/pe/pe_settag.py
+++ b/app/pe/pe_settag.py
@@ -97,7 +97,7 @@ class SetTag(pe_tagtable.TableTag):
"""Mémorise les semtag nécessaires au jury."""
self.SemTagDict = {fid: SemTagDict[fid] for fid in self.get_Fids_in_settag()}
if PE_DEBUG >= 1:
- pe_print(u" => %d semestres fusionnés" % len(self.SemTagDict))
+ pe_print(" => %d semestres fusionnés" % len(self.SemTagDict))
# -------------------------------------------------------------------------------------------------------------------
def comp_data_settag(self):
@@ -210,7 +210,7 @@ class SetTagInterClasse(pe_tagtable.TableTag):
# -------------------------------------------------------------------------------------------------------------------
def __init__(self, nom_combinaison, diplome):
- pe_tagtable.TableTag.__init__(self, nom=nom_combinaison + "_%d" % diplome)
+ pe_tagtable.TableTag.__init__(self, nom=f"{nom_combinaison}_{diplome or ''}")
self.combinaison = nom_combinaison
self.parcoursDict = {}
@@ -243,7 +243,7 @@ class SetTagInterClasse(pe_tagtable.TableTag):
fid: SetTagDict[fid] for fid in self.get_Fids_in_settag() if fid != None
}
if PE_DEBUG >= 1:
- pe_print(u" => %d semestres utilisés" % len(self.SetTagDict))
+ pe_print(" => %d semestres utilisés" % len(self.SetTagDict))
# -------------------------------------------------------------------------------------------------------------------
def comp_data_settag(self):
From 9935aabf3be5f3c0f7b90f68ed0f7f82054e87a4 Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 22:02:21 +0200
Subject: [PATCH 6/9] Bonus IUT de Blagnac
---
app/comp/bonus_spo.py | 13 +++++++++++++
1 file changed, 13 insertions(+)
diff --git a/app/comp/bonus_spo.py b/app/comp/bonus_spo.py
index 86d444c3..8d426b68 100644
--- a/app/comp/bonus_spo.py
+++ b/app/comp/bonus_spo.py
@@ -481,6 +481,19 @@ class BonusBezier(BonusSportAdditif):
proportion_point = 0.03
+class BonusBlagnac(BonusSportAdditif):
+ """Calcul bonus modules optionnels (sport, culture), règle IUT de Blagnac.
+
+ Le bonus est égal à 5% des points au dessus de 10 à appliquer sur toutes
+ les UE du semestre, applicable dans toutes les formations (DUT, BUT, ...).
+ """
+
+ name = "bonus_iutblagnac"
+ displayed_name = "IUT de Blagnac"
+ proportion_point = 0.05
+ classic_use_bonus_ues = True # toujours sur les UE
+
+
class BonusBordeaux1(BonusSportMultiplicatif):
"""Calcul bonus modules optionnels (sport, culture), règle IUT Bordeaux 1,
sur moyenne générale et UEs.
From 1209cf4ef3a8475fd5c82ceadbc86a409dcf48de Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Tue, 10 May 2022 22:40:00 +0200
Subject: [PATCH 7/9] formatage
---
app/scodoc/sco_formsemestre_exterieurs.py | 23 +++++++++++------------
1 file changed, 11 insertions(+), 12 deletions(-)
diff --git a/app/scodoc/sco_formsemestre_exterieurs.py b/app/scodoc/sco_formsemestre_exterieurs.py
index 9dfcbbb6..0da85f2c 100644
--- a/app/scodoc/sco_formsemestre_exterieurs.py
+++ b/app/scodoc/sco_formsemestre_exterieurs.py
@@ -262,7 +262,7 @@ def formsemestre_ext_edit_ue_validations(formsemestre_id, etudid):
)
-def _make_page(etud, sem, tf, message=""):
+def _make_page(etud: dict, sem, tf, message="") -> list:
formsemestre = FormSemestre.query.get_or_404(sem["formsemestre_id"])
nt: NotesTableCompat = res_sem.load_formsemestre_results(formsemestre)
moy_gen = nt.get_etud_moy_gen(etud["etudid"])
@@ -277,21 +277,20 @@ def _make_page(etud, sem, tf, message=""):
"""
% etud,
- """La moyenne de ce semestre serait:
- %s / 20
+ f"""
La moyenne de ce semestre serait:
+ {moy_gen} / 20
- """
- % moy_gen,
+ """,
'',
tf[1],
"
",
- """
- """
- % (sem["formsemestre_id"], etud["etudid"]),
+ f"""
+ """,
html_sco_header.sco_footer(),
]
return H
From f9ec454da5a038b4c364e854deafc985283f955b Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Wed, 11 May 2022 00:59:51 +0200
Subject: [PATCH 8/9] =?UTF-8?q?API:=20revision=20pour=20multi-d=C3=A9parte?=
=?UTF-8?q?ments?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
app/api/absences.py | 133 ++++---
app/api/etudiants.py | 544 ++++++++++++++++-------------
app/models/etudiants.py | 1 +
tests/api/exemple-api-basic.py | 25 +-
tests/api/test_api_absences.py | 33 +-
tests/api/test_api_etudiants.py | 2 +-
tests/api/test_api_formsemestre.py | 9 +-
7 files changed, 396 insertions(+), 351 deletions(-)
diff --git a/app/api/absences.py b/app/api/absences.py
index 44b1c350..51c4673e 100644
--- a/app/api/absences.py
+++ b/app/api/absences.py
@@ -5,73 +5,62 @@ from flask import jsonify
from app.api import bp
from app.api.errors import error_response
from app.api.auth import token_auth, token_permission_required
-from app.api.tools import get_etud_from_etudid_or_nip_or_ine
-from app.scodoc import notesdb as ndb
+from app.models import Identite
+from app.scodoc import notesdb as ndb
from app.scodoc import sco_abs
-from app.scodoc.sco_groups import get_group_members
from app.scodoc.sco_permissions import Permission
@bp.route("/absences/etudid/", methods=["GET"])
-@bp.route("/absences/nip/", methods=["GET"])
-@bp.route("/absences/ine/", methods=["GET"])
@token_auth.login_required
@token_permission_required(Permission.APIView)
-def absences(etudid: int = None, nip: int = None, ine: int = None):
+def absences(etudid: int = None):
"""
Retourne la liste des absences d'un étudiant donné
etudid : l'etudid d'un étudiant
- nip: le code nip d'un étudiant
- ine : le code ine d'un étudiant
Exemple de résultat:
- [
- {
- "jour": "2022-04-15",
- "matin": true,
- "estabs": true,
- "estjust": true,
- "description": "",
- "begin": "2022-04-15 08:00:00",
- "end": "2022-04-15 11:59:59"
- },
- {
- "jour": "2022-04-15",
- "matin": false,
- "estabs": true,
- "estjust": false,
- "description": "",
- "begin": "2022-04-15 12:00:00",
- "end": "2022-04-15 17:59:59"
- }
- ]
+ [
+ {
+ "jour": "2022-04-15",
+ "matin": true,
+ "estabs": true,
+ "estjust": true,
+ "description": "",
+ "begin": "2022-04-15 08:00:00",
+ "end": "2022-04-15 11:59:59"
+ },
+ {
+ "jour": "2022-04-15",
+ "matin": false,
+ "estabs": true,
+ "estjust": false,
+ "description": "",
+ "begin": "2022-04-15 12:00:00",
+ "end": "2022-04-15 17:59:59"
+ }
+ ]
"""
- if etudid is None:
- # Récupération de l'étudiant
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
- if etud is None:
- return error_response(
- 404,
- message="id de l'étudiant (etudid, nip, ine) inconnu",
- )
- etudid = etud.etudid
-
- # Récupération des absences de l'étudiant
+ etud = Identite.query.get(etudid)
+ if etud is None:
+ return error_response(
+ 404,
+ message="id de l'étudiant (etudid, nip, ine) inconnu",
+ )
+ # Absences de l'étudiant
ndb.open_db_connection()
- absences = sco_abs.list_abs_date(etudid)
+ absences = sco_abs.list_abs_date(etud.id)
for absence in absences:
absence["jour"] = absence["jour"].isoformat()
return jsonify(absences)
@bp.route("/absences/etudid//just", methods=["GET"])
-@bp.route("/absences/nip//just", methods=["GET"])
-@bp.route("/absences/ine//just", methods=["GET"])
@token_auth.login_required
@token_permission_required(Permission.APIView)
-def absences_just(etudid: int = None, nip: int = None, ine: int = None):
+def absences_just(etudid: int = None):
"""
Retourne la liste des absences justifiées d'un étudiant donné
@@ -80,39 +69,37 @@ def absences_just(etudid: int = None, nip: int = None, ine: int = None):
ine : le code ine d'un étudiant
Exemple de résultat :
- [
- {
- "jour": "2022-04-15",
- "matin": true,
- "estabs": true,
- "estjust": true,
- "description": "",
- "begin": "2022-04-15 08:00:00",
- "end": "2022-04-15 11:59:59"
- },
- {
- "jour": "Fri, 15 Apr 2022 00:00:00 GMT",
- "matin": false,
- "estabs": true,
- "estjust": true,
- "description": "",
- "begin": "2022-04-15 12:00:00",
- "end": "2022-04-15 17:59:59"
- }
- ]
+ [
+ {
+ "jour": "2022-04-15",
+ "matin": true,
+ "estabs": true,
+ "estjust": true,
+ "description": "",
+ "begin": "2022-04-15 08:00:00",
+ "end": "2022-04-15 11:59:59"
+ },
+ {
+ "jour": "Fri, 15 Apr 2022 00:00:00 GMT",
+ "matin": false,
+ "estabs": true,
+ "estjust": true,
+ "description": "",
+ "begin": "2022-04-15 12:00:00",
+ "end": "2022-04-15 17:59:59"
+ }
+ ]
"""
- if etudid is None:
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
- if etud is None:
- return error_response(
- 404,
- message="id de l'étudiant (etudid, nip, ine) inconnu",
- )
- etudid = etud.etudid
+ etud = Identite.query.get(etudid)
+ if etud is None:
+ return error_response(
+ 404,
+ message="id de l'étudiant (etudid, nip, ine) inconnu",
+ )
- # Récupération des absences justifiées de l'étudiant
+ # Absences justifiées de l'étudiant
abs_just = [
- absence for absence in sco_abs.list_abs_date(etudid) if absence["estjust"]
+ absence for absence in sco_abs.list_abs_date(etud.id) if absence["estjust"]
]
for absence in abs_just:
absence["jour"] = absence["jour"].isoformat()
diff --git a/app/api/etudiants.py b/app/api/etudiants.py
index a41634f9..a193553f 100644
--- a/app/api/etudiants.py
+++ b/app/api/etudiants.py
@@ -1,14 +1,20 @@
-#################################################### Etudiants ########################################################
+##############################################################################
+# ScoDoc
+# Copyright (c) 1999 - 2022 Emmanuel Viennet. All rights reserved.
+# See LICENSE
+##############################################################################
+
+"""
+ API : accès aux étudiants
+"""
from flask import jsonify
import app
-from app import models
from app.api import bp
from app.api.errors import error_response
from app.api.auth import token_auth, token_permission_required
-from app.api.tools import get_etud_from_etudid_or_nip_or_ine
-from app.models import FormSemestreInscription, FormSemestre, Identite
+from app.models import Departement, FormSemestreInscription, FormSemestre, Identite
from app.scodoc import sco_bulletins
from app.scodoc import sco_groups
from app.scodoc.sco_permissions import Permission
@@ -59,53 +65,102 @@ def etudiants_courant(long=False):
@bp.route("/etudiant/ine/", methods=["GET"])
@token_auth.login_required
@token_permission_required(Permission.APIView)
-def etudiant(etudid: int = None, nip: int = None, ine: int = None):
+def etudiant(etudid: int = None, nip: str = None, ine: str = None):
"""
- Retourne les informations de l'étudiant correspondant à l'id passé en paramètres.
+ Retourne les informations de l'étudiant correspondant, ou 404 si non trouvé.
- etudid : l'etudid d'un étudiant
- nip : le code nip d'un étudiant
- ine : le code ine d'un étudiant
+ etudid : l'etudid de l'étudiant
+ nip : le code nip de l'étudiant
+ ine : le code ine de l'étudiant
+
+ Les codes INE et NIP sont uniques au sein d'un département.
+ Si plusieurs objets ont le même code, on ramène le plus récemment inscrit.
Exemple de résultat :
- {
- "civilite": "X",
- "code_ine": "1",
- "code_nip": "1",
- "date_naissance": "",
- "email": "SACHA.COSTA@example.com",
- "emailperso": "",
- "etudid": 1,
- "nom": "COSTA",
- "prenom": "SACHA",
- "nomprenom": "Sacha COSTA",
- "lieu_naissance": "",
- "dept_naissance": "",
- "nationalite": "",
- "boursier": "",
- "id": 1,
- "codepostaldomicile": "",
- "paysdomicile": "",
- "telephonemobile": "",
- "typeadresse": "domicile",
- "domicile": "",
- "villedomicile": "",
- "telephone": "",
- "fax": "",
- "description": ""
- }
+ {
+ "civilite": "X",
+ "code_ine": "1",
+ "code_nip": "1",
+ "date_naissance": "",
+ "email": "SACHA.COSTA@example.com",
+ "emailperso": "",
+ "etudid": 1,
+ "nom": "COSTA",
+ "prenom": "SACHA",
+ "nomprenom": "Sacha COSTA",
+ "lieu_naissance": "",
+ "dept_naissance": "",
+ "nationalite": "",
+ "boursier": "",
+ "id": 1,
+ "codepostaldomicile": "",
+ "paysdomicile": "",
+ "telephonemobile": "",
+ "typeadresse": "domicile",
+ "domicile": "",
+ "villedomicile": "",
+ "telephone": "",
+ "fax": "",
+ "description": ""
+ }
"""
- # Récupération de l'étudiant
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
+ if etudid is not None:
+ etud = Identite.query.get(etudid)
+ else:
+ if nip is not None:
+ query = Identite.query.filter_by(code_nip=nip)
+ elif ine is not None:
+ query = Identite.query.filter_by(code_ine=ine)
+ else:
+ return error_response(
+ 404,
+ message="parametre manquant",
+ )
+ if query.count() > 1: # cas rare d'un étudiant présent dans plusieurs depts
+ etuds = []
+ for e in query:
+ admission = e.admission.first()
+ etuds.append((((admission.annee or 0) if admission else 0), e))
+ etuds.sort()
+ etud = etuds[-1][1]
+ else:
+ etud = query.first()
+
if etud is None:
return error_response(
404,
- message="id de l'étudiant (etudid, nip, ine) inconnu",
+ message="étudiant inconnu",
)
- # Mise en forme des données
- data = etud.to_dict_bul(include_urls=False)
- return jsonify(data)
+ return jsonify(etud.to_dict_bul(include_urls=False))
+
+
+@bp.route("/etudiants/etudid/", methods=["GET"])
+@bp.route("/etudiants/nip/", methods=["GET"])
+@bp.route("/etudiants/ine/", methods=["GET"])
+@token_auth.login_required
+@token_permission_required(Permission.APIView)
+def etudiants(etudid: int = None, nip: str = None, ine: str = None):
+ """
+ Info sur le ou les étudiants correspondant. Comme /etudiant mais renvoie
+ toujours une liste.
+ Si non trouvé, liste vide, pas d'erreur.
+ Dans 99% des cas, la liste contient un seul étudiant, mais si l'étudiant a
+ été inscrit dans plusieurs départements, on a plusieurs objets (1 par dept.).
+ """
+ if etudid is not None:
+ query = Identite.query.filter_by(id=etudid)
+ elif nip is not None:
+ query = Identite.query.filter_by(code_nip=nip)
+ elif ine is not None:
+ query = Identite.query.filter_by(code_ine=ine)
+ else:
+ return error_response(
+ 404,
+ message="parametre manquant",
+ )
+
+ return jsonify([etud.to_dict_bul(include_urls=False) for etud in query])
@bp.route("/etudiant/etudid//formsemestres")
@@ -115,56 +170,65 @@ def etudiant(etudid: int = None, nip: int = None, ine: int = None):
@token_permission_required(Permission.APIView)
def etudiant_formsemestres(etudid: int = None, nip: int = None, ine: int = None):
"""
- Retourne la liste des semestres qu'un étudiant a suivis, triés par ordre chronologique.
-
- etudid : l'etudid d'un étudiant
- nip : le code nip d'un étudiant
- ine : le code ine d'un étudiant
+ Liste des semestres qu'un étudiant a suivi, triés par ordre chronologique.
+ Accès par etudid, nip ou ine
Exemple de résultat :
- [
- {
- "date_fin": "31/08/2022",
- "resp_can_edit": false,
- "dept_id": 1,
- "etat": true,
- "resp_can_change_ens": true,
- "id": 1,
- "modalite": "FI",
- "ens_can_edit_eval": false,
- "formation_id": 1,
- "gestion_compensation": false,
- "elt_sem_apo": null,
- "semestre_id": 1,
- "bul_hide_xml": false,
- "elt_annee_apo": null,
- "titre": "Semestre test",
- "block_moyennes": false,
- "scodoc7_id": null,
- "date_debut": "01/09/2021",
- "gestion_semestrielle": false,
- "bul_bgcolor": "white",
- "formsemestre_id": 1,
- "titre_num": "Semestre test semestre 1",
- "date_debut_iso": "2021-09-01",
- "date_fin_iso": "2022-08-31",
- "responsables": []
- },
- ...
- ]
+ [
+ {
+ "date_fin": "31/08/2022",
+ "resp_can_edit": false,
+ "dept_id": 1,
+ "etat": true,
+ "resp_can_change_ens": true,
+ "id": 1,
+ "modalite": "FI",
+ "ens_can_edit_eval": false,
+ "formation_id": 1,
+ "gestion_compensation": false,
+ "elt_sem_apo": null,
+ "semestre_id": 1,
+ "bul_hide_xml": false,
+ "elt_annee_apo": null,
+ "titre": "Semestre test",
+ "block_moyennes": false,
+ "scodoc7_id": null,
+ "date_debut": "01/09/2021",
+ "gestion_semestrielle": false,
+ "bul_bgcolor": "white",
+ "formsemestre_id": 1,
+ "titre_num": "Semestre test semestre 1",
+ "date_debut_iso": "2021-09-01",
+ "date_fin_iso": "2022-08-31",
+ "responsables": []
+ },
+ ...
+ ]
"""
- # Récupération de l'étudiant
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
- if etud is None:
+ if etudid is not None:
+ query = FormSemestre.query.filter(
+ FormSemestreInscription.etudid == etudid,
+ FormSemestreInscription.formsemestre_id == FormSemestre.id,
+ )
+ elif nip is not None:
+ query = FormSemestre.query.filter(
+ Identite.code_nip == nip,
+ FormSemestreInscription.etudid == Identite.id,
+ FormSemestreInscription.formsemestre_id == FormSemestre.id,
+ )
+ elif ine is not None:
+ query = FormSemestre.query.filter(
+ Identite.code_ine == ine,
+ FormSemestreInscription.etudid == Identite.id,
+ FormSemestreInscription.formsemestre_id == FormSemestre.id,
+ )
+ else:
return error_response(
404,
- message="id de l'étudiant (etudid, nip, ine) inconnu",
+ message="parametre manquant",
)
- formsemestres = models.FormSemestre.query.filter(
- models.FormSemestreInscription.etudid == etud.id,
- models.FormSemestreInscription.formsemestre_id == models.FormSemestre.id,
- ).order_by(models.FormSemestre.date_debut)
+ formsemestres = query.order_by(FormSemestre.date_debut)
return jsonify([formsemestre.to_dict() for formsemestre in formsemestres])
@@ -204,8 +268,8 @@ def etudiant_formsemestres(etudid: int = None, nip: int = None, ine: int = None)
def etudiant_bulletin_semestre(
formsemestre_id,
etudid: int = None,
- nip: int = None,
- ine: int = None,
+ nip: str = None,
+ ine: str = None,
version="long",
):
"""
@@ -216,12 +280,12 @@ def etudiant_bulletin_semestre(
nip : le code nip d'un étudiant
ine : le code ine d'un étudiant
Exemple de résultat :
- {
- "version": "0",
- "type": "BUT",
- "date": "2022-04-27T07:18:16.450634Z",
- "publie": true,
- "etudiant": {
+ {
+ "version": "0",
+ "type": "BUT",
+ "date": "2022-04-27T07:18:16.450634Z",
+ "publie": true,
+ "etudiant": {
"civilite": "X",
"code_ine": "1",
"code_nip": "1",
@@ -247,17 +311,17 @@ def etudiant_bulletin_semestre(
"villedomicile": "",
"telephone": "",
"fax": "",
- "description": ""
- },
- "formation": {
+ "description": "",
+ },
+ "formation": {
"id": 1,
"acronyme": "BUT R&T",
"titre_officiel": "Bachelor technologique r\u00e9seaux et t\u00e9l\u00e9communications",
- "titre": "BUT R&T"
- },
- "formsemestre_id": 1,
- "etat_inscription": "I",
- "options": {
+ "titre": "BUT R&T",
+ },
+ "formsemestre_id": 1,
+ "etat_inscription": "I",
+ "options": {
"show_abs": true,
"show_abs_modules": false,
"show_ects": true,
@@ -276,128 +340,113 @@ def etudiant_bulletin_semestre(
"show_temporary": true,
"temporary_txt": "Provisoire",
"show_uevalid": true,
- "show_date_inscr": true
- },
- "ressources": {
+ "show_date_inscr": true,
+ },
+ "ressources": {
"R101": {
- "id": 1,
- "titre": "Initiation aux r\u00e9seaux informatiques",
- "code_apogee": null,
- "url": "/ScoDoc/TAPI/Scolarite/Notes/moduleimpl_status?moduleimpl_id=1",
- "moyenne": {},
- "evaluations": [
- {
- "id": 1,
- "description": "eval1",
- "date": "2022-04-20",
- "heure_debut": "08:00",
- "heure_fin": "09:00",
- "coef": "01.00",
- "poids": {
- "RT1.1": 1.0,
- },
- "note": {
- "value": "12.00",
- "min": "00.00",
- "max": "18.00",
- "moy": "10.88"
- },
- "url": "/ScoDoc/TAPI/Scolarite/Notes/evaluation_listenotes?evaluation_id=1"
- }
- ]
+ "id": 1,
+ "titre": "Initiation aux r\u00e9seaux informatiques",
+ "code_apogee": null,
+ "url": "/ScoDoc/TAPI/Scolarite/Notes/moduleimpl_status?moduleimpl_id=1",
+ "moyenne": {},
+ "evaluations": [
+ {
+ "id": 1,
+ "description": "eval1",
+ "date": "2022-04-20",
+ "heure_debut": "08:00",
+ "heure_fin": "09:00",
+ "coef": "01.00",
+ "poids": {
+ "RT1.1": 1.0,
+ },
+ "note": {
+ "value": "12.00",
+ "min": "00.00",
+ "max": "18.00",
+ "moy": "10.88",
+ },
+ "url": "/ScoDoc/TAPI/Scolarite/Notes/evaluation_listenotes?evaluation_id=1",
+ }
+ ],
},
- },
- "saes": {
+ },
+ "saes": {
"SAE11": {
- "id": 2,
- "titre": "Se sensibiliser \u00e0 l'hygi\u00e8ne informatique et \u00e0 la cybers\u00e9curit\u00e9",
- "code_apogee": null,
- "url": "/ScoDoc/TAPI/Scolarite/Notes/moduleimpl_status?moduleimpl_id=2",
- "moyenne": {},
- "evaluations": []
+ "id": 2,
+ "titre": "Se sensibiliser \u00e0 l'hygi\u00e8ne informatique et \u00e0 la cybers\u00e9curit\u00e9",
+ "code_apogee": null,
+ "url": "/ScoDoc/TAPI/Scolarite/Notes/moduleimpl_status?moduleimpl_id=2",
+ "moyenne": {},
+ "evaluations": [],
},
- },
- "ues": {
+ },
+ "ues": {
"RT1.1": {
- "id": 1,
- "titre": "Administrer les r\u00e9seaux et l\u2019Internet",
- "numero": 1,
- "type": 0,
- "color": "#B80004",
- "competence": null,
- "moyenne": {
- "value": "08.50",
- "min": "06.00",
- "max": "16.50",
- "moy": "11.31",
- "rang": "12",
- "total": 16
- },
- "bonus": "00.00",
- "malus": "00.00",
- "capitalise": null,
- "ressources": {
- "R101": {
- "id": 1,
- "coef": 12.0,
- "moyenne": "12.00"
+ "id": 1,
+ "titre": "Administrer les r\u00e9seaux et l\u2019Internet",
+ "numero": 1,
+ "type": 0,
+ "color": "#B80004",
+ "competence": null,
+ "moyenne": {
+ "value": "08.50",
+ "min": "06.00",
+ "max": "16.50",
+ "moy": "11.31",
+ "rang": "12",
+ "total": 16,
},
- },
- "saes": {
- "SAE11": {
- "id": 2,
- "coef": 16.0,
- "moyenne": "~"
+ "bonus": "00.00",
+ "malus": "00.00",
+ "capitalise": null,
+ "ressources": {
+ "R101": {"id": 1, "coef": 12.0, "moyenne": "12.00"},
},
- },
- "ECTS": {
- "acquis": 0.0,
- "total": 12.0
- }
+ "saes": {
+ "SAE11": {"id": 2, "coef": 16.0, "moyenne": "~"},
+ },
+ "ECTS": {"acquis": 0.0, "total": 12.0},
},
- "semestre": {
- "etapes": [],
- "date_debut": "2021-09-01",
- "date_fin": "2022-08-31",
- "annee_universitaire": "2021 - 2022",
- "numero": 1,
- "inscription": "",
- "groupes": [],
- "absences": {
- "injustifie": 1,
- "total": 2
+ "semestre": {
+ "etapes": [],
+ "date_debut": "2021-09-01",
+ "date_fin": "2022-08-31",
+ "annee_universitaire": "2021 - 2022",
+ "numero": 1,
+ "inscription": "",
+ "groupes": [],
+ "absences": {"injustifie": 1, "total": 2},
+ "ECTS": {"acquis": 0, "total": 30.0},
+ "notes": {"value": "10.60", "min": "02.40", "moy": "11.05", "max": "17.40"},
+ "rang": {"value": "10", "total": 16},
},
- "ECTS": {
- "acquis": 0,
- "total": 30.0
- },
- "notes": {
- "value": "10.60",
- "min": "02.40",
- "moy": "11.05",
- "max": "17.40"
- },
- "rang": {
- "value": "10",
- "total": 16
- }
- }
- }
+ },
+ }
"""
- formsemestre = models.FormSemestre.query.filter_by(
- id=formsemestre_id
- ).first_or_404()
+ formsemestre = FormSemestre.query.filter_by(id=formsemestre_id).first_or_404()
+ dept = Departement.query.filter_by(id=formsemestre.dept_id).first_or_404()
- dept = models.Departement.query.filter_by(id=formsemestre.dept_id).first_or_404()
+ if etudid is not None:
+ query = Identite.query.filter_by(id=etudid)
+ elif nip is not None:
+ query = Identite.query.filter_by(code_nip=nip, dept_id=dept.id)
+ elif ine is not None:
+ query = Identite.query.filter_by(code_ine=ine, dept_id=dept.id)
+ else:
+ return error_response(
+ 404,
+ message="parametre manquant",
+ )
- app.set_sco_dept(dept.acronym)
-
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
+ etud = query.first()
if etud is None:
return error_response(
404,
message="id de l'étudiant (etudid, nip, ine) inconnu",
)
+
+ app.set_sco_dept(dept.acronym)
return sco_bulletins.get_formsemestre_bulletin_etud_json(
formsemestre, etud, version
)
@@ -429,44 +478,57 @@ def etudiant_groups(
ine : le code ine d'un étudiant
Exemple de résultat :
- [
- {
- "partition_id": 1,
- "id": 1,
- "formsemestre_id": 1,
- "partition_name": null,
- "numero": 0,
- "bul_show_rank": false,
- "show_in_lists": true,
- "group_id": 1,
- "group_name": null
- },
- {
- "partition_id": 2,
- "id": 2,
- "formsemestre_id": 1,
- "partition_name": "TD",
- "numero": 1,
- "bul_show_rank": false,
- "show_in_lists": true,
- "group_id": 2,
- "group_name": "A"
- }
- ]
+ [
+ {
+ "partition_id": 1,
+ "id": 1,
+ "formsemestre_id": 1,
+ "partition_name": null,
+ "numero": 0,
+ "bul_show_rank": false,
+ "show_in_lists": true,
+ "group_id": 1,
+ "group_name": null
+ },
+ {
+ "partition_id": 2,
+ "id": 2,
+ "formsemestre_id": 1,
+ "partition_name": "TD",
+ "numero": 1,
+ "bul_show_rank": false,
+ "show_in_lists": true,
+ "group_id": 2,
+ "group_name": "A"
+ }
+ ]
"""
- if etudid is None:
- etud = get_etud_from_etudid_or_nip_or_ine(etudid, nip, ine)
- if etud is None:
- return error_response(
- 404,
- message="id de l'étudiant (etudid, nip, ine) inconnu",
- )
- etudid = etud.etudid
- # Récupération du formsemestre
- sem = models.FormSemestre.query.filter_by(id=formsemestre_id).first_or_404()
- dept = models.Departement.query.get(sem.dept_id)
+ formsemestre = FormSemestre.query.filter_by(id=formsemestre_id).first()
+ if formsemestre is None:
+ return error_response(
+ 404,
+ message="formsemestre inconnu",
+ )
+ dept = Departement.query.get(formsemestre.dept_id)
+ if etudid is not None:
+ query = Identite.query.filter_by(id=etudid)
+ elif nip is not None:
+ query = Identite.query.filter_by(code_nip=nip, dept_id=dept.id)
+ elif ine is not None:
+ query = Identite.query.filter_by(code_ine=ine, dept_id=dept.id)
+ else:
+ return error_response(
+ 404,
+ message="parametre manquant",
+ )
+ etud = query.first()
+ if etud is None:
+ return error_response(
+ 404,
+ message="etudiant inconnu",
+ )
app.set_sco_dept(dept.acronym)
- data = sco_groups.get_etud_groups(etudid, sem.id)
+ data = sco_groups.get_etud_groups(etud.id, formsemestre.id)
return jsonify(data)
diff --git a/app/models/etudiants.py b/app/models/etudiants.py
index 912136e6..44b7ec36 100644
--- a/app/models/etudiants.py
+++ b/app/models/etudiants.py
@@ -178,6 +178,7 @@ class Identite(db.Model):
"date_naissance": self.date_naissance.strftime("%d/%m/%Y")
if self.date_naissance
else "",
+ "dept_id": self.dept_id,
"email": self.get_first_email() or "",
"emailperso": self.get_first_email("emailperso"),
"etudid": self.id,
diff --git a/tests/api/exemple-api-basic.py b/tests/api/exemple-api-basic.py
index a9f10928..1c7d61b8 100644
--- a/tests/api/exemple-api-basic.py
+++ b/tests/api/exemple-api-basic.py
@@ -26,13 +26,17 @@ import urllib3
from pprint import pprint as pp
# --- Lecture configuration (variables d'env ou .env)
-BASEDIR = os.path.abspath(os.path.dirname(__file__))
+try:
+ BASEDIR = os.path.abspath(os.path.dirname(__file__))
+except NameError:
+ BASEDIR = "."
+
load_dotenv(os.path.join(BASEDIR, ".env"))
CHK_CERT = bool(int(os.environ.get("CHECK_CERTIFICATE", False)))
SCODOC_URL = os.environ["SCODOC_URL"]
API_URL = SCODOC_URL + "/ScoDoc/api"
SCODOC_USER = os.environ["SCODOC_USER"]
-SCODOC_PASSWORD = os.environ["SCODOC_PASSWD"]
+SCODOC_PASSWORD = os.environ["SCODOC_PASSWORD"]
print(f"SCODOC_URL={SCODOC_URL}")
print(f"API URL={API_URL}")
@@ -90,6 +94,23 @@ formsemestre_id = 1028 # A adapter
etudid = 14721
bul_dut = GET(f"/etudiant/etudid/{etudid}/formsemestre/{formsemestre_id}/bulletin")
+
+# Infos sur un étudiant
+etudid = 3561
+code_nip = "11303314"
+etud = GET(f"/etudiant/etudid/{etudid}")
+print(etud)
+
+etud = GET(f"/etudiant/nip/{code_nip}")
+print(etud)
+
+sems = GET(f"/etudiant/etudid/{etudid}/formsemestres")
+print("\n".join([s["titre_num"] for s in sems]))
+
+sems = GET(f"/etudiant/nip/{code_nip}/formsemestres")
+print("\n".join([s["titre_num"] for s in sems]))
+
+
# # --- Recupere la liste de tous les semestres:
# sems = GET(s, "Notes/formsemestre_list?format=json", "Aucun semestre !")
diff --git a/tests/api/test_api_absences.py b/tests/api/test_api_absences.py
index a0a90c57..687af7d6 100644
--- a/tests/api/test_api_absences.py
+++ b/tests/api/test_api_absences.py
@@ -22,8 +22,7 @@ from tests.api.setup_test_api import API_URL, CHECK_CERTIFICATE, api_headers
# Etudiant pour les tests
ETUDID = 1
-INE = "1"
-NIP = "1"
+
# absences
def test_absences(api_headers):
@@ -37,20 +36,6 @@ def test_absences(api_headers):
)
assert r.status_code == 200
- r = requests.get(
- f"{API_URL}/absences/nip/{NIP}",
- headers=api_headers,
- verify=CHECK_CERTIFICATE,
- )
- assert r.status_code == 200
-
- r = requests.get(
- f"{API_URL}/absences/ine/{INE}",
- headers=api_headers,
- verify=CHECK_CERTIFICATE,
- )
- assert r.status_code == 200
-
# absences_justify
def test_absences_justify(api_headers):
@@ -65,22 +50,6 @@ def test_absences_justify(api_headers):
assert r.status_code == 200
# TODO vérifier résultat
- r = requests.get(
- API_URL + f"/absences/nip/{NIP}/just",
- headers=api_headers,
- verify=CHECK_CERTIFICATE,
- )
- assert r.status_code == 200
- # TODO vérifier résultat
-
- r = requests.get(
- API_URL + f"/absences/ine/{INE}/just",
- headers=api_headers,
- verify=CHECK_CERTIFICATE,
- )
- assert r.status_code == 200
- # TODO vérifier résultat
-
# XXX TODO
# def test_abs_groupe_etat(api_headers):
diff --git a/tests/api/test_api_etudiants.py b/tests/api/test_api_etudiants.py
index c0ccfc9e..4c88d861 100644
--- a/tests/api/test_api_etudiants.py
+++ b/tests/api/test_api_etudiants.py
@@ -93,7 +93,7 @@ def test_etudiant(api_headers):
)
assert r.status_code == 200
etud = r.json()
- assert len(etud) == 24
+ assert len(etud) == 25
fields_ok = verify_fields(etud, ETUD_FIELDS)
assert fields_ok is True
diff --git a/tests/api/test_api_formsemestre.py b/tests/api/test_api_formsemestre.py
index 57d79772..bdcf7e93 100644
--- a/tests/api/test_api_formsemestre.py
+++ b/tests/api/test_api_formsemestre.py
@@ -24,6 +24,11 @@ from tests.api.setup_test_api import API_URL, CHECK_CERTIFICATE, api_headers
from tests.api.tools_test_api import MODIMPL_FIELDS, verify_fields
from tests.api.tools_test_api import FSEM_FIELDS, UE_FIELDS, MODULE_FIELDS
+# Etudiant pour les tests
+ETUDID = 1
+NIP = "1"
+INE = "INE1"
+
def test_formsemestre(api_headers):
"""
@@ -53,7 +58,7 @@ def test_etudiant_bulletin(api_headers):
bull_a = r.json()
r = requests.get(
- f"{API_URL}/etudiant/nip/1/formsemestre/{formsemestre_id}/bulletin",
+ f"{API_URL}/etudiant/nip/{NIP}/formsemestre/{formsemestre_id}/bulletin",
headers=api_headers,
verify=CHECK_CERTIFICATE,
)
@@ -61,7 +66,7 @@ def test_etudiant_bulletin(api_headers):
bull_b = r.json()
r = requests.get(
- f"{API_URL}/etudiant/ine/1/formsemestre/{formsemestre_id}/bulletin",
+ f"{API_URL}/etudiant/ine/{INE}/formsemestre/{formsemestre_id}/bulletin",
headers=api_headers,
verify=CHECK_CERTIFICATE,
)
From 54d65c01ae3825af33904f8b16597369efd5583e Mon Sep 17 00:00:00 2001
From: Emmanuel Viennet
Date: Wed, 11 May 2022 01:00:52 +0200
Subject: [PATCH 9/9] 9.2.20 (API)
---
sco_version.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/sco_version.py b/sco_version.py
index 027f1d59..aaeef296 100644
--- a/sco_version.py
+++ b/sco_version.py
@@ -1,7 +1,7 @@
# -*- mode: python -*-
# -*- coding: utf-8 -*-
-SCOVERSION = "9.2.19"
+SCOVERSION = "9.2.20"
SCONAME = "ScoDoc"