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@gmail.com
#
##############################################################################
""" Evaluations
"""
2021-06-21 11:22:55 +02:00
import datetime
import operator
import pprint
2020-12-23 23:49:11 +01:00
import time
2021-07-09 17:47:06 +02:00
import six . moves . urllib . request , six . moves . urllib . parse , six . moves . urllib . error
2020-09-26 16:19:37 +02:00
2021-07-29 11:19:00 +03:00
import flask
from flask import url_for
from flask import g
from flask_login import current_user
2021-06-19 23:21:37 +02:00
from app . scodoc . notes_log import log , logCallStack
import app . scodoc . sco_utils as scu
import app . scodoc . notesdb as ndb
from app . scodoc . sco_exceptions import AccessDenied , ScoValueError
from app . scodoc import VERSION
from app . scodoc . gen_tables import GenTable
from app . scodoc . TrivialFormulator import TrivialFormulator
from app . scodoc import html_sco_header
from app . scodoc import sco_abs
2021-07-19 20:53:01 +03:00
from app . scodoc import sco_cache
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_edit_module
from app . scodoc import sco_edit_ue
from app . scodoc import sco_formsemestre
2021-06-21 10:17:16 +02:00
from app . scodoc import sco_formsemestre_inscriptions
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_groups
from app . scodoc import sco_moduleimpl
from app . scodoc import sco_news
from app . scodoc import sco_permissions_check
from app . scodoc import sco_preferences
2021-07-03 23:35:32 +02:00
from app . scodoc import sco_users
2021-06-19 23:21:37 +02:00
2020-09-26 16:19:37 +02:00
# --------------------------------------------------------------------
#
# MISC AUXILIARY FUNCTIONS
#
# --------------------------------------------------------------------
def notes_moyenne_median_mini_maxi ( notes ) :
" calcule moyenne et mediane d ' une liste de valeurs (floats) "
notes = [
x
for x in notes
2020-12-23 23:49:11 +01:00
if ( x != None ) and ( x != scu . NOTES_NEUTRALISE ) and ( x != scu . NOTES_ATTENTE )
2020-09-26 16:19:37 +02:00
]
n = len ( notes )
if not n :
return None , None , None , None
moy = sum ( notes ) / n
median = ListMedian ( notes )
mini = min ( notes )
maxi = max ( notes )
return moy , median , mini , maxi
def ListMedian ( L ) :
""" Median of a list L """
n = len ( L )
if not n :
raise ValueError ( " empty list " )
L . sort ( )
if n % 2 :
2021-07-09 19:50:40 +02:00
return L [ n / / 2 ]
2020-09-26 16:19:37 +02:00
else :
2021-07-09 19:50:40 +02:00
return ( L [ n / / 2 ] + L [ n / / 2 - 1 ] ) / 2
2020-09-26 16:19:37 +02:00
# --------------------------------------------------------------------
2021-06-17 00:08:37 +02:00
_evaluationEditor = ndb . EditableTable (
" notes_evaluation " ,
" evaluation_id " ,
(
" evaluation_id " ,
" moduleimpl_id " ,
" jour " ,
" heure_debut " ,
" heure_fin " ,
" description " ,
" note_max " ,
" coefficient " ,
" visibulletin " ,
" publish_incomplete " ,
" evaluation_type " ,
" numero " ,
) ,
sortkey = " numero desc, jour desc, heure_debut desc " , # plus recente d'abord
output_formators = {
" jour " : ndb . DateISOtoDMY ,
" visibulletin " : str ,
" publish_incomplete " : str ,
" numero " : ndb . int_null_is_zero ,
} ,
input_formators = {
" jour " : ndb . DateDMYtoISO ,
" heure_debut " : ndb . TimetoISO8601 , # converti par do_evaluation_list
" heure_fin " : ndb . TimetoISO8601 , # converti par do_evaluation_list
" visibulletin " : int ,
" publish_incomplete " : int ,
} ,
)
2021-07-29 11:19:00 +03:00
def do_evaluation_list ( args , sortkey = None ) :
2021-06-17 00:08:37 +02:00
""" List evaluations, sorted by numero (or most recent date first).
Ajoute les champs :
' duree ' : ' 2h30 '
' matin ' : 1 ( commence avant 12 : 00 ) ou 0
' apresmidi ' : 1 ( termine après 12 : 00 ) ou 0
' descrheure ' : ' de 15h00 à 16h30 '
"""
cnx = ndb . GetDBConnexion ( )
evals = _evaluationEditor . list ( cnx , args , sortkey = sortkey )
# calcule duree (chaine de car.) de chaque evaluation et ajoute jouriso, matin, apresmidi
for e in evals :
heure_debut_dt = e [ " heure_debut " ] or datetime . time (
8 , 00
) # au cas ou pas d'heure (note externe?)
heure_fin_dt = e [ " heure_fin " ] or datetime . time ( 8 , 00 )
e [ " heure_debut " ] = ndb . TimefromISO8601 ( e [ " heure_debut " ] )
e [ " heure_fin " ] = ndb . TimefromISO8601 ( e [ " heure_fin " ] )
e [ " jouriso " ] = ndb . DateDMYtoISO ( e [ " jour " ] )
heure_debut , heure_fin = e [ " heure_debut " ] , e [ " heure_fin " ]
d = ndb . TimeDuration ( heure_debut , heure_fin )
if d is not None :
m = d % 60
e [ " duree " ] = " %d h " % ( d / 60 )
if m != 0 :
e [ " duree " ] + = " %02d " % m
else :
e [ " duree " ] = " "
if heure_debut and ( not heure_fin or heure_fin == heure_debut ) :
e [ " descrheure " ] = " à " + heure_debut
elif heure_debut and heure_fin :
e [ " descrheure " ] = " de %s à %s " % ( heure_debut , heure_fin )
else :
e [ " descrheure " ] = " "
# matin, apresmidi: utile pour se referer aux absences:
if heure_debut_dt < datetime . time ( 12 , 00 ) :
e [ " matin " ] = 1
else :
e [ " matin " ] = 0
if heure_fin_dt > datetime . time ( 12 , 00 ) :
e [ " apresmidi " ] = 1
else :
e [ " apresmidi " ] = 0
return evals
2021-07-29 11:19:00 +03:00
def do_evaluation_list_in_formsemestre ( formsemestre_id ) :
2021-06-17 00:08:37 +02:00
" list evaluations in this formsemestre "
2021-07-29 11:19:00 +03:00
context = None # #context
2021-06-17 00:08:37 +02:00
mods = sco_moduleimpl . do_moduleimpl_list ( context , formsemestre_id = formsemestre_id )
evals = [ ]
for mod in mods :
2021-07-29 11:19:00 +03:00
evals + = do_evaluation_list ( args = { " moduleimpl_id " : mod [ " moduleimpl_id " ] } )
2021-06-17 00:08:37 +02:00
return evals
2021-07-29 11:19:00 +03:00
def _check_evaluation_args ( args ) :
2021-06-21 10:17:16 +02:00
" Check coefficient, dates and duration, raises exception if invalid "
2021-07-29 11:19:00 +03:00
context = None # #context
2021-06-21 10:17:16 +02:00
moduleimpl_id = args [ " moduleimpl_id " ]
# check bareme
note_max = args . get ( " note_max " , None )
if note_max is None :
raise ScoValueError ( " missing note_max " )
try :
note_max = float ( note_max )
except ValueError :
raise ScoValueError ( " Invalid note_max value " )
if note_max < 0 :
raise ScoValueError ( " Invalid note_max value (must be positive or null) " )
# check coefficient
coef = args . get ( " coefficient " , None )
if coef is None :
raise ScoValueError ( " missing coefficient " )
try :
coef = float ( coef )
except ValueError :
raise ScoValueError ( " Invalid coefficient value " )
if coef < 0 :
raise ScoValueError ( " Invalid coefficient value (must be positive or null) " )
# check date
jour = args . get ( " jour " , None )
args [ " jour " ] = jour
if jour :
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = moduleimpl_id ) [ 0 ]
sem = sco_formsemestre . get_formsemestre ( context , M [ " formsemestre_id " ] )
d , m , y = [ int ( x ) for x in sem [ " date_debut " ] . split ( " / " ) ]
date_debut = datetime . date ( y , m , d )
d , m , y = [ int ( x ) for x in sem [ " date_fin " ] . split ( " / " ) ]
date_fin = datetime . date ( y , m , d )
# passe par ndb.DateDMYtoISO pour avoir date pivot
y , m , d = [ int ( x ) for x in ndb . DateDMYtoISO ( jour ) . split ( " - " ) ]
jour = datetime . date ( y , m , d )
if ( jour > date_fin ) or ( jour < date_debut ) :
raise ScoValueError (
" La date de l ' évaluation ( %s / %s / %s ) n ' est pas dans le semestre ! "
% ( d , m , y )
)
heure_debut = args . get ( " heure_debut " , None )
args [ " heure_debut " ] = heure_debut
heure_fin = args . get ( " heure_fin " , None )
args [ " heure_fin " ] = heure_fin
if jour and ( ( not heure_debut ) or ( not heure_fin ) ) :
raise ScoValueError ( " Les heures doivent être précisées " )
d = ndb . TimeDuration ( heure_debut , heure_fin )
if d and ( ( d < 0 ) or ( d > 60 * 12 ) ) :
raise ScoValueError ( " Heures de l ' évaluation incohérentes ! " )
def do_evaluation_create (
moduleimpl_id = None ,
jour = None ,
heure_debut = None ,
heure_fin = None ,
description = None ,
note_max = None ,
coefficient = None ,
visibulletin = None ,
publish_incomplete = None ,
evaluation_type = None ,
numero = None ,
REQUEST = None ,
2021-07-29 17:31:15 +03:00
* * kw , # ceci pour absorber les arguments excedentaires de tf #sco8
2021-06-21 10:17:16 +02:00
) :
""" Create an evaluation """
2021-07-29 11:19:00 +03:00
context = None # #context
if not sco_permissions_check . can_edit_evaluation ( moduleimpl_id = moduleimpl_id ) :
2021-06-21 10:17:16 +02:00
raise AccessDenied (
2021-07-29 11:19:00 +03:00
" Modification évaluation impossible pour %s " % current_user . get_nomplogin ( )
2021-06-21 10:17:16 +02:00
)
args = locals ( )
log ( " do_evaluation_create: args= " + str ( args ) )
2021-07-29 11:19:00 +03:00
_check_evaluation_args ( args )
2021-06-21 10:17:16 +02:00
# Check numeros
2021-07-29 11:19:00 +03:00
module_evaluation_renumber ( moduleimpl_id , only_if_unumbered = True )
2021-06-21 10:17:16 +02:00
if not " numero " in args or args [ " numero " ] is None :
n = None
# determine le numero avec la date
# Liste des eval existantes triees par date, la plus ancienne en tete
ModEvals = do_evaluation_list (
args = { " moduleimpl_id " : moduleimpl_id } ,
sortkey = " jour asc, heure_debut asc " ,
)
if args [ " jour " ] :
next_eval = None
t = (
ndb . DateDMYtoISO ( args [ " jour " ] ) ,
ndb . TimetoISO8601 ( args [ " heure_debut " ] ) ,
)
for e in ModEvals :
if (
ndb . DateDMYtoISO ( e [ " jour " ] ) ,
ndb . TimetoISO8601 ( e [ " heure_debut " ] ) ,
) > t :
next_eval = e
break
if next_eval :
2021-07-29 11:19:00 +03:00
n = module_evaluation_insert_before ( ModEvals , next_eval )
2021-06-21 10:17:16 +02:00
else :
n = None # a placer en fin
if n is None : # pas de date ou en fin:
if ModEvals :
log ( pprint . pformat ( ModEvals [ - 1 ] ) )
n = ModEvals [ - 1 ] [ " numero " ] + 1
else :
n = 0 # the only one
# log("creating with numero n=%d" % n)
args [ " numero " ] = n
#
cnx = ndb . GetDBConnexion ( )
r = _evaluationEditor . create ( cnx , args )
# news
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = moduleimpl_id ) [ 0 ]
mod = sco_edit_module . do_module_list ( context , args = { " module_id " : M [ " module_id " ] } ) [ 0 ]
mod [ " moduleimpl_id " ] = M [ " moduleimpl_id " ]
mod [ " url " ] = " Notes/moduleimpl_status?moduleimpl_id= %(moduleimpl_id)s " % mod
sco_news . add (
typ = sco_news . NEWS_NOTE ,
object = moduleimpl_id ,
text = ' Création d \' une évaluation dans <a href= " %(url)s " > %(titre)s </a> ' % mod ,
url = mod [ " url " ] ,
)
return r
2021-07-29 11:19:00 +03:00
def do_evaluation_edit ( args ) :
2021-06-17 00:08:37 +02:00
" edit an evaluation "
2021-07-29 11:19:00 +03:00
context = None # #context
2021-06-17 00:08:37 +02:00
evaluation_id = args [ " evaluation_id " ]
2021-07-29 11:19:00 +03:00
the_evals = do_evaluation_list ( { " evaluation_id " : evaluation_id } )
2021-06-17 00:08:37 +02:00
if not the_evals :
raise ValueError ( " evaluation inexistante ! " )
moduleimpl_id = the_evals [ 0 ] [ " moduleimpl_id " ]
2021-07-29 11:19:00 +03:00
if not sco_permissions_check . can_edit_evaluation ( moduleimpl_id = moduleimpl_id ) :
2021-06-21 10:17:16 +02:00
raise AccessDenied (
2021-07-29 11:19:00 +03:00
" Modification évaluation impossible pour %s " % current_user . get_nomplogin ( )
2021-06-21 10:17:16 +02:00
)
2021-06-17 00:08:37 +02:00
args [ " moduleimpl_id " ] = moduleimpl_id
2021-07-29 11:19:00 +03:00
_check_evaluation_args ( args )
2021-06-21 10:17:16 +02:00
2021-06-17 00:08:37 +02:00
cnx = ndb . GetDBConnexion ( )
2021-06-21 11:22:55 +02:00
_evaluationEditor . edit ( cnx , args )
2021-06-17 00:08:37 +02:00
# inval cache pour ce semestre
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = moduleimpl_id ) [ 0 ]
2021-07-19 20:53:01 +03:00
sco_cache . invalidate_formsemestre ( formsemestre_id = M [ " formsemestre_id " ] )
2020-09-26 16:19:37 +02:00
2021-07-29 11:19:00 +03:00
def do_evaluation_delete ( evaluation_id ) :
2020-09-26 16:19:37 +02:00
" delete evaluation "
2021-07-29 11:19:00 +03:00
context = None # #context
the_evals = do_evaluation_list ( { " evaluation_id " : evaluation_id } )
2020-09-26 16:19:37 +02:00
if not the_evals :
raise ValueError ( " evaluation inexistante ! " )
2021-06-21 10:17:16 +02:00
moduleimpl_id = the_evals [ 0 ] [ " moduleimpl_id " ]
2021-07-29 11:19:00 +03:00
if not sco_permissions_check . can_edit_evaluation ( moduleimpl_id = moduleimpl_id ) :
2021-06-21 10:17:16 +02:00
raise AccessDenied (
2021-07-29 11:19:00 +03:00
" Modification évaluation impossible pour %s " % current_user . get_nomplogin ( )
2021-06-21 10:17:16 +02:00
)
2021-07-29 11:19:00 +03:00
NotesDB = do_evaluation_get_all_notes ( evaluation_id ) # { etudid : value }
2020-09-26 16:19:37 +02:00
notes = [ x [ " value " ] for x in NotesDB . values ( ) ]
if notes :
raise ScoValueError (
" Impossible de supprimer cette évaluation: il reste des notes "
)
2021-06-15 13:59:56 +02:00
cnx = ndb . GetDBConnexion ( )
2020-09-26 16:19:37 +02:00
2021-06-21 11:22:55 +02:00
_evaluationEditor . delete ( cnx , evaluation_id )
2020-09-26 16:19:37 +02:00
# inval cache pour ce semestre
2021-01-17 22:31:28 +01:00
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = moduleimpl_id ) [ 0 ]
2021-07-19 20:53:01 +03:00
sco_cache . invalidate_formsemestre ( formsemestre_id = M [ " formsemestre_id " ] )
2020-09-26 16:19:37 +02:00
# news
2021-06-16 18:18:32 +02:00
mod = sco_edit_module . do_module_list ( context , args = { " module_id " : M [ " module_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
mod [ " moduleimpl_id " ] = M [ " moduleimpl_id " ]
2021-02-24 09:18:53 +01:00
mod [ " url " ] = (
2021-06-15 12:34:33 +02:00
scu . NotesURL ( ) + " /moduleimpl_status?moduleimpl_id= %(moduleimpl_id)s " % mod
2021-02-24 09:18:53 +01:00
)
2020-09-26 16:19:37 +02:00
sco_news . add (
typ = sco_news . NEWS_NOTE ,
object = moduleimpl_id ,
text = ' Suppression d \' une évaluation dans <a href= " %(url)s " > %(titre)s </a> ' % mod ,
url = mod [ " url " ] ,
)
_DEE_TOT = 0
2021-07-29 11:19:00 +03:00
def do_evaluation_etat ( evaluation_id , partition_id = None , select_first_partition = False ) :
2020-09-26 16:19:37 +02:00
""" donne infos sur l ' etat du evaluation
2020-12-02 01:00:23 +01:00
{ nb_inscrits , nb_notes , nb_abs , nb_neutre , nb_att ,
2020-09-26 16:19:37 +02:00
moyenne , mediane , mini , maxi ,
date_last_modif , gr_complets , gr_incomplets , evalcomplete }
evalcomplete est vrai si l ' eval est complete (tous les inscrits
à ce module ont des notes )
evalattente est vrai s ' il ne manque que des notes en attente
"""
2021-07-29 11:19:00 +03:00
context = None # #context
2020-09-26 16:19:37 +02:00
nb_inscrits = len (
2021-07-29 11:19:00 +03:00
sco_groups . do_evaluation_listeetuds_groups ( evaluation_id , getallstudents = True )
2020-09-26 16:19:37 +02:00
)
2021-07-29 11:19:00 +03:00
NotesDB = do_evaluation_get_all_notes ( evaluation_id ) # { etudid : value }
2020-09-26 16:19:37 +02:00
notes = [ x [ " value " ] for x in NotesDB . values ( ) ]
nb_abs = len ( [ x for x in notes if x is None ] )
2020-12-23 23:49:11 +01:00
nb_neutre = len ( [ x for x in notes if x == scu . NOTES_NEUTRALISE ] )
nb_att = len ( [ x for x in notes if x == scu . NOTES_ATTENTE ] )
2020-09-26 16:19:37 +02:00
moy_num , median_num , mini_num , maxi_num = notes_moyenne_median_mini_maxi ( notes )
if moy_num is None :
median , moy = " " , " "
median_num , moy_num = None , None
mini , maxi = " " , " "
mini_num , maxi_num = None , None
else :
2020-12-23 23:49:11 +01:00
median = scu . fmt_note ( median_num )
moy = scu . fmt_note ( moy_num )
mini = scu . fmt_note ( mini_num )
maxi = scu . fmt_note ( maxi_num )
2020-09-26 16:19:37 +02:00
# cherche date derniere modif note
if len ( NotesDB ) :
t = [ x [ " date " ] for x in NotesDB . values ( ) ]
last_modif = max ( t )
else :
last_modif = None
# ---- Liste des groupes complets et incomplets
2021-07-29 11:19:00 +03:00
E = do_evaluation_list ( args = { " evaluation_id " : evaluation_id } ) [ 0 ]
2021-01-17 22:31:28 +01:00
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = E [ " moduleimpl_id " ] ) [ 0 ]
2021-06-16 18:18:32 +02:00
Mod = sco_edit_module . do_module_list ( context , args = { " module_id " : M [ " module_id " ] } ) [ 0 ]
2020-12-23 23:49:11 +01:00
is_malus = Mod [ " module_type " ] == scu . MODULE_MALUS # True si module de malus
2020-09-26 16:19:37 +02:00
formsemestre_id = M [ " formsemestre_id " ]
# Si partition_id is None, prend 'all' ou bien la premiere:
if partition_id is None :
if select_first_partition :
partitions = sco_groups . get_partitions_list ( context , formsemestre_id )
partition = partitions [ 0 ]
else :
partition = sco_groups . get_default_partition ( context , formsemestre_id )
partition_id = partition [ " partition_id " ]
# Il faut considerer les inscriptions au semestre
# (pour avoir l'etat et le groupe) et aussi les inscriptions
# au module (pour gerer les modules optionnels correctement)
2021-06-21 10:17:16 +02:00
insem = sco_formsemestre_inscriptions . do_formsemestre_inscription_listinscrits (
context , formsemestre_id
)
2021-01-17 22:31:28 +01:00
insmod = sco_moduleimpl . do_moduleimpl_inscription_list (
context , moduleimpl_id = E [ " moduleimpl_id " ]
)
2020-12-02 01:00:23 +01:00
insmodset = set ( [ x [ " etudid " ] for x in insmod ] )
2020-09-26 16:19:37 +02:00
# retire de insem ceux qui ne sont pas inscrits au module
ins = [ i for i in insem if i [ " etudid " ] in insmodset ]
# Nombre de notes valides d'étudiants inscrits au module
# (car il peut y avoir des notes d'étudiants désinscrits depuis l'évaluation)
nb_notes = len ( insmodset . intersection ( NotesDB ) )
nb_notes_total = len ( NotesDB )
# On considere une note "manquante" lorsqu'elle n'existe pas
# ou qu'elle est en attente (ATT)
2020-12-23 23:49:11 +01:00
GrNbMissing = scu . DictDefault ( ) # group_id : nb notes manquantes
GrNotes = scu . DictDefault ( defaultvalue = [ ] ) # group_id: liste notes valides
2020-09-26 16:19:37 +02:00
TotalNbMissing = 0
TotalNbAtt = 0
groups = { } # group_id : group
etud_groups = sco_groups . get_etud_groups_in_partition ( context , partition_id )
for i in ins :
group = etud_groups . get ( i [ " etudid " ] , None )
if group and not group [ " group_id " ] in groups :
groups [ group [ " group_id " ] ] = group
#
isMissing = False
2021-07-09 17:47:06 +02:00
if i [ " etudid " ] in NotesDB :
2020-09-26 16:19:37 +02:00
val = NotesDB [ i [ " etudid " ] ] [ " value " ]
2020-12-23 23:49:11 +01:00
if val == scu . NOTES_ATTENTE :
2020-09-26 16:19:37 +02:00
isMissing = True
TotalNbAtt + = 1
if group :
GrNotes [ group [ " group_id " ] ] . append ( val )
else :
if group :
2020-12-23 23:49:11 +01:00
_ = GrNotes [ group [ " group_id " ] ] # create group
2020-09-26 16:19:37 +02:00
isMissing = True
if isMissing :
TotalNbMissing + = 1
if group :
GrNbMissing [ group [ " group_id " ] ] + = 1
gr_incomplets = [ x for x in GrNbMissing . keys ( ) ]
gr_incomplets . sort ( )
if (
( TotalNbMissing > 0 )
2020-12-23 23:49:11 +01:00
and ( E [ " evaluation_type " ] != scu . EVALUATION_RATTRAPAGE )
2021-03-11 14:49:37 +01:00
and ( E [ " evaluation_type " ] != scu . EVALUATION_SESSION2 )
2020-09-26 16:19:37 +02:00
and not is_malus
) :
complete = False
else :
complete = True
if (
2021-03-02 23:53:43 +01:00
TotalNbMissing > 0
2020-09-26 16:19:37 +02:00
and ( TotalNbMissing == TotalNbAtt or E [ " publish_incomplete " ] != " 0 " )
and not is_malus
) :
evalattente = True
else :
evalattente = False
2021-03-02 23:53:43 +01:00
# mais ne met pas en attente les evals immediates sans aucune notes:
if E [ " publish_incomplete " ] != " 0 " and nb_notes == 0 :
evalattente = False
2020-09-26 16:19:37 +02:00
# Calcul moyenne dans chaque groupe de TD
gr_moyennes = [ ] # group : {moy,median, nb_notes}
for group_id in GrNotes . keys ( ) :
notes = GrNotes [ group_id ]
gr_moy , gr_median , gr_mini , gr_maxi = notes_moyenne_median_mini_maxi ( notes )
gr_moyennes . append (
{
" group_id " : group_id ,
" group_name " : groups [ group_id ] [ " group_name " ] ,
" gr_moy_num " : gr_moy ,
2020-12-23 23:49:11 +01:00
" gr_moy " : scu . fmt_note ( gr_moy ) ,
2020-09-26 16:19:37 +02:00
" gr_median_num " : gr_median ,
2020-12-23 23:49:11 +01:00
" gr_median " : scu . fmt_note ( gr_median ) ,
" gr_mini " : scu . fmt_note ( gr_mini ) ,
" gr_maxi " : scu . fmt_note ( gr_maxi ) ,
" gr_mini_num " : gr_mini ,
" gr_maxi_num " : gr_maxi ,
2020-09-26 16:19:37 +02:00
" gr_nb_notes " : len ( notes ) ,
2020-12-23 23:49:11 +01:00
" gr_nb_att " : len ( [ x for x in notes if x == scu . NOTES_ATTENTE ] ) ,
2020-09-26 16:19:37 +02:00
}
)
gr_moyennes . sort ( key = operator . itemgetter ( " group_name " ) )
# retourne mapping
return {
" evaluation_id " : evaluation_id ,
" nb_inscrits " : nb_inscrits ,
" nb_notes " : nb_notes , # nb notes etudiants inscrits
" nb_notes_total " : nb_notes_total , # nb de notes (incluant desinscrits)
" nb_abs " : nb_abs ,
" nb_neutre " : nb_neutre ,
" nb_att " : nb_att ,
" moy " : moy ,
" moy_num " : moy_num ,
" median " : median ,
" mini " : mini ,
" mini_num " : mini_num ,
" maxi " : maxi ,
" maxi_num " : maxi_num ,
" median_num " : median_num ,
" last_modif " : last_modif ,
" gr_incomplets " : gr_incomplets ,
" gr_moyennes " : gr_moyennes ,
" groups " : groups ,
" evalcomplete " : complete ,
" evalattente " : evalattente ,
" is_malus " : is_malus ,
}
2021-07-29 11:19:00 +03:00
def do_evaluation_list_in_sem ( formsemestre_id , with_etat = True ) :
2020-09-26 16:19:37 +02:00
""" Liste les evaluations de tous les modules de ce semestre.
2020-12-02 01:00:23 +01:00
Donne pour chaque eval son état ( voir do_evaluation_etat )
{ evaluation_id , nb_inscrits , nb_notes , nb_abs , nb_neutre , moy , median , last_modif . . . }
Exemple :
[ {
' coefficient ' : 1.0 ,
' description ' : ' QCM et cas pratiques ' ,
' etat ' : { ' evalattente ' : False ,
' evalcomplete ' : True ,
' evaluation_id ' : ' GEAEVAL82883 ' ,
' gr_incomplets ' : [ ] ,
' gr_moyennes ' : [ { ' gr_median ' : ' 12.00 ' ,
' gr_median_num ' : 12. ,
' gr_moy ' : ' 11.88 ' ,
' gr_moy_num ' : 11.88 ,
' gr_nb_att ' : 0 ,
' gr_nb_notes ' : 166 ,
' group_id ' : ' GEAG266762 ' ,
' group_name ' : None } ] ,
' groups ' : { ' GEAG266762 ' : { ' etudid ' : ' GEAEID80603 ' ,
' group_id ' : ' GEAG266762 ' ,
' group_name ' : None ,
' partition_id ' : ' GEAP266761 ' }
} ,
' last_modif ' : datetime . datetime ( 2015 , 12 , 3 , 15 , 15 , 16 ) ,
' median ' : ' 12.00 ' ,
' moy ' : ' 11.84 ' ,
' nb_abs ' : 2 ,
' nb_att ' : 0 ,
' nb_inscrits ' : 166 ,
' nb_neutre ' : 0 ,
' nb_notes ' : 168 ,
' nb_notes_total ' : 169
} ,
' evaluation_id ' : ' GEAEVAL82883 ' ,
' evaluation_type ' : 0 ,
' heure_debut ' : datetime . time ( 8 , 0 ) ,
' heure_fin ' : datetime . time ( 9 , 30 ) ,
' jour ' : datetime . date ( 2015 , 11 , 3 ) , / / vide = > 1 / 1 / 1
' moduleimpl_id ' : ' GEAMIP80490 ' ,
' note_max ' : 20.0 ,
' numero ' : 0 ,
' publish_incomplete ' : 0 ,
' visibulletin ' : 1 } ]
2020-09-26 16:19:37 +02:00
"""
2020-12-23 22:27:21 +01:00
req = " select E.* from notes_evaluation E, notes_moduleimpl MI where MI.formsemestre_id = %(formsemestre_id)s and MI.moduleimpl_id = E.moduleimpl_id order by moduleimpl_id, numero desc, jour desc, heure_debut desc "
2021-06-15 13:59:56 +02:00
cnx = ndb . GetDBConnexion ( )
2021-06-19 23:21:37 +02:00
cursor = cnx . cursor ( cursor_factory = ndb . ScoDocCursor )
2020-09-26 16:19:37 +02:00
cursor . execute ( req , { " formsemestre_id " : formsemestre_id } )
res = cursor . dictfetchall ( )
# etat de chaque evaluation:
for r in res :
r [ " jour " ] = r [ " jour " ] or datetime . date ( 1900 , 1 , 1 ) # pour les comparaisons
2021-07-21 15:58:49 +03:00
if with_etat :
2021-07-29 11:19:00 +03:00
r [ " etat " ] = do_evaluation_etat ( r [ " evaluation_id " ] )
2020-09-26 16:19:37 +02:00
return res
2021-06-19 23:21:37 +02:00
# ancien _notes_getall
def do_evaluation_get_all_notes (
2021-07-29 11:19:00 +03:00
evaluation_id , table = " notes_notes " , filter_suppressed = True , by_uid = None
2021-06-19 23:21:37 +02:00
) :
""" Toutes les notes pour une evaluation: { etudid : { ' value ' : value, ' date ' : date ... }}
Attention : inclut aussi les notes des étudiants qui ne sont plus inscrits au module .
"""
do_cache = (
filter_suppressed and table == " notes_notes " and ( by_uid is None )
) # pas de cache pour (rares) appels via undo_notes ou specifiant un enseignant
if do_cache :
2021-07-19 20:53:01 +03:00
r = sco_cache . EvaluationCache . get ( evaluation_id )
2021-06-19 23:21:37 +02:00
if r != None :
return r
cnx = ndb . GetDBConnexion ( )
cursor = cnx . cursor ( cursor_factory = ndb . ScoDocCursor )
cond = " where evaluation_id= %(evaluation_id)s "
if by_uid :
cond + = " and uid= %(by_uid)s "
cursor . execute (
" select * from " + table + cond ,
{ " evaluation_id " : evaluation_id , " by_uid " : by_uid } ,
)
res = cursor . dictfetchall ( )
d = { }
if filter_suppressed :
for x in res :
if x [ " value " ] != scu . NOTES_SUPPRESS :
d [ x [ " etudid " ] ] = x
else :
for x in res :
d [ x [ " etudid " ] ] = x
if do_cache :
2021-07-29 17:31:15 +03:00
status = sco_cache . EvaluationCache . set ( evaluation_id , d )
2021-07-29 17:48:27 +03:00
if not status :
log ( f " Warning: EvaluationCache.set: { evaluation_id } \t { status } " )
2021-06-19 23:21:37 +02:00
return d
2020-09-26 16:19:37 +02:00
def _eval_etat ( evals ) :
""" evals: list of mappings (etats)
- > nb_eval_completes , nb_evals_en_cours ,
nb_evals_vides , date derniere modif
Une eval est " complete " ssi tous les etudiants * inscrits * ont une note .
"""
nb_evals_completes , nb_evals_en_cours , nb_evals_vides = 0 , 0 , 0
dates = [ ]
for e in evals :
if e [ " etat " ] [ " evalcomplete " ] :
nb_evals_completes + = 1
elif e [ " etat " ] [ " nb_notes " ] == 0 :
nb_evals_vides + = 1
else :
nb_evals_en_cours + = 1
dates . append ( e [ " etat " ] [ " last_modif " ] )
2020-12-23 23:49:11 +01:00
dates = scu . sort_dates ( dates )
2020-09-26 16:19:37 +02:00
if len ( dates ) :
last_modif = dates [ - 1 ] # date de derniere modif d'une note dans un module
else :
last_modif = " "
return {
" nb_evals_completes " : nb_evals_completes ,
" nb_evals_en_cours " : nb_evals_en_cours ,
" nb_evals_vides " : nb_evals_vides ,
" last_modif " : last_modif ,
}
2021-07-29 11:19:00 +03:00
def do_evaluation_etat_in_sem ( formsemestre_id ) :
2020-09-26 16:19:37 +02:00
""" -> nb_eval_completes, nb_evals_en_cours, nb_evals_vides,
date derniere modif , attente """
2021-07-19 20:53:01 +03:00
nt = sco_cache . NotesTableCache . get (
formsemestre_id
2020-09-26 16:19:37 +02:00
) # > liste evaluations et moduleimpl en attente
evals = nt . get_sem_evaluation_etat_list ( )
etat = _eval_etat ( evals )
# Ajoute information sur notes en attente
etat [ " attente " ] = len ( nt . get_moduleimpls_attente ( ) ) > 0
return etat
2021-07-29 11:19:00 +03:00
def do_evaluation_etat_in_mod ( nt , moduleimpl_id ) :
2020-12-02 01:00:23 +01:00
""" """
2020-09-26 16:19:37 +02:00
evals = nt . get_mod_evaluation_etat_list ( moduleimpl_id )
etat = _eval_etat ( evals )
etat [ " attente " ] = moduleimpl_id in [
m [ " moduleimpl_id " ] for m in nt . get_moduleimpls_attente ( )
] # > liste moduleimpl en attente
return etat
2021-07-29 11:19:00 +03:00
def formsemestre_evaluations_cal ( formsemestre_id , REQUEST = None ) :
2020-09-26 16:19:37 +02:00
""" Page avec calendrier de toutes les evaluations de ce semestre """
2021-07-29 11:19:00 +03:00
context = None # #context
2020-09-26 16:19:37 +02:00
sem = sco_formsemestre . get_formsemestre ( context , formsemestre_id )
2021-07-19 20:53:01 +03:00
nt = sco_cache . NotesTableCache . get ( formsemestre_id ) # > liste evaluations
2020-09-26 16:19:37 +02:00
evals = nt . get_sem_evaluation_etat_list ( )
nb_evals = len ( evals )
color_incomplete = " #FF6060 "
color_complete = " #A0FFA0 "
color_futur = " #70E0FF "
today = time . strftime ( " % Y- % m- %d " )
year = int ( sem [ " annee_debut " ] )
if sem [ " mois_debut_ord " ] < 8 :
year - = 1 # calendrier septembre a septembre
events = { } # (day, halfday) : event
for e in evals :
etat = e [ " etat " ]
if not e [ " jour " ] :
continue
day = e [ " jour " ] . strftime ( " % Y- % m- %d " )
2021-01-17 22:31:28 +01:00
mod = sco_moduleimpl . do_moduleimpl_withmodule_list (
context , moduleimpl_id = e [ " moduleimpl_id " ]
) [ 0 ]
2020-09-26 16:19:37 +02:00
txt = mod [ " module " ] [ " code " ] or mod [ " module " ] [ " abbrev " ] or " eval "
if e [ " heure_debut " ] :
debut = e [ " heure_debut " ] . strftime ( " % Hh % M " )
else :
debut = " ? "
if e [ " heure_fin " ] :
fin = e [ " heure_fin " ] . strftime ( " % Hh % M " )
else :
fin = " ? "
description = " %s , de %s à %s " % ( mod [ " module " ] [ " titre " ] , debut , fin )
if etat [ " evalcomplete " ] :
color = color_complete
else :
color = color_incomplete
if day > today :
color = color_futur
href = " moduleimpl_status?moduleimpl_id= %s " % e [ " moduleimpl_id " ]
# if e['heure_debut'].hour < 12:
# halfday = True
# else:
# halfday = False
if not day in events :
# events[(day,halfday)] = [day, txt, color, href, halfday, description, mod]
events [ day ] = [ day , txt , color , href , description , mod ]
else :
e = events [ day ]
if e [ - 1 ] [ " moduleimpl_id " ] != mod [ " moduleimpl_id " ] :
# plusieurs evals de modules differents a la meme date
e [ 1 ] + = " , " + txt
e [ 4 ] + = " , " + description
if not etat [ " evalcomplete " ] :
e [ 2 ] = color_incomplete
if day > today :
e [ 2 ] = color_futur
2021-01-10 18:54:39 +01:00
CalHTML = sco_abs . YearTable (
2021-07-09 17:47:06 +02:00
context , year , events = list ( events . values ( ) ) , halfday = False , pad_width = None
2020-09-26 16:19:37 +02:00
)
H = [
2021-06-13 19:12:20 +02:00
html_sco_header . html_sem_header (
context ,
REQUEST ,
" Evaluations du semestre " ,
sem ,
cssstyles = [ " css/calabs.css " ] ,
2020-09-26 16:19:37 +02:00
) ,
' <div class= " cal_evaluations " > ' ,
CalHTML ,
" </div> " ,
" <p>soit %s évaluations planifiées; " % nb_evals ,
""" <ul><li>en <span style= " background-color: %s " >rouge</span> les évaluations passées auxquelles il manque des notes</li>
< li > en < span style = " background-color: %s " > vert < / span > les évaluations déjà notées < / li >
< li > en < span style = " background-color: %s " > bleu < / span > les évaluations futures < / li > < / ul > < / p > """
% ( color_incomplete , color_complete , color_futur ) ,
""" <p><a href= " formsemestre_evaluations_delai_correction?formsemestre_id= %s " class= " stdlink " >voir les délais de correction</a></p>
"""
% ( formsemestre_id , ) ,
2021-07-29 11:19:00 +03:00
html_sco_header . sco_footer ( ) ,
2020-09-26 16:19:37 +02:00
]
return " \n " . join ( H )
2021-07-29 11:19:00 +03:00
def evaluation_date_first_completion ( evaluation_id ) :
2020-09-26 16:19:37 +02:00
""" Première date à laquelle l ' évaluation a été complète
ou None si actuellement incomplète
"""
2021-07-29 11:19:00 +03:00
etat = do_evaluation_etat ( evaluation_id )
2020-09-26 16:19:37 +02:00
if not etat [ " evalcomplete " ] :
return None
2020-12-23 23:49:11 +01:00
# XXX inachevé ou à revoir ?
2020-09-26 16:19:37 +02:00
# Il faut considerer les inscriptions au semestre
# (pour avoir l'etat et le groupe) et aussi les inscriptions
# au module (pour gerer les modules optionnels correctement)
2021-07-29 11:19:00 +03:00
# E = do_evaluation_list(args={"evaluation_id": evaluation_id})[0]
2021-01-17 22:31:28 +01:00
# M = sco_moduleimpl.do_moduleimpl_list(context,moduleimpl_id=E["moduleimpl_id"])[0]
2020-12-23 23:49:11 +01:00
# formsemestre_id = M["formsemestre_id"]
2021-06-21 10:17:16 +02:00
# insem = sco_formsemestre_inscriptions.do_formsemestre_inscription_listinscrits(context, formsemestre_id)
2021-01-17 22:31:28 +01:00
# insmod = sco_moduleimpl.do_moduleimpl_inscription_list(context,moduleimpl_id=E["moduleimpl_id"])
2020-12-23 23:49:11 +01:00
# insmodset = set([x["etudid"] for x in insmod])
2020-09-26 16:19:37 +02:00
# retire de insem ceux qui ne sont pas inscrits au module
2020-12-23 23:49:11 +01:00
# ins = [i for i in insem if i["etudid"] in insmodset]
2020-09-26 16:19:37 +02:00
2021-07-09 17:47:06 +02:00
notes = list (
2021-07-29 11:19:00 +03:00
do_evaluation_get_all_notes ( evaluation_id , filter_suppressed = False ) . values ( )
2021-07-09 17:47:06 +02:00
)
notes_log = list (
do_evaluation_get_all_notes (
2021-07-29 11:19:00 +03:00
evaluation_id , filter_suppressed = False , table = " notes_notes_log "
2021-07-09 17:47:06 +02:00
) . values ( )
)
2020-09-26 16:19:37 +02:00
date_premiere_note = { } # etudid : date
for note in notes + notes_log :
etudid = note [ " etudid " ]
if etudid in date_premiere_note :
date_premiere_note [ etudid ] = min ( note [ " date " ] , date_premiere_note [ etudid ] )
else :
date_premiere_note [ etudid ] = note [ " date " ]
if not date_premiere_note :
return None # complete mais aucun etudiant non démissionnaires
# complet au moment du max (date la plus tardive) des premieres dates de saisie
return max ( date_premiere_note . values ( ) )
def formsemestre_evaluations_delai_correction (
2021-07-29 11:19:00 +03:00
formsemestre_id , format = " html " , REQUEST = None
2020-09-26 16:19:37 +02:00
) :
""" Experimental: un tableau indiquant pour chaque évaluation
le nombre de jours avant la publication des notes .
N ' indique pas les évaluations de ratrapage ni celles des modules de bonus/malus.
"""
2021-07-29 11:19:00 +03:00
context = None # #context
2020-09-26 16:19:37 +02:00
sem = sco_formsemestre . get_formsemestre ( context , formsemestre_id )
2021-07-19 20:53:01 +03:00
nt = sco_cache . NotesTableCache . get ( formsemestre_id ) # > liste evaluations
2020-09-26 16:19:37 +02:00
evals = nt . get_sem_evaluation_etat_list ( )
T = [ ]
for e in evals :
2021-01-17 22:31:28 +01:00
M = sco_moduleimpl . do_moduleimpl_list (
context , moduleimpl_id = e [ " moduleimpl_id " ]
) [ 0 ]
2021-06-17 00:08:37 +02:00
Mod = sco_edit_module . do_module_list (
context , args = { " module_id " : M [ " module_id " ] }
) [ 0 ]
2020-12-23 23:49:11 +01:00
if ( e [ " evaluation_type " ] != scu . EVALUATION_NORMALE ) or (
Mod [ " module_type " ] == scu . MODULE_MALUS
2020-09-26 16:19:37 +02:00
) :
continue
2021-07-29 11:19:00 +03:00
e [ " date_first_complete " ] = evaluation_date_first_completion ( e [ " evaluation_id " ] )
2020-09-26 16:19:37 +02:00
if e [ " date_first_complete " ] :
e [ " delai_correction " ] = ( e [ " date_first_complete " ] . date ( ) - e [ " jour " ] ) . days
else :
e [ " delai_correction " ] = None
e [ " module_code " ] = Mod [ " code " ]
e [ " _module_code_target " ] = (
" moduleimpl_status?moduleimpl_id= " + M [ " moduleimpl_id " ]
)
e [ " module_titre " ] = Mod [ " titre " ]
e [ " responsable_id " ] = M [ " responsable_id " ]
2021-07-03 23:35:32 +02:00
e [ " responsable_nomplogin " ] = sco_users . user_info ( M [ " responsable_id " ] ) [
2020-09-26 16:19:37 +02:00
" nomplogin "
]
e [ " _jour_target " ] = " evaluation_listenotes?evaluation_id= " + e [ " evaluation_id " ]
T . append ( e )
columns_ids = (
" module_code " ,
" module_titre " ,
" responsable_nomplogin " ,
" jour " ,
" date_first_complete " ,
" delai_correction " ,
" description " ,
)
titles = {
" module_code " : " Code " ,
" module_titre " : " Module " ,
" responsable_nomplogin " : " Responsable " ,
" jour " : " Date " ,
" date_first_complete " : " Fin saisie " ,
" delai_correction " : " Délai " ,
" description " : " Description " ,
}
tab = GenTable (
titles = titles ,
columns_ids = columns_ids ,
rows = T ,
html_class = " table_leftalign table_coldate " ,
html_sortable = True ,
html_title = " <h2>Correction des évaluations du semestre</h2> " ,
caption = " Correction des évaluations du semestre " ,
2021-07-28 18:03:54 +03:00
preferences = sco_preferences . SemPreferences ( formsemestre_id ) ,
2020-09-26 16:19:37 +02:00
base_url = " %s ?formsemestre_id= %s " % ( REQUEST . URL0 , formsemestre_id ) ,
2020-12-23 23:49:11 +01:00
origin = " Généré par %s le " % VERSION . SCONAME + scu . timedate_human_repr ( ) + " " ,
filename = scu . make_filename ( " evaluations_delais_ " + sem [ " titreannee " ] ) ,
2020-09-26 16:19:37 +02:00
)
return tab . make_page ( context , format = format , REQUEST = REQUEST )
2021-07-29 11:19:00 +03:00
def module_evaluation_insert_before ( ModEvals , next_eval ) :
2020-09-26 16:19:37 +02:00
""" Renumber evals such that an evaluation with can be inserted before next_eval
Returns numero suitable for the inserted evaluation
"""
if next_eval :
n = next_eval [ " numero " ]
if not n :
log ( " renumbering old evals " )
2021-07-29 11:19:00 +03:00
module_evaluation_renumber ( next_eval [ " moduleimpl_id " ] )
2021-06-17 00:08:37 +02:00
next_eval = do_evaluation_list (
2021-07-29 11:19:00 +03:00
args = { " evaluation_id " : next_eval [ " evaluation_id " ] }
2020-09-26 16:19:37 +02:00
) [ 0 ]
n = next_eval [ " numero " ]
else :
n = 1
# log('inserting at position numero %s' % n )
# all numeros >= n are incremented
for e in ModEvals :
if e [ " numero " ] > = n :
e [ " numero " ] + = 1
# log('incrementing %s to %s' % (e['evaluation_id'], e['numero']))
2021-07-29 11:19:00 +03:00
do_evaluation_edit ( e )
2020-09-26 16:19:37 +02:00
return n
2021-07-29 11:19:00 +03:00
def module_evaluation_move ( evaluation_id , after = 0 , redirect = 1 ) :
2020-09-26 16:19:37 +02:00
""" Move before/after previous one (decrement/increment numero)
( published )
"""
2021-07-29 11:19:00 +03:00
e = do_evaluation_list ( args = { " evaluation_id " : evaluation_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
redirect = int ( redirect )
2021-06-21 10:17:16 +02:00
# access: can change eval ?
2021-07-29 11:19:00 +03:00
if not sco_permissions_check . can_edit_evaluation ( moduleimpl_id = e [ " moduleimpl_id " ] ) :
2021-06-21 10:17:16 +02:00
raise AccessDenied (
2021-07-29 11:19:00 +03:00
" Modification évaluation impossible pour %s " % current_user . get_nomplogin ( )
2021-06-21 10:17:16 +02:00
)
2020-09-26 16:19:37 +02:00
2021-07-29 11:19:00 +03:00
module_evaluation_renumber ( e [ " moduleimpl_id " ] , only_if_unumbered = True )
e = do_evaluation_list ( args = { " evaluation_id " : evaluation_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
after = int ( after ) # 0: deplace avant, 1 deplace apres
if after not in ( 0 , 1 ) :
raise ValueError ( ' invalid value for " after " ' )
2021-07-29 11:19:00 +03:00
ModEvals = do_evaluation_list ( { " moduleimpl_id " : e [ " moduleimpl_id " ] } )
2020-09-26 16:19:37 +02:00
# log('ModEvals=%s' % [ x['evaluation_id'] for x in ModEvals] )
if len ( ModEvals ) > 1 :
idx = [ p [ " evaluation_id " ] for p in ModEvals ] . index ( evaluation_id )
neigh = None # object to swap with
if after == 0 and idx > 0 :
neigh = ModEvals [ idx - 1 ]
elif after == 1 and idx < len ( ModEvals ) - 1 :
neigh = ModEvals [ idx + 1 ]
if neigh : #
# swap numero with neighbor
e [ " numero " ] , neigh [ " numero " ] = neigh [ " numero " ] , e [ " numero " ]
2021-07-29 11:19:00 +03:00
do_evaluation_edit ( e )
do_evaluation_edit ( neigh )
2020-09-26 16:19:37 +02:00
# redirect to moduleimpl page:
if redirect :
2021-07-29 11:19:00 +03:00
return flask . redirect (
url_for (
" notes.moduleimpl_status " ,
scodoc_dept = g . scodoc_dept ,
moduleimpl_id = e [ " moduleimpl_id " ] ,
)
2020-09-26 16:19:37 +02:00
)
2021-07-29 11:19:00 +03:00
def module_evaluation_renumber ( moduleimpl_id , only_if_unumbered = False , redirect = 0 ) :
2020-09-26 16:19:37 +02:00
""" Renumber evaluations in this module, according to their date. (numero=0: oldest one)
Needed because previous versions of ScoDoc did not have eval numeros
Note : existing numeros are ignored
"""
redirect = int ( redirect )
# log('module_evaluation_renumber( moduleimpl_id=%s )' % moduleimpl_id )
# List sorted according to date/heure, ignoring numeros:
# (note that we place evaluations with NULL date at the end)
2021-06-17 00:08:37 +02:00
ModEvals = do_evaluation_list (
args = { " moduleimpl_id " : moduleimpl_id } ,
sortkey = " jour asc, heure_debut asc " ,
2020-09-26 16:19:37 +02:00
)
all_numbered = False not in [ x [ " numero " ] > 0 for x in ModEvals ]
if all_numbered and only_if_unumbered :
return # all ok
# log('module_evaluation_renumber')
# Reset all numeros:
i = 1
for e in ModEvals :
e [ " numero " ] = i
2021-07-29 11:19:00 +03:00
do_evaluation_edit ( e )
2020-09-26 16:19:37 +02:00
i + = 1
# If requested, redirect to moduleimpl page:
if redirect :
2021-07-29 11:19:00 +03:00
return flask . redirect (
url_for (
" notes.moduleimpl_status " ,
scodoc_dept = g . scodoc_dept ,
moduleimpl_id = moduleimpl_id ,
)
2020-09-26 16:19:37 +02:00
)
# -------------- VIEWS
2021-07-29 11:19:00 +03:00
def evaluation_describe ( evaluation_id = " " , edit_in_place = True , REQUEST = None ) :
2020-09-26 16:19:37 +02:00
""" HTML description of evaluation, for page headers
edit_in_place : allow in - place editing when permitted ( not implemented )
"""
2021-07-29 11:19:00 +03:00
context = None # #context
2021-06-19 23:21:37 +02:00
from app . scodoc import sco_saisie_notes
2021-07-29 11:19:00 +03:00
E = do_evaluation_list ( { " evaluation_id " : evaluation_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
moduleimpl_id = E [ " moduleimpl_id " ]
2021-01-17 22:31:28 +01:00
M = sco_moduleimpl . do_moduleimpl_list ( context , moduleimpl_id = moduleimpl_id ) [ 0 ]
2021-06-16 18:18:32 +02:00
Mod = sco_edit_module . do_module_list ( context , args = { " module_id " : M [ " module_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
formsemestre_id = M [ " formsemestre_id " ]
2021-07-03 23:35:32 +02:00
u = sco_users . user_info ( M [ " responsable_id " ] )
2020-09-26 16:19:37 +02:00
resp = u [ " prenomnom " ]
nomcomplet = u [ " nomcomplet " ]
2021-06-19 23:21:37 +02:00
can_edit = sco_permissions_check . can_edit_notes (
2021-01-17 22:31:28 +01:00
context , REQUEST . AUTHENTICATED_USER , moduleimpl_id , allow_ens = False
2020-09-26 16:19:37 +02:00
)
link = (
' <span class= " evallink " ><a class= " stdlink " href= " evaluation_listenotes?moduleimpl_id= %s " >voir toutes les notes du module</a></span> '
% moduleimpl_id
)
mod_descr = (
' <a href= " moduleimpl_status?moduleimpl_id= %s " > %s %s </a> <span class= " resp " >(resp. <a title= " %s " > %s </a>)</span> %s '
% ( moduleimpl_id , Mod [ " code " ] , Mod [ " titre " ] , nomcomplet , resp , link )
)
etit = E [ " description " ] or " "
if etit :
etit = ' " ' + etit + ' " '
2020-12-23 23:49:11 +01:00
if Mod [ " module_type " ] == scu . MODULE_MALUS :
2020-09-26 16:19:37 +02:00
etit + = ' <span class= " eval_malus " >(points de malus)</span> '
H = [
' <span class= " eval_title " >Evaluation %s </span><p><b>Module : %s </b></p> '
% ( etit , mod_descr )
]
2020-12-23 23:49:11 +01:00
if Mod [ " module_type " ] == scu . MODULE_MALUS :
2020-09-26 16:19:37 +02:00
# Indique l'UE
2021-06-16 18:18:32 +02:00
ue = sco_edit_ue . do_ue_list ( context , args = { " ue_id " : Mod [ " ue_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
H . append ( " <p><b>UE : %(acronyme)s </b></p> " % ue )
# store min/max values used by JS client-side checks:
H . append (
' <span id= " eval_note_min " class= " sco-hidden " >-20.</span><span id= " eval_note_max " class= " sco-hidden " >20.</span> '
)
else :
# date et absences (pas pour evals de malus)
jour = E [ " jour " ] or " <em>pas de date</em> "
H . append (
" <p>Réalisée le <b> %s </b> de %s à %s "
% ( jour , E [ " heure_debut " ] , E [ " heure_fin " ] )
)
if E [ " jour " ] :
2021-07-29 17:31:15 +03:00
group_id = sco_groups . get_default_group ( formsemestre_id )
2020-09-26 16:19:37 +02:00
H . append (
' <span class= " noprint " ><a href= " %s /Absences/EtatAbsencesDate?group_ids= %s &date= %s " >(absences ce jour)</a></span> '
2021-07-09 17:47:06 +02:00
% (
scu . ScoURL ( ) ,
group_id ,
six . moves . urllib . parse . quote ( E [ " jour " ] , safe = " " ) ,
)
2020-09-26 16:19:37 +02:00
)
H . append (
' </p><p>Coefficient dans le module: <b> %s </b>, notes sur <span id= " eval_note_max " > %g </span> '
% ( E [ " coefficient " ] , E [ " note_max " ] )
)
H . append ( ' <span id= " eval_note_min " class= " sco-hidden " >0.</span> ' )
if can_edit :
H . append (
' <a href= " evaluation_edit?evaluation_id= %s " >(modifier l \' évaluation)</a> '
% evaluation_id
)
H . append ( " </p> " )
return ' <div class= " eval_description " > ' + " \n " . join ( H ) + " </div> "
def evaluation_create_form (
moduleimpl_id = None ,
evaluation_id = None ,
REQUEST = None ,
edit = False ,
readonly = False ,
page_title = " Evaluation " ,
) :
" formulaire creation/edition des evaluations (pas des notes) "
2021-07-29 11:19:00 +03:00
context = None # #context
2020-09-26 16:19:37 +02:00
if evaluation_id != None :
2021-07-29 11:19:00 +03:00
the_eval = do_evaluation_list ( { " evaluation_id " : evaluation_id } ) [ 0 ]
2020-09-26 16:19:37 +02:00
moduleimpl_id = the_eval [ " moduleimpl_id " ]
#
2021-01-17 22:31:28 +01:00
M = sco_moduleimpl . do_moduleimpl_withmodule_list (
context , moduleimpl_id = moduleimpl_id
) [ 0 ]
2020-12-23 23:49:11 +01:00
is_malus = M [ " module " ] [ " module_type " ] == scu . MODULE_MALUS # True si module de malus
2020-09-26 16:19:37 +02:00
formsemestre_id = M [ " formsemestre_id " ]
2020-12-23 23:49:11 +01:00
min_note_max = scu . NOTES_PRECISION # le plus petit bareme possible
2020-09-26 16:19:37 +02:00
if not readonly :
2021-07-29 11:19:00 +03:00
if not sco_permissions_check . can_edit_evaluation ( moduleimpl_id = moduleimpl_id ) :
2020-09-26 16:19:37 +02:00
return (
2021-07-29 17:31:15 +03:00
html_sco_header . sco_header ( )
2020-09-26 16:19:37 +02:00
+ " <h2>Opération non autorisée</h2><p> "
2021-06-21 10:17:16 +02:00
+ " Modification évaluation impossible pour %s "
2021-07-29 11:19:00 +03:00
% current_user . get_nomplogin ( )
2020-09-26 16:19:37 +02:00
+ " </p> "
2021-06-21 23:13:57 +02:00
+ ' <p><a href= " moduleimpl_status?moduleimpl_id= %s " >Revenir</a></p> '
% ( moduleimpl_id , )
2021-07-29 11:19:00 +03:00
+ html_sco_header . sco_footer ( )
2020-09-26 16:19:37 +02:00
)
if readonly :
edit = True # montre les donnees existantes
if not edit :
# creation nouvel
if moduleimpl_id is None :
raise ValueError ( " missing moduleimpl_id parameter " )
initvalues = {
" note_max " : 20 ,
" jour " : time . strftime ( " %d / % m/ % Y " , time . localtime ( ) ) ,
" publish_incomplete " : is_malus ,
}
submitlabel = " Créer cette évaluation "
action = " Création d ' une é "
link = " "
else :
# edition donnees existantes
# setup form init values
if evaluation_id is None :
raise ValueError ( " missing evaluation_id parameter " )
initvalues = the_eval
moduleimpl_id = initvalues [ " moduleimpl_id " ]
submitlabel = " Modifier les données "
if readonly :
action = " E "
link = (
' <span class= " evallink " ><a class= " stdlink " href= " evaluation_listenotes?moduleimpl_id= %s " >voir toutes les notes du module</a></span> '
% M [ " moduleimpl_id " ]
)
else :
action = " Modification d ' une é "
link = " "
# Note maximale actuelle dans cette eval ?
2021-07-29 11:19:00 +03:00
etat = do_evaluation_etat ( evaluation_id )
2020-09-26 16:19:37 +02:00
if etat [ " maxi_num " ] is not None :
2020-12-23 23:49:11 +01:00
min_note_max = max ( scu . NOTES_PRECISION , etat [ " maxi_num " ] )
2020-09-26 16:19:37 +02:00
else :
2020-12-23 23:49:11 +01:00
min_note_max = scu . NOTES_PRECISION
2020-09-26 16:19:37 +02:00
#
2020-12-23 23:49:11 +01:00
if min_note_max > scu . NOTES_PRECISION :
min_note_max_str = scu . fmt_note ( min_note_max )
2020-09-26 16:19:37 +02:00
else :
min_note_max_str = " 0 "
#
2021-06-16 18:18:32 +02:00
Mod = sco_edit_module . do_module_list ( context , args = { " module_id " : M [ " module_id " ] } ) [ 0 ]
2020-09-26 16:19:37 +02:00
#
help = """ <div class= " help " ><p class= " help " >
Le coefficient d ' une évaluation n ' est utilisé que pour pondérer les évaluations au sein d ' un module.
Il est fixé librement par l ' enseignant pour refléter l ' importance de ses différentes notes
( examens , projets , travaux pratiques . . . ) . Ce coefficient est utilisé pour calculer la note
moyenne de chaque étudiant dans ce module .
< / p > < p class = " help " >
Ne pas confondre ce coefficient avec le coefficient du module , qui est lui fixé par le programme
pédagogique ( le PPN pour les DUT ) et pondère les moyennes de chaque module pour obtenir
les moyennes d ' UE et la moyenne générale.
< / p > < p class = " help " >
L ' option <em>Visible sur bulletins</em> indique que la note sera reportée sur les bulletins
en version dite " intermédiaire " ( dans cette version , on peut ne faire apparaitre que certaines
notes , en sus des moyennes de modules . Attention , cette option n ' empêche pas la publication sur
les bulletins en version " longue " ( la note est donc visible par les étudiants sur le portail ) .
< / p > < p class = " help " >
2021-03-11 14:49:37 +01:00
Les modalités " rattrapage " et " deuxième session " définissent des évaluations prises en compte de
façon spéciale : < / p >
< ul >
< li > les notes d ' une évaluation de " rattrapage " remplaceront les moyennes du module
< em > si elles sont meilleures que celles calculées < / em > . < / li >
< li > les notes de " deuxième session " remplacent , lorsqu ' elles sont saisies, la moyenne de l ' étudiant
à ce module , même si la note de deuxième session est plus faible . < / li >
< / ul >
< p class = " help " >
Dans ces deux cas , le coefficient est ignoré , et toutes les notes n ' ont
2020-09-26 16:19:37 +02:00
pas besoin d ' être rentrées.
< / p >
< p class = " help " >
2021-03-11 14:49:37 +01:00
Par ailleurs , les évaluations des modules de type " malus " sont toujours spéciales : le coefficient n ' est pas utilisé.
2020-09-26 16:19:37 +02:00
Les notes de malus sont toujours comprises entre - 20 et 20. Les points sont soustraits à la moyenne
de l ' UE à laquelle appartient le module malus (si la note est négative, la moyenne est donc augmentée).
< / p >
"""
mod_descr = ' <a href= " moduleimpl_status?moduleimpl_id= %s " > %s %s </a> %s ' % (
moduleimpl_id ,
Mod [ " code " ] ,
Mod [ " titre " ] ,
link ,
)
if not readonly :
H = [ " <h3> %s valuation en %s </h3> " % ( action , mod_descr ) ]
else :
2021-07-29 11:19:00 +03:00
return evaluation_describe ( evaluation_id , REQUEST = REQUEST )
2020-09-26 16:19:37 +02:00
heures = [ " %02d h %02d " % ( h , m ) for h in range ( 8 , 19 ) for m in ( 0 , 30 ) ]
#
initvalues [ " visibulletin " ] = initvalues . get ( " visibulletin " , " 1 " )
if initvalues [ " visibulletin " ] == " 1 " :
initvalues [ " visibulletinlist " ] = [ " X " ]
else :
initvalues [ " visibulletinlist " ] = [ ]
2021-07-09 17:47:06 +02:00
if (
REQUEST . form . get ( " tf-submitted " , False )
and " visibulletinlist " not in REQUEST . form
2020-09-26 16:19:37 +02:00
) :
REQUEST . form [ " visibulletinlist " ] = [ ]
#
form = [
( " evaluation_id " , { " default " : evaluation_id , " input_type " : " hidden " } ) ,
( " formsemestre_id " , { " default " : formsemestre_id , " input_type " : " hidden " } ) ,
( " moduleimpl_id " , { " default " : moduleimpl_id , " input_type " : " hidden " } ) ,
# ('jour', { 'title' : 'Date (j/m/a)', 'size' : 12, 'explanation' : 'date de l\'examen, devoir ou contrôle' }),
(
" jour " ,
{
" input_type " : " date " ,
" title " : " Date " ,
" size " : 12 ,
" explanation " : " date de l ' examen, devoir ou contrôle " ,
} ,
) ,
(
" heure_debut " ,
{
" title " : " Heure de début " ,
" explanation " : " heure du début de l ' épreuve " ,
" input_type " : " menu " ,
" allowed_values " : heures ,
" labels " : heures ,
} ,
) ,
(
" heure_fin " ,
{
" title " : " Heure de fin " ,
" explanation " : " heure de fin de l ' épreuve " ,
" input_type " : " menu " ,
" allowed_values " : heures ,
" labels " : heures ,
} ,
) ,
]
if is_malus : # pas de coefficient
form . append ( ( " coefficient " , { " input_type " : " hidden " , " default " : " 1. " } ) )
else :
form . append (
(
" coefficient " ,
{
" size " : 10 ,
" type " : " float " ,
" explanation " : " coef. dans le module (choisi librement par l ' enseignant) " ,
" allow_null " : False ,
} ,
)
)
form + = [
(
" note_max " ,
{
" size " : 4 ,
" type " : " float " ,
" title " : " Notes de 0 à " ,
" explanation " : " barème (note max actuelle: %s ) " % min_note_max_str ,
" allow_null " : False ,
2020-12-23 23:49:11 +01:00
" max_value " : scu . NOTES_MAX ,
2020-09-26 16:19:37 +02:00
" min_value " : min_note_max ,
} ,
) ,
(
" description " ,
{
" size " : 36 ,
" type " : " text " ,
" explanation " : ' type d \' évaluation, apparait sur le bulletins longs. Exemples: " contrôle court " , " examen de TP " , " examen final " . ' ,
} ,
) ,
(
" visibulletinlist " ,
{
" input_type " : " checkbox " ,
" allowed_values " : [ " X " ] ,
" labels " : [ " " ] ,
" title " : " Visible sur bulletins " ,
" explanation " : " (pour les bulletins en version intermédiaire) " ,
} ,
) ,
(
" publish_incomplete " ,
{
" input_type " : " boolcheckbox " ,
" title " : " Prise en compte immédiate " ,
" explanation " : " notes utilisées même si incomplètes " ,
} ,
) ,
(
" evaluation_type " ,
{
" input_type " : " menu " ,
" title " : " Modalité " ,
2021-03-11 14:49:37 +01:00
" allowed_values " : (
scu . EVALUATION_NORMALE ,
scu . EVALUATION_RATTRAPAGE ,
scu . EVALUATION_SESSION2 ,
) ,
2020-09-26 16:19:37 +02:00
" type " : " int " ,
2021-03-11 14:49:37 +01:00
" labels " : (
" Normale " ,
" Rattrapage (remplace si meilleure note) " ,
" Deuxième session (remplace toujours) " ,
) ,
2020-09-26 16:19:37 +02:00
} ,
) ,
]
tf = TrivialFormulator (
REQUEST . URL0 ,
REQUEST . form ,
form ,
cancelbutton = " Annuler " ,
submitlabel = submitlabel ,
initvalues = initvalues ,
readonly = readonly ,
)
dest_url = " moduleimpl_status?moduleimpl_id= %s " % M [ " moduleimpl_id " ]
if tf [ 0 ] == 0 :
2021-07-29 17:31:15 +03:00
head = html_sco_header . sco_header ( page_title = page_title )
2021-07-29 11:19:00 +03:00
return head + " \n " . join ( H ) + " \n " + tf [ 1 ] + help + html_sco_header . sco_footer ( )
2020-09-26 16:19:37 +02:00
elif tf [ 0 ] == - 1 :
return REQUEST . RESPONSE . redirect ( dest_url )
else :
# form submission
if tf [ 2 ] [ " visibulletinlist " ] :
tf [ 2 ] [ " visibulletin " ] = 1
else :
tf [ 2 ] [ " visibulletin " ] = 0
if not edit :
# creation d'une evaluation
2021-07-29 11:19:00 +03:00
evaluation_id = do_evaluation_create ( REQUEST = REQUEST , * * tf [ 2 ] )
2020-09-26 16:19:37 +02:00
return REQUEST . RESPONSE . redirect ( dest_url )
else :
2021-07-29 11:19:00 +03:00
do_evaluation_edit ( tf [ 2 ] )
2020-09-26 16:19:37 +02:00
return REQUEST . RESPONSE . redirect ( dest_url )