2021-12-08 22:33:32 +01:00
|
|
|
"""ScoDoc 9 models : Modules
|
|
|
|
"""
|
2023-06-13 21:28:37 +02:00
|
|
|
from operator import attrgetter
|
2023-02-28 22:06:25 +01:00
|
|
|
from flask import current_app
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
from app import db
|
|
|
|
from app.models import APO_CODE_STR_LEN
|
2022-10-30 22:42:10 +01:00
|
|
|
from app.models.but_refcomp import ApcParcours, app_critiques_modules, parcours_modules
|
2021-12-08 22:33:32 +01:00
|
|
|
from app.scodoc import sco_utils as scu
|
2023-02-12 13:36:47 +01:00
|
|
|
from app.scodoc.codes_cursus import UE_SPORT
|
2023-02-06 10:13:01 +01:00
|
|
|
from app.scodoc.sco_exceptions import ScoValueError
|
2022-11-25 22:32:06 +01:00
|
|
|
from app.scodoc.sco_utils import ModuleType
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Module(db.Model):
|
|
|
|
"""Module"""
|
|
|
|
|
|
|
|
__tablename__ = "notes_modules"
|
|
|
|
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
module_id = db.synonym("id")
|
|
|
|
titre = db.Column(db.Text())
|
|
|
|
abbrev = db.Column(db.Text()) # nom court
|
|
|
|
# certains départements ont des codes infiniment longs: donc Text !
|
|
|
|
code = db.Column(db.Text(), nullable=False)
|
|
|
|
heures_cours = db.Column(db.Float)
|
|
|
|
heures_td = db.Column(db.Float)
|
|
|
|
heures_tp = db.Column(db.Float)
|
|
|
|
coefficient = db.Column(db.Float) # coef PPN (sauf en APC)
|
|
|
|
ects = db.Column(db.Float) # Crédits ECTS
|
|
|
|
ue_id = db.Column(db.Integer, db.ForeignKey("notes_ue.id"), index=True)
|
|
|
|
formation_id = db.Column(db.Integer, db.ForeignKey("notes_formations.id"))
|
|
|
|
matiere_id = db.Column(db.Integer, db.ForeignKey("notes_matieres.id"))
|
|
|
|
# pas un id mais le numéro du semestre: 1, 2, ...
|
2021-12-16 16:27:35 +01:00
|
|
|
# note: en APC, le semestre qui fait autorité est celui de l'UE
|
2021-12-08 22:33:32 +01:00
|
|
|
semestre_id = db.Column(db.Integer, nullable=False, default=1, server_default="1")
|
2023-04-03 17:46:31 +02:00
|
|
|
numero = db.Column(db.Integer, nullable=False, default=0) # ordre de présentation
|
2021-12-08 22:33:32 +01:00
|
|
|
# id de l'element pedagogique Apogee correspondant:
|
|
|
|
code_apogee = db.Column(db.String(APO_CODE_STR_LEN))
|
2022-03-26 23:33:57 +01:00
|
|
|
# Type: ModuleType.STANDARD, MALUS, RESSOURCE, SAE (enum)
|
2022-02-15 21:55:21 +01:00
|
|
|
module_type = db.Column(db.Integer, nullable=False, default=0, server_default="0")
|
2021-12-08 22:33:32 +01:00
|
|
|
# Relations:
|
2023-01-12 08:52:32 -03:00
|
|
|
modimpls = db.relationship(
|
|
|
|
"ModuleImpl", backref="module", lazy="dynamic", cascade="all, delete-orphan"
|
|
|
|
)
|
2021-12-08 22:33:32 +01:00
|
|
|
ues_apc = db.relationship("UniteEns", secondary="module_ue_coef", viewonly=True)
|
|
|
|
tags = db.relationship(
|
|
|
|
"NotesTag",
|
|
|
|
secondary="notes_modules_tags",
|
|
|
|
lazy=True,
|
|
|
|
backref=db.backref("modules", lazy=True),
|
|
|
|
)
|
2022-05-01 23:58:41 +02:00
|
|
|
# BUT
|
|
|
|
parcours = db.relationship(
|
|
|
|
"ApcParcours",
|
|
|
|
secondary=parcours_modules,
|
|
|
|
lazy="subquery",
|
2022-05-02 14:39:06 +02:00
|
|
|
backref=db.backref("modules", lazy=True),
|
2023-06-03 23:18:54 +02:00
|
|
|
order_by="ApcParcours.numero, ApcParcours.code",
|
2022-05-02 14:39:06 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
app_critiques = db.relationship(
|
|
|
|
"ApcAppCritique",
|
|
|
|
secondary=app_critiques_modules,
|
|
|
|
lazy="subquery",
|
2022-05-01 23:58:41 +02:00
|
|
|
backref=db.backref("modules", lazy=True),
|
|
|
|
)
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
self.ue_coefs = []
|
|
|
|
super(Module, self).__init__(**kwargs)
|
|
|
|
|
|
|
|
def __repr__(self):
|
2022-10-29 08:22:17 +02:00
|
|
|
return f"""<Module{ModuleType(self.module_type or ModuleType.STANDARD).name
|
|
|
|
} id={self.id} code={self.code!r} semestre_id={self.semestre_id}>"""
|
2021-12-08 22:33:32 +01:00
|
|
|
|
2022-11-02 10:41:31 +01:00
|
|
|
def clone(self):
|
|
|
|
"""Create a new copy of this module."""
|
|
|
|
mod = Module(
|
|
|
|
titre=self.titre,
|
|
|
|
abbrev=self.abbrev,
|
|
|
|
code=self.code + "-copie",
|
|
|
|
heures_cours=self.heures_cours,
|
|
|
|
heures_td=self.heures_td,
|
|
|
|
heures_tp=self.heures_tp,
|
|
|
|
coefficient=self.coefficient,
|
|
|
|
ects=self.ects,
|
|
|
|
ue_id=self.ue_id,
|
|
|
|
matiere_id=self.matiere_id,
|
|
|
|
formation_id=self.formation_id,
|
|
|
|
semestre_id=self.semestre_id,
|
|
|
|
numero=self.numero, # il est conseillé de renuméroter
|
|
|
|
code_apogee="", # volontairement vide pour éviter les erreurs
|
|
|
|
module_type=self.module_type,
|
|
|
|
)
|
|
|
|
|
|
|
|
# Les tags:
|
|
|
|
for tag in self.tags:
|
|
|
|
mod.tags.append(tag)
|
|
|
|
# Les parcours
|
|
|
|
for parcour in self.parcours:
|
|
|
|
mod.parcours.append(parcour)
|
|
|
|
# Les AC
|
|
|
|
for app_critique in self.app_critiques:
|
|
|
|
mod.app_critiques.append(app_critique)
|
|
|
|
return mod
|
|
|
|
|
2022-07-21 14:44:19 +02:00
|
|
|
def to_dict(self, convert_objects=False, with_matiere=False, with_ue=False) -> dict:
|
2022-07-21 14:21:06 +02:00
|
|
|
"""If convert_objects, convert all attributes to native types
|
|
|
|
(suitable jor json encoding).
|
|
|
|
"""
|
2022-07-21 14:44:19 +02:00
|
|
|
d = dict(self.__dict__)
|
|
|
|
d.pop("_sa_instance_state", None)
|
2022-07-21 14:21:06 +02:00
|
|
|
if convert_objects:
|
2022-07-21 14:44:19 +02:00
|
|
|
d["parcours"] = [p.to_dict() for p in self.parcours]
|
2022-08-02 17:13:13 +02:00
|
|
|
d["ue_coefs"] = [
|
|
|
|
c.to_dict(convert_objects=convert_objects) for c in self.ue_coefs
|
|
|
|
]
|
2022-09-07 10:07:46 +02:00
|
|
|
d["app_critiques"] = {x.code: x.to_dict() for x in self.app_critiques}
|
2022-07-21 14:44:19 +02:00
|
|
|
if not with_matiere:
|
|
|
|
d.pop("matiere", None)
|
|
|
|
if not with_ue:
|
|
|
|
d.pop("ue", None)
|
|
|
|
if convert_objects and with_matiere:
|
|
|
|
d["matiere"] = self.matiere.to_dict(convert_objects=True)
|
|
|
|
if convert_objects and with_ue:
|
|
|
|
d["ue"] = self.ue.to_dict(convert_objects=True)
|
|
|
|
|
2021-12-08 22:33:32 +01:00
|
|
|
# ScoDoc7 output_formators: (backward compat)
|
2022-07-21 14:44:19 +02:00
|
|
|
d["module_id"] = self.id
|
|
|
|
d["heures_cours"] = 0.0 if self.heures_cours is None else self.heures_cours
|
|
|
|
d["heures_td"] = 0.0 if self.heures_td is None else self.heures_td
|
|
|
|
d["heures_tp"] = 0.0 if self.heures_tp is None else self.heures_tp
|
|
|
|
d["numero"] = 0 if self.numero is None else self.numero
|
|
|
|
d["coefficient"] = 0.0 if self.coefficient is None else self.coefficient
|
|
|
|
d["module_type"] = 0 if self.module_type is None else self.module_type
|
|
|
|
d["code_apogee"] = d["code_apogee"] or "" # pas de None
|
|
|
|
return d
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
def is_apc(self):
|
|
|
|
"True si module SAÉ ou Ressource"
|
|
|
|
return self.module_type and scu.ModuleType(self.module_type) in {
|
|
|
|
scu.ModuleType.RESSOURCE,
|
|
|
|
scu.ModuleType.SAE,
|
|
|
|
}
|
|
|
|
|
2022-08-25 12:04:10 +02:00
|
|
|
def type_name(self) -> str:
|
|
|
|
"Le nom du type de module, pour les humains (avec majuscules et accents)"
|
2021-12-08 22:33:32 +01:00
|
|
|
return scu.MODULE_TYPE_NAMES[self.module_type]
|
|
|
|
|
2022-08-25 12:04:10 +02:00
|
|
|
def type_abbrv(self) -> str:
|
|
|
|
"""Le nom du type de module, pour les styles CSS.
|
|
|
|
"mod", "malus", "res", "sae"
|
|
|
|
"""
|
2022-03-26 23:33:57 +01:00
|
|
|
return scu.ModuleType.get_abbrev(self.module_type)
|
|
|
|
|
2022-08-31 19:14:21 +02:00
|
|
|
def sort_key_apc(self) -> tuple:
|
|
|
|
"""Clé de tri pour avoir
|
|
|
|
présentation par type (res, sae), parcours, type, numéro
|
|
|
|
"""
|
|
|
|
if (
|
2022-09-01 15:25:34 +02:00
|
|
|
self.formation.referentiel_competence is None
|
|
|
|
or len(self.parcours)
|
|
|
|
== self.formation.referentiel_competence.parcours.count()
|
2022-08-31 19:14:21 +02:00
|
|
|
or len(self.parcours) == 0
|
|
|
|
):
|
|
|
|
key_parcours = ""
|
|
|
|
else:
|
|
|
|
key_parcours = "/".join([p.code for p in self.parcours])
|
2022-09-05 14:58:57 +02:00
|
|
|
return self.module_type, key_parcours, self.numero or 0
|
2022-08-31 19:14:21 +02:00
|
|
|
|
2021-12-08 22:33:32 +01:00
|
|
|
def set_ue_coef(self, ue, coef: float) -> None:
|
|
|
|
"""Set coef module vers cette UE"""
|
|
|
|
self.update_ue_coef_dict({ue.id: coef})
|
|
|
|
|
|
|
|
def set_ue_coef_dict(self, ue_coef_dict: dict) -> None:
|
|
|
|
"""set coefs vers les UE (remplace existants)
|
|
|
|
ue_coef_dict = { ue_id : coef }
|
|
|
|
Les coefs nuls (zéro) ne sont pas stockés: la relation est supprimée.
|
|
|
|
"""
|
2023-02-28 22:06:25 +01:00
|
|
|
if self.formation.has_locked_sems(self.ue.semestre_idx):
|
|
|
|
current_app.logguer.info(
|
|
|
|
f"set_ue_coef_dict: locked formation, ignoring request"
|
|
|
|
)
|
2023-02-06 10:13:01 +01:00
|
|
|
raise ScoValueError("Formation verrouillée")
|
2021-12-10 16:45:36 +01:00
|
|
|
changed = False
|
2021-12-08 22:33:32 +01:00
|
|
|
for ue_id, coef in ue_coef_dict.items():
|
2021-12-10 16:45:36 +01:00
|
|
|
# Existant ?
|
|
|
|
coefs = [c for c in self.ue_coefs if c.ue_id == ue_id]
|
|
|
|
if coefs:
|
|
|
|
ue_coef = coefs[0]
|
|
|
|
if coef == 0.0: # supprime ce coef
|
|
|
|
db.session.delete(ue_coef)
|
|
|
|
changed = True
|
|
|
|
elif coef != ue_coef.coef:
|
|
|
|
ue_coef.coef = coef
|
|
|
|
db.session.add(ue_coef)
|
|
|
|
changed = True
|
2021-12-08 22:33:32 +01:00
|
|
|
else:
|
2021-12-10 16:45:36 +01:00
|
|
|
# crée nouveau coef:
|
|
|
|
if coef != 0.0:
|
|
|
|
ue = UniteEns.query.get(ue_id)
|
|
|
|
ue_coef = ModuleUECoef(module=self, ue=ue, coef=coef)
|
2022-08-31 12:02:19 +02:00
|
|
|
db.session.add(ue_coef)
|
2021-12-10 16:45:36 +01:00
|
|
|
self.ue_coefs.append(ue_coef)
|
|
|
|
changed = True
|
|
|
|
if changed:
|
|
|
|
self.formation.invalidate_module_coefs()
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
def update_ue_coef_dict(self, ue_coef_dict: dict):
|
|
|
|
"""update coefs vers UE (ajoute aux existants)"""
|
2023-02-28 22:06:25 +01:00
|
|
|
if self.formation.has_locked_sems(self.ue.semestre_idx):
|
|
|
|
current_app.logguer.info(
|
|
|
|
f"update_ue_coef_dict: locked formation, ignoring request"
|
|
|
|
)
|
2023-02-06 10:13:01 +01:00
|
|
|
raise ScoValueError("Formation verrouillée")
|
2021-12-08 22:33:32 +01:00
|
|
|
current = self.get_ue_coef_dict()
|
|
|
|
current.update(ue_coef_dict)
|
|
|
|
self.set_ue_coef_dict(current)
|
|
|
|
|
|
|
|
def get_ue_coef_dict(self):
|
|
|
|
"""returns { ue_id : coef }"""
|
|
|
|
return {p.ue.id: p.coef for p in self.ue_coefs}
|
|
|
|
|
2023-03-21 21:14:38 +01:00
|
|
|
def get_ue_coef_dict_acronyme(self):
|
|
|
|
"""returns { ue_acronyme : coef }"""
|
|
|
|
return {p.ue.acronyme: p.coef for p in self.ue_coefs}
|
|
|
|
|
2021-12-08 22:33:32 +01:00
|
|
|
def delete_ue_coef(self, ue):
|
|
|
|
"""delete coef"""
|
2023-02-28 22:06:25 +01:00
|
|
|
if self.formation.has_locked_sems(self.ue.semestre_idx):
|
|
|
|
current_app.logguer.info(
|
2023-06-13 21:28:37 +02:00
|
|
|
"delete_ue_coef: locked formation, ignoring request"
|
2023-02-28 22:06:25 +01:00
|
|
|
)
|
2023-02-06 10:13:01 +01:00
|
|
|
raise ScoValueError("Formation verrouillée")
|
2021-12-08 22:33:32 +01:00
|
|
|
ue_coef = ModuleUECoef.query.get((self.id, ue.id))
|
|
|
|
if ue_coef:
|
|
|
|
db.session.delete(ue_coef)
|
|
|
|
self.formation.invalidate_module_coefs()
|
|
|
|
|
2021-12-10 01:55:13 +01:00
|
|
|
def get_ue_coefs_sorted(self):
|
2023-06-13 21:28:37 +02:00
|
|
|
"les coefs d'UE, trié par numéro et acronyme d'UE"
|
2021-12-10 01:55:13 +01:00
|
|
|
# je n'ai pas su mettre un order_by sur le backref sans avoir
|
|
|
|
# à redéfinir les relationships...
|
2023-06-15 17:14:37 +02:00
|
|
|
return sorted(self.ue_coefs, key=lambda uc: (uc.ue.numero, uc.ue.acronyme))
|
2021-12-10 01:55:13 +01:00
|
|
|
|
2022-12-09 04:24:32 +01:00
|
|
|
def ue_coefs_list(
|
|
|
|
self, include_zeros=True, ues: list["UniteEns"] = None
|
|
|
|
) -> list[tuple["UniteEns", float]]:
|
2022-01-06 22:42:26 +01:00
|
|
|
"""Liste des coefs vers les UE (pour les modules APC).
|
2022-12-09 04:24:32 +01:00
|
|
|
Si ues est spécifié, restreint aux UE indiquées.
|
|
|
|
Sinon si include_zeros, liste aussi les UE sans coef (donc nul) de ce semestre,
|
2022-01-25 10:45:13 +01:00
|
|
|
sauf UE bonus sport.
|
2022-01-06 22:42:26 +01:00
|
|
|
Result: List of tuples [ (ue, coef) ]
|
|
|
|
"""
|
|
|
|
if not self.is_apc():
|
|
|
|
return []
|
2022-12-09 04:24:32 +01:00
|
|
|
if include_zeros and ues is None:
|
2022-01-06 22:42:26 +01:00
|
|
|
# Toutes les UE du même semestre:
|
2022-12-09 04:24:32 +01:00
|
|
|
ues = (
|
2022-01-06 22:42:26 +01:00
|
|
|
self.formation.ues.filter_by(semestre_idx=self.ue.semestre_idx)
|
2022-01-25 10:45:13 +01:00
|
|
|
.filter(UniteEns.type != UE_SPORT)
|
2022-01-06 22:42:26 +01:00
|
|
|
.order_by(UniteEns.numero)
|
|
|
|
.all()
|
|
|
|
)
|
2022-12-09 04:24:32 +01:00
|
|
|
if not ues:
|
|
|
|
return []
|
|
|
|
if ues:
|
2022-01-06 22:42:26 +01:00
|
|
|
coefs_dict = self.get_ue_coef_dict()
|
|
|
|
coefs_list = []
|
2022-12-09 04:24:32 +01:00
|
|
|
for ue in ues:
|
2022-01-06 22:42:26 +01:00
|
|
|
coefs_list.append((ue, coefs_dict.get(ue.id, 0.0)))
|
|
|
|
return coefs_list
|
|
|
|
# Liste seulement les coefs définis:
|
|
|
|
return [(c.ue, c.coef) for c in self.get_ue_coefs_sorted()]
|
2021-12-08 22:33:32 +01:00
|
|
|
|
2022-07-02 00:00:29 +02:00
|
|
|
def get_codes_apogee(self) -> set[str]:
|
|
|
|
"""Les codes Apogée (codés en base comme "VRT1,VRT2")"""
|
|
|
|
if self.code_apogee:
|
2022-07-02 11:17:04 +02:00
|
|
|
return {x.strip() for x in self.code_apogee.split(",") if x}
|
2022-07-02 00:00:29 +02:00
|
|
|
return set()
|
|
|
|
|
2022-10-30 22:42:10 +01:00
|
|
|
def get_parcours(self) -> list[ApcParcours]:
|
|
|
|
"""Les parcours utilisant ce module.
|
|
|
|
Si tous les parcours, liste vide (!).
|
|
|
|
"""
|
|
|
|
ref_comp = self.formation.referentiel_competence
|
|
|
|
if not ref_comp:
|
|
|
|
return []
|
|
|
|
tous_parcours_ids = {p.id for p in ref_comp.parcours}
|
|
|
|
parcours_ids = {p.id for p in self.parcours}
|
|
|
|
if tous_parcours_ids == parcours_ids:
|
|
|
|
return []
|
|
|
|
return self.parcours
|
|
|
|
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
class ModuleUECoef(db.Model):
|
|
|
|
"""Coefficients des modules vers les UE (APC, BUT)
|
|
|
|
En mode APC, ces coefs remplacent le coefficient "PPN" du module.
|
|
|
|
"""
|
|
|
|
|
|
|
|
__tablename__ = "module_ue_coef"
|
|
|
|
|
|
|
|
module_id = db.Column(
|
|
|
|
db.Integer,
|
|
|
|
db.ForeignKey("notes_modules.id", ondelete="CASCADE"),
|
|
|
|
primary_key=True,
|
|
|
|
)
|
|
|
|
ue_id = db.Column(
|
|
|
|
db.Integer,
|
|
|
|
db.ForeignKey("notes_ue.id", ondelete="CASCADE"),
|
|
|
|
primary_key=True,
|
|
|
|
)
|
|
|
|
coef = db.Column(
|
|
|
|
db.Float,
|
|
|
|
nullable=False,
|
|
|
|
)
|
|
|
|
module = db.relationship(
|
|
|
|
Module,
|
|
|
|
backref=db.backref(
|
|
|
|
"ue_coefs",
|
|
|
|
passive_deletes=True,
|
|
|
|
cascade="save-update, merge, delete, delete-orphan",
|
|
|
|
),
|
|
|
|
)
|
|
|
|
ue = db.relationship(
|
|
|
|
"UniteEns",
|
|
|
|
backref=db.backref(
|
|
|
|
"module_ue_coefs",
|
|
|
|
passive_deletes=True,
|
|
|
|
cascade="save-update, merge, delete, delete-orphan",
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
2022-07-21 14:21:06 +02:00
|
|
|
def to_dict(self, convert_objects=False) -> dict:
|
|
|
|
"""If convert_objects, convert all attributes to native types
|
2022-07-21 14:44:19 +02:00
|
|
|
(suitable for json encoding).
|
2022-07-21 14:21:06 +02:00
|
|
|
"""
|
|
|
|
d = dict(self.__dict__)
|
|
|
|
d.pop("_sa_instance_state", None)
|
2022-08-02 17:13:13 +02:00
|
|
|
if convert_objects:
|
|
|
|
d["ue"] = self.ue.to_dict(with_module_ue_coefs=False, convert_objects=True)
|
2022-07-21 14:21:06 +02:00
|
|
|
return d
|
|
|
|
|
2021-12-08 22:33:32 +01:00
|
|
|
|
|
|
|
class NotesTag(db.Model):
|
|
|
|
"""Tag sur un module"""
|
|
|
|
|
|
|
|
__tablename__ = "notes_tags"
|
|
|
|
__table_args__ = (db.UniqueConstraint("title", "dept_id"),)
|
|
|
|
|
|
|
|
id = db.Column(db.Integer, primary_key=True)
|
|
|
|
tag_id = db.synonym("id")
|
|
|
|
|
|
|
|
dept_id = db.Column(db.Integer, db.ForeignKey("departement.id"), index=True)
|
|
|
|
title = db.Column(db.Text(), nullable=False)
|
|
|
|
|
|
|
|
|
|
|
|
# Association tag <-> module
|
|
|
|
notes_modules_tags = db.Table(
|
|
|
|
"notes_modules_tags",
|
|
|
|
db.Column(
|
|
|
|
"tag_id",
|
|
|
|
db.Integer,
|
|
|
|
db.ForeignKey("notes_tags.id", ondelete="CASCADE"),
|
|
|
|
),
|
|
|
|
db.Column(
|
|
|
|
"module_id", db.Integer, db.ForeignKey("notes_modules.id", ondelete="CASCADE")
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
from app.models.ues import UniteEns
|