2020-09-26 16:19:37 +02:00
|
|
|
# -*- mode: python -*-
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
##############################################################################
|
|
|
|
#
|
|
|
|
# Gestion scolarite IUT
|
|
|
|
#
|
2021-01-01 17:51:08 +01:00
|
|
|
# Copyright (c) 1999 - 2021 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
|
|
|
|
#
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
"""ScoDoc : annulation des saisies de notes
|
|
|
|
|
|
|
|
|
|
|
|
note = {evaluation_id, etudid, value, date, uid, comment}
|
|
|
|
|
|
|
|
Pour une évaluation:
|
|
|
|
- notes actuelles: table notes_notes
|
|
|
|
- historique: table notes_notes_log
|
|
|
|
|
|
|
|
saisie de notes == saisir ou supprimer une ou plusieurs notes (mêmes date et uid)
|
2021-02-04 20:02:44 +01:00
|
|
|
! tolérance sur les dates (200ms ?)
|
2020-09-26 16:19:37 +02:00
|
|
|
Chaque saisie affecte ou remplace une ou plusieurs notes.
|
|
|
|
|
|
|
|
Opérations:
|
|
|
|
- lister les saisies de notes
|
|
|
|
- annuler une saisie complète
|
|
|
|
- lister les modifs d'une seule note
|
|
|
|
- annuler une modif d'une note
|
|
|
|
"""
|
|
|
|
|
|
|
|
import datetime
|
2021-09-18 10:10:02 +02:00
|
|
|
from flask import request
|
|
|
|
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc.intervals import intervalmap
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-06-19 23:21:37 +02:00
|
|
|
import app.scodoc.sco_utils as scu
|
|
|
|
import app.scodoc.notesdb as ndb
|
|
|
|
from app.scodoc import sco_evaluations
|
|
|
|
from app.scodoc import sco_formsemestre
|
|
|
|
from app.scodoc import sco_moduleimpl
|
|
|
|
from app.scodoc import sco_preferences
|
2021-07-03 23:35:32 +02:00
|
|
|
from app.scodoc import sco_users
|
2021-08-21 17:07:44 +02:00
|
|
|
import sco_version
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc.gen_tables import GenTable
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
# deux notes (de même uid) sont considérées comme de la même opération si
|
|
|
|
# elles sont séparées de moins de 2*tolerance:
|
|
|
|
OPERATION_DATE_TOLERANCE = datetime.timedelta(seconds=0.1)
|
|
|
|
|
|
|
|
|
|
|
|
class NotesOperation(dict):
|
|
|
|
"""Represents an operation on an evaluation
|
|
|
|
Keys: evaluation_id, date, uid, notes
|
|
|
|
"""
|
|
|
|
|
|
|
|
def get_comment(self):
|
|
|
|
if self["notes"]:
|
|
|
|
return self["notes"][0]["comment"]
|
|
|
|
else:
|
|
|
|
return ""
|
|
|
|
|
|
|
|
def comp_values(self):
|
|
|
|
"compute keys: comment, nb_notes"
|
|
|
|
self["comment"] = self.get_comment()
|
|
|
|
self["nb_notes"] = len(self["notes"])
|
|
|
|
self["datestr"] = self["date"].strftime("%a %d/%m/%y %Hh%M")
|
|
|
|
|
2021-08-21 00:24:51 +02:00
|
|
|
def undo(self):
|
2020-09-26 16:19:37 +02:00
|
|
|
"undo operation"
|
|
|
|
pass
|
|
|
|
# replace notes by last found in notes_log
|
|
|
|
# and suppress log entry
|
|
|
|
# select * from notes_notes_log where evaluation_id= and etudid= and date <
|
|
|
|
#
|
|
|
|
# verrouille tables notes, notes_log
|
|
|
|
# pour chaque note qui n'est pas plus recente que l'operation:
|
|
|
|
# recupere valeurs precedentes dans log
|
|
|
|
# affecte valeurs notes
|
|
|
|
# suppr log
|
|
|
|
# deverrouille tablesj
|
|
|
|
# for note in self['notes']:
|
|
|
|
# # il y a-t-il une modif plus recente ?
|
|
|
|
# if self['current_notes_by_etud']['date'] <= self['date'] + OPERATION_DATE_TOLERANCE:
|
|
|
|
#
|
2021-07-19 20:53:01 +03:00
|
|
|
# + invalider cache sco_cache.EvaluationCache.delete(evaluation_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
|
2021-08-21 00:24:51 +02:00
|
|
|
def list_operations(evaluation_id):
|
2020-09-26 16:19:37 +02:00
|
|
|
"""returns list of NotesOperation for this evaluation"""
|
2021-07-19 20:53:01 +03:00
|
|
|
notes = list(
|
|
|
|
sco_evaluations.do_evaluation_get_all_notes(
|
2021-07-29 11:19:00 +03:00
|
|
|
evaluation_id, filter_suppressed=False
|
2021-07-19 20:53:01 +03:00
|
|
|
).values()
|
|
|
|
)
|
|
|
|
notes_log = list(
|
|
|
|
sco_evaluations.do_evaluation_get_all_notes(
|
2021-07-29 11:19:00 +03:00
|
|
|
evaluation_id, filter_suppressed=False, table="notes_notes_log"
|
2021-07-19 20:53:01 +03:00
|
|
|
).values()
|
|
|
|
)
|
2020-09-26 16:19:37 +02:00
|
|
|
dt = OPERATION_DATE_TOLERANCE
|
|
|
|
NotesDates = {} # { uid : intervalmap }
|
|
|
|
|
|
|
|
for note in notes + notes_log:
|
2021-07-09 17:47:06 +02:00
|
|
|
if note["uid"] not in NotesDates:
|
2020-09-26 16:19:37 +02:00
|
|
|
NotesDates[note["uid"]] = intervalmap()
|
|
|
|
nd = NotesDates[note["uid"]]
|
|
|
|
if nd[note["date"]] is None:
|
|
|
|
nd[note["date"] - dt : note["date"] + dt] = [note]
|
|
|
|
else:
|
|
|
|
nd[note["date"]].append(note)
|
|
|
|
|
|
|
|
current_notes_by_etud = {} # { etudid : note }
|
|
|
|
for note in notes:
|
|
|
|
current_notes_by_etud[note["etudid"]] = note
|
|
|
|
|
|
|
|
Ops = []
|
|
|
|
for uid in NotesDates.keys():
|
2021-02-04 20:02:44 +01:00
|
|
|
for (t0, _), notes in NotesDates[uid].items():
|
2020-09-26 16:19:37 +02:00
|
|
|
Op = NotesOperation(
|
|
|
|
evaluation_id=evaluation_id,
|
|
|
|
date=t0,
|
|
|
|
uid=uid,
|
|
|
|
notes=NotesDates[uid][t0],
|
|
|
|
current_notes_by_etud=current_notes_by_etud,
|
|
|
|
)
|
|
|
|
Op.comp_values()
|
|
|
|
Ops.append(Op)
|
|
|
|
|
|
|
|
return Ops
|
|
|
|
|
|
|
|
|
2021-09-27 10:20:10 +02:00
|
|
|
def evaluation_list_operations(evaluation_id):
|
2020-09-26 16:19:37 +02:00
|
|
|
"""Page listing operations on evaluation"""
|
2021-07-29 11:19:00 +03:00
|
|
|
E = sco_evaluations.do_evaluation_list({"evaluation_id": evaluation_id})[0]
|
2021-10-15 14:00:51 +02:00
|
|
|
M = sco_moduleimpl.moduleimpl_list(moduleimpl_id=E["moduleimpl_id"])[0]
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-08-21 00:24:51 +02:00
|
|
|
Ops = list_operations(evaluation_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
columns_ids = ("datestr", "uid", "nb_notes", "comment")
|
|
|
|
titles = {
|
|
|
|
"datestr": "Date",
|
|
|
|
"uid": "Enseignant",
|
|
|
|
"nb_notes": "Nb de notes",
|
|
|
|
"comment": "Commentaire",
|
|
|
|
}
|
|
|
|
tab = GenTable(
|
|
|
|
titles=titles,
|
|
|
|
columns_ids=columns_ids,
|
|
|
|
rows=Ops,
|
|
|
|
html_sortable=False,
|
|
|
|
html_title="<h2>Opérations sur l'évaluation %s du %s</h2>"
|
|
|
|
% (E["description"], E["jour"]),
|
2021-07-28 18:03:54 +03:00
|
|
|
preferences=sco_preferences.SemPreferences(M["formsemestre_id"]),
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2021-09-16 00:15:10 +02:00
|
|
|
return tab.make_page()
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
|
2021-09-27 10:20:10 +02:00
|
|
|
def formsemestre_list_saisies_notes(formsemestre_id, format="html"):
|
2021-08-08 17:38:46 +02:00
|
|
|
"""Table listant toutes les opérations de saisies de notes, dans toutes
|
|
|
|
les évaluations du semestre.
|
|
|
|
"""
|
2021-08-19 10:28:35 +02:00
|
|
|
sem = sco_formsemestre.get_formsemestre(formsemestre_id)
|
2021-02-03 22:00:41 +01:00
|
|
|
r = ndb.SimpleDictFetch(
|
2021-11-12 11:30:59 +01:00
|
|
|
"""SELECT i.nom, i.id as etudid, n.*, mod.titre, e.description, e.jour
|
2021-08-16 22:48:22 +02:00
|
|
|
FROM notes_notes n, notes_evaluation e, notes_moduleimpl mi,
|
2021-08-08 17:38:46 +02:00
|
|
|
notes_modules mod, identite i
|
|
|
|
WHERE mi.id = e.moduleimpl_id
|
2021-08-16 22:48:22 +02:00
|
|
|
and mi.module_id = mod.id
|
|
|
|
and e.id = n.evaluation_id
|
|
|
|
and i.id = n.etudid
|
|
|
|
and mi.formsemestre_id = %(formsemestre_id)s
|
2021-08-08 17:38:46 +02:00
|
|
|
ORDER BY date desc
|
|
|
|
""",
|
2020-09-26 16:19:37 +02:00
|
|
|
{"formsemestre_id": formsemestre_id},
|
|
|
|
)
|
|
|
|
columns_ids = (
|
|
|
|
"date",
|
2021-11-12 11:30:59 +01:00
|
|
|
"code_nip",
|
2020-09-26 16:19:37 +02:00
|
|
|
"nom",
|
|
|
|
"value",
|
|
|
|
"uid",
|
|
|
|
"titre",
|
|
|
|
"description",
|
|
|
|
"jour",
|
|
|
|
"comment",
|
|
|
|
)
|
|
|
|
titles = {
|
2021-11-12 11:30:59 +01:00
|
|
|
"code_nip": "NIP",
|
2020-09-26 16:19:37 +02:00
|
|
|
"nom": "Etudiant",
|
|
|
|
"date": "Date",
|
|
|
|
"value": "Note",
|
|
|
|
"comment": "Remarque",
|
|
|
|
"uid": "Enseignant",
|
|
|
|
"titre": "Module",
|
|
|
|
"description": "Evaluation",
|
|
|
|
"jour": "Date éval.",
|
|
|
|
}
|
|
|
|
tab = GenTable(
|
|
|
|
titles=titles,
|
|
|
|
columns_ids=columns_ids,
|
|
|
|
rows=r,
|
|
|
|
html_title="<h2>Saisies de notes dans %s</h2>" % sem["titreannee"],
|
|
|
|
html_class="table_leftalign table_coldate",
|
|
|
|
html_sortable=True,
|
|
|
|
caption="Saisies de notes dans %s" % sem["titreannee"],
|
2021-07-28 18:03:54 +03:00
|
|
|
preferences=sco_preferences.SemPreferences(formsemestre_id),
|
2021-09-18 10:10:02 +02:00
|
|
|
base_url="%s?formsemestre_id=%s" % (request.base_url, formsemestre_id),
|
2021-08-21 17:07:44 +02:00
|
|
|
origin="Généré par %s le " % sco_version.SCONAME
|
|
|
|
+ scu.timedate_human_repr()
|
|
|
|
+ "",
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
2021-09-16 00:15:10 +02:00
|
|
|
return tab.make_page(format=format)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
|
2021-09-27 10:20:10 +02:00
|
|
|
def get_note_history(evaluation_id, etudid, fmt=""):
|
2020-09-26 16:19:37 +02:00
|
|
|
"""Historique d'une note
|
|
|
|
= liste chronologique d'opérations, la plus récente d'abord
|
|
|
|
[ { 'value', 'date', 'comment', 'uid' } ]
|
|
|
|
"""
|
2021-06-15 13:59:56 +02:00
|
|
|
cnx = ndb.GetDBConnexion()
|
2021-02-03 22:00:41 +01:00
|
|
|
cursor = cnx.cursor(cursor_factory=ndb.ScoDocCursor)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
# Valeur courante
|
|
|
|
cursor.execute(
|
|
|
|
"""
|
|
|
|
SELECT * FROM notes_notes
|
|
|
|
WHERE evaluation_id=%(evaluation_id)s AND etudid=%(etudid)s
|
|
|
|
""",
|
|
|
|
{"evaluation_id": evaluation_id, "etudid": etudid},
|
|
|
|
)
|
|
|
|
history = cursor.dictfetchall()
|
|
|
|
|
|
|
|
# Historique
|
|
|
|
cursor.execute(
|
|
|
|
"""
|
|
|
|
SELECT * FROM notes_notes_log
|
|
|
|
WHERE evaluation_id=%(evaluation_id)s AND etudid=%(etudid)s
|
|
|
|
ORDER BY date DESC""",
|
|
|
|
{"evaluation_id": evaluation_id, "etudid": etudid},
|
|
|
|
)
|
|
|
|
|
|
|
|
history += cursor.dictfetchall()
|
|
|
|
|
|
|
|
# Replace None comments by ''
|
|
|
|
# et cherche nom complet de l'enseignant:
|
|
|
|
for x in history:
|
|
|
|
x["comment"] = x["comment"] or ""
|
2021-07-03 23:35:32 +02:00
|
|
|
x["user_name"] = sco_users.user_info(x["uid"])["nomcomplet"]
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
if fmt == "json":
|
2021-09-21 13:36:56 +02:00
|
|
|
return scu.sendJSON(history)
|
2020-09-26 16:19:37 +02:00
|
|
|
else:
|
|
|
|
return history
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
from debug import *
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc.sco_undo_notes import *
|
2021-08-21 00:24:51 +02:00
|
|
|
_ = go_dept(app, 'RT').Notes
|
|
|
|
get_note_history( 'EVAL29740', 'EID28403')
|
2020-09-26 16:19:37 +02:00
|
|
|
"""
|