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
|
|
|
|
#
|
|
|
|
##############################################################################
|
|
|
|
|
2021-09-29 20:08:18 +02:00
|
|
|
""" Importation des étudiants à partir de fichiers CSV
|
2020-09-26 16:19:37 +02:00
|
|
|
"""
|
|
|
|
|
2021-02-03 22:00:41 +01:00
|
|
|
import collections
|
2021-09-29 20:08:18 +02:00
|
|
|
import io
|
2021-07-25 10:51:09 +03:00
|
|
|
import os
|
2021-02-04 20:02:44 +01:00
|
|
|
import re
|
2021-07-25 10:51:09 +03:00
|
|
|
import time
|
2021-08-14 10:12:40 +02:00
|
|
|
from datetime import date
|
|
|
|
|
2021-08-15 21:33:47 +02:00
|
|
|
from flask import g, url_for
|
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
|
2021-08-29 19:57:32 +02:00
|
|
|
from app import log
|
2021-08-14 10:12:40 +02:00
|
|
|
from app.scodoc.sco_excel import COLORS
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc.sco_formsemestre_inscriptions import (
|
|
|
|
do_formsemestre_inscription_with_modules,
|
|
|
|
)
|
|
|
|
from app.scodoc.gen_tables import GenTable
|
|
|
|
from app.scodoc.sco_exceptions import (
|
2021-02-03 22:00:41 +01:00
|
|
|
AccessDenied,
|
|
|
|
FormatError,
|
|
|
|
ScoException,
|
|
|
|
ScoValueError,
|
|
|
|
ScoInvalidDateError,
|
|
|
|
ScoLockedFormError,
|
|
|
|
ScoGenError,
|
|
|
|
)
|
2021-06-19 23:21:37 +02:00
|
|
|
from app.scodoc import html_sco_header
|
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_etud
|
|
|
|
from app.scodoc import sco_formsemestre
|
|
|
|
from app.scodoc import sco_groups
|
|
|
|
from app.scodoc import sco_excel
|
|
|
|
from app.scodoc import sco_groups_view
|
|
|
|
from app.scodoc import sco_news
|
|
|
|
from app.scodoc import sco_preferences
|
2020-09-26 16:19:37 +02:00
|
|
|
|
2021-07-25 10:51:09 +03:00
|
|
|
# format description (in tools/)
|
|
|
|
FORMAT_FILE = "format_import_etudiants.txt"
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
# Champs modifiables via "Import données admission"
|
|
|
|
ADMISSION_MODIFIABLE_FIELDS = (
|
|
|
|
"code_nip",
|
|
|
|
"code_ine",
|
|
|
|
"date_naissance",
|
|
|
|
"lieu_naissance",
|
|
|
|
"bac",
|
|
|
|
"specialite",
|
|
|
|
"annee_bac",
|
|
|
|
"math",
|
|
|
|
"physique",
|
|
|
|
"anglais",
|
|
|
|
"francais",
|
|
|
|
"type_admission",
|
|
|
|
"boursier_prec",
|
|
|
|
"qualite",
|
|
|
|
"rapporteur",
|
|
|
|
"score",
|
|
|
|
"commentaire",
|
|
|
|
"classement",
|
|
|
|
"apb_groupe",
|
|
|
|
"apb_classement_gr",
|
|
|
|
"nomlycee",
|
|
|
|
"villelycee",
|
|
|
|
"codepostallycee",
|
|
|
|
"codelycee",
|
|
|
|
# Adresse:
|
|
|
|
"email",
|
|
|
|
"emailperso",
|
|
|
|
"domicile",
|
|
|
|
"codepostaldomicile",
|
|
|
|
"villedomicile",
|
|
|
|
"paysdomicile",
|
|
|
|
"telephone",
|
|
|
|
"telephonemobile",
|
|
|
|
# Groupes
|
|
|
|
"groupes",
|
|
|
|
)
|
|
|
|
|
|
|
|
# ----
|
|
|
|
|
|
|
|
|
|
|
|
def sco_import_format(with_codesemestre=True):
|
|
|
|
"returns tuples (Attribut, Type, Table, AllowNulls, Description)"
|
|
|
|
r = []
|
2021-07-25 10:51:09 +03:00
|
|
|
for l in open(os.path.join(scu.SCO_TOOLS_DIR, FORMAT_FILE)):
|
2020-09-26 16:19:37 +02:00
|
|
|
l = l.strip()
|
|
|
|
if l and l[0] != "#":
|
|
|
|
fs = l.split(";")
|
|
|
|
if len(fs) < 5:
|
|
|
|
# Bug: invalid format file (fatal)
|
|
|
|
raise ScoException(
|
|
|
|
"file %s has invalid format (expected %d fields, got %d) (%s)"
|
|
|
|
% (FORMAT_FILE, 5, len(fs), l)
|
|
|
|
)
|
|
|
|
fieldname = (
|
|
|
|
fs[0].strip().lower().split()[0]
|
|
|
|
) # titre attribut: normalize, 1er mot seulement (nom du champ en BD)
|
|
|
|
typ, table, allow_nulls, description = [x.strip() for x in fs[1:5]]
|
|
|
|
aliases = [x.strip() for x in fs[5:] if x.strip()]
|
|
|
|
if fieldname not in aliases:
|
|
|
|
aliases.insert(0, fieldname) # prepend
|
|
|
|
if with_codesemestre or fs[0] != "codesemestre":
|
|
|
|
r.append((fieldname, typ, table, allow_nulls, description, aliases))
|
|
|
|
return r
|
|
|
|
|
|
|
|
|
|
|
|
def sco_import_format_dict(with_codesemestre=True):
|
2020-12-02 01:00:23 +01:00
|
|
|
"""Attribut: { 'type': , 'table', 'allow_nulls' , 'description' }"""
|
2020-09-26 16:19:37 +02:00
|
|
|
fmt = sco_import_format(with_codesemestre=with_codesemestre)
|
|
|
|
R = collections.OrderedDict()
|
|
|
|
for l in fmt:
|
|
|
|
R[l[0]] = {
|
|
|
|
"type": l[1],
|
|
|
|
"table": l[2],
|
|
|
|
"allow_nulls": l[3],
|
|
|
|
"description": l[4],
|
|
|
|
"aliases": l[5],
|
|
|
|
}
|
|
|
|
return R
|
|
|
|
|
|
|
|
|
|
|
|
def sco_import_generate_excel_sample(
|
|
|
|
fmt,
|
|
|
|
with_codesemestre=True,
|
|
|
|
only_tables=None,
|
|
|
|
with_groups=True,
|
|
|
|
exclude_cols=[],
|
|
|
|
extra_cols=[],
|
|
|
|
group_ids=[],
|
|
|
|
):
|
|
|
|
"""Generates an excel document based on format fmt
|
|
|
|
(format is the result of sco_import_format())
|
|
|
|
If not None, only_tables can specify a list of sql table names
|
|
|
|
(only columns from these tables will be generated)
|
|
|
|
If group_ids, liste les etudiants de ces groupes
|
|
|
|
"""
|
2021-08-12 14:55:25 +02:00
|
|
|
style = sco_excel.excel_make_style(bold=True)
|
2021-08-14 10:12:40 +02:00
|
|
|
style_required = sco_excel.excel_make_style(bold=True, color=COLORS.RED)
|
2020-09-26 16:19:37 +02:00
|
|
|
titles = []
|
|
|
|
titlesStyles = []
|
|
|
|
for l in fmt:
|
2021-08-21 00:24:51 +02:00
|
|
|
name = l[0].lower()
|
2020-09-26 16:19:37 +02:00
|
|
|
if (not with_codesemestre) and name == "codesemestre":
|
|
|
|
continue # pas de colonne codesemestre
|
2021-08-21 00:24:51 +02:00
|
|
|
if only_tables is not None and l[2].lower() not in only_tables:
|
2020-09-26 16:19:37 +02:00
|
|
|
continue # table non demandée
|
|
|
|
if name in exclude_cols:
|
|
|
|
continue # colonne exclue
|
|
|
|
if int(l[3]):
|
|
|
|
titlesStyles.append(style)
|
|
|
|
else:
|
|
|
|
titlesStyles.append(style_required)
|
|
|
|
titles.append(name)
|
|
|
|
if with_groups and "groupes" not in titles:
|
|
|
|
titles.append("groupes")
|
|
|
|
titlesStyles.append(style)
|
|
|
|
titles += extra_cols
|
|
|
|
titlesStyles += [style] * len(extra_cols)
|
2021-08-21 00:24:51 +02:00
|
|
|
if group_ids:
|
2021-09-08 00:34:45 +02:00
|
|
|
groups_infos = sco_groups_view.DisplayedGroupsInfos(group_ids)
|
2020-09-26 16:19:37 +02:00
|
|
|
members = groups_infos.members
|
|
|
|
log(
|
|
|
|
"sco_import_generate_excel_sample: group_ids=%s %d members"
|
|
|
|
% (group_ids, len(members))
|
|
|
|
)
|
|
|
|
titles = ["etudid"] + titles
|
|
|
|
titlesStyles = [style] + titlesStyles
|
|
|
|
# rempli table avec données actuelles
|
|
|
|
lines = []
|
|
|
|
for i in members:
|
2021-06-19 23:21:37 +02:00
|
|
|
etud = sco_etud.get_etud_info(etudid=i["etudid"], filled=True)[0]
|
2020-09-26 16:19:37 +02:00
|
|
|
l = []
|
|
|
|
for field in titles:
|
|
|
|
if field == "groupes":
|
|
|
|
sco_groups.etud_add_group_infos(
|
2021-08-19 10:28:35 +02:00
|
|
|
etud, groups_infos.formsemestre, sep=";"
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
l.append(etud["partitionsgroupes"])
|
|
|
|
else:
|
2021-08-21 00:24:51 +02:00
|
|
|
key = field.lower().split()[0]
|
2020-09-26 16:19:37 +02:00
|
|
|
l.append(etud.get(key, ""))
|
|
|
|
lines.append(l)
|
|
|
|
else:
|
|
|
|
lines = [[]] # empty content, titles only
|
2021-08-12 14:55:25 +02:00
|
|
|
return sco_excel.excel_simple_table(
|
2021-08-14 10:12:40 +02:00
|
|
|
titles=titles, titles_styles=titlesStyles, sheet_name="Etudiants", lines=lines
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def students_import_excel(
|
|
|
|
csvfile,
|
|
|
|
formsemestre_id=None,
|
|
|
|
check_homonyms=True,
|
|
|
|
require_ine=False,
|
2021-09-27 10:20:10 +02:00
|
|
|
return_html=True,
|
2020-09-26 16:19:37 +02:00
|
|
|
):
|
|
|
|
"import students from Excel file"
|
|
|
|
diag = scolars_import_excel_file(
|
|
|
|
csvfile,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
check_homonyms=check_homonyms,
|
|
|
|
require_ine=require_ine,
|
|
|
|
exclude_cols=["photo_filename"],
|
|
|
|
)
|
2021-09-27 10:20:10 +02:00
|
|
|
if return_html:
|
2020-09-26 16:19:37 +02:00
|
|
|
if formsemestre_id:
|
2021-08-15 21:33:47 +02:00
|
|
|
dest = url_for(
|
|
|
|
"notes.formsemestre_status",
|
|
|
|
scodoc_dept=g.scodoc_dept,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
)
|
2020-09-26 16:19:37 +02:00
|
|
|
else:
|
2021-08-15 21:33:47 +02:00
|
|
|
dest = url_for("notes.index_html", scodoc_dept=g.scodoc_dept)
|
2021-07-29 17:31:15 +03:00
|
|
|
H = [html_sco_header.sco_header(page_title="Import etudiants")]
|
2020-09-26 16:19:37 +02:00
|
|
|
H.append("<ul>")
|
|
|
|
for d in diag:
|
|
|
|
H.append("<li>%s</li>" % d)
|
|
|
|
H.append("</ul>")
|
|
|
|
H.append("<p>Import terminé !</p>")
|
|
|
|
H.append('<p><a class="stdlink" href="%s">Continuer</a></p>' % dest)
|
2021-07-29 11:19:00 +03:00
|
|
|
return "\n".join(H) + html_sco_header.sco_footer()
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
|
|
|
|
def scolars_import_excel_file(
|
2021-09-29 20:08:18 +02:00
|
|
|
datafile: io.BytesIO,
|
2020-09-26 16:19:37 +02:00
|
|
|
formsemestre_id=None,
|
|
|
|
check_homonyms=True,
|
|
|
|
require_ine=False,
|
|
|
|
exclude_cols=[],
|
|
|
|
):
|
|
|
|
"""Importe etudiants depuis fichier Excel
|
|
|
|
et les inscrit dans le semestre indiqué (et à TOUS ses modules)
|
|
|
|
"""
|
|
|
|
log("scolars_import_excel_file: formsemestre_id=%s" % formsemestre_id)
|
2021-11-12 22:17:46 +01: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
|
|
|
annee_courante = time.localtime()[0]
|
2021-07-28 18:03:54 +03:00
|
|
|
always_require_ine = sco_preferences.get_preference("always_require_ine")
|
2020-09-26 16:19:37 +02:00
|
|
|
exceldata = datafile.read()
|
|
|
|
if not exceldata:
|
|
|
|
raise ScoValueError("Ficher excel vide ou invalide")
|
2021-08-14 10:12:40 +02:00
|
|
|
diag, data = sco_excel.excel_bytes_to_list(exceldata)
|
2020-09-26 16:19:37 +02:00
|
|
|
if not data: # probably a bug
|
|
|
|
raise ScoException("scolars_import_excel_file: empty file !")
|
|
|
|
|
2020-12-02 01:00:23 +01:00
|
|
|
formsemestre_to_invalidate = set()
|
2020-09-26 16:19:37 +02:00
|
|
|
# 1- --- check title line
|
|
|
|
titles = {}
|
|
|
|
fmt = sco_import_format()
|
|
|
|
for l in fmt:
|
2021-08-21 00:24:51 +02:00
|
|
|
tit = l[0].lower().split()[0] # titles in lowercase, and take 1st word
|
2020-09-26 16:19:37 +02:00
|
|
|
if (
|
|
|
|
(not formsemestre_id) or (tit != "codesemestre")
|
|
|
|
) and tit not in exclude_cols:
|
|
|
|
titles[tit] = l[1:] # title : (Type, Table, AllowNulls, Description)
|
|
|
|
|
|
|
|
# log("titles=%s" % titles)
|
|
|
|
# remove quotes, downcase and keep only 1st word
|
|
|
|
try:
|
2021-08-21 00:24:51 +02:00
|
|
|
fs = [scu.stripquotes(s).lower().split()[0] for s in data[0]]
|
2020-09-26 16:19:37 +02:00
|
|
|
except:
|
|
|
|
raise ScoValueError("Titres de colonnes invalides (ou vides ?)")
|
|
|
|
# log("excel: fs='%s'\ndata=%s" % (str(fs), str(data)))
|
|
|
|
|
|
|
|
# check columns titles
|
|
|
|
if len(fs) != len(titles):
|
2021-07-09 17:47:06 +02:00
|
|
|
missing = {}.fromkeys(list(titles.keys()))
|
2020-09-26 16:19:37 +02:00
|
|
|
unknown = []
|
|
|
|
for f in fs:
|
2021-07-09 17:47:06 +02:00
|
|
|
if f in missing:
|
2020-09-26 16:19:37 +02:00
|
|
|
del missing[f]
|
|
|
|
else:
|
|
|
|
unknown.append(f)
|
|
|
|
raise ScoValueError(
|
|
|
|
"Nombre de colonnes incorrect (devrait être %d, et non %d) <br/> (colonnes manquantes: %s, colonnes invalides: %s)"
|
2021-07-09 17:47:06 +02:00
|
|
|
% (len(titles), len(fs), list(missing.keys()), unknown)
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
titleslist = []
|
|
|
|
for t in fs:
|
2021-07-09 17:47:06 +02:00
|
|
|
if t not in titles:
|
2020-09-26 16:19:37 +02:00
|
|
|
raise ScoValueError('Colonne invalide: "%s"' % t)
|
|
|
|
titleslist.append(t) #
|
|
|
|
# ok, same titles
|
|
|
|
# Start inserting data, abort whole transaction in case of error
|
|
|
|
created_etudids = []
|
|
|
|
NbImportedHomonyms = 0
|
|
|
|
GroupIdInferers = {}
|
|
|
|
try: # --- begin DB transaction
|
|
|
|
linenum = 0
|
|
|
|
for line in data[1:]:
|
|
|
|
linenum += 1
|
|
|
|
# Read fields, check and convert type
|
|
|
|
values = {}
|
|
|
|
fs = line
|
|
|
|
# remove quotes
|
|
|
|
for i in range(len(fs)):
|
|
|
|
if fs[i] and (
|
|
|
|
(fs[i][0] == '"' and fs[i][-1] == '"')
|
|
|
|
or (fs[i][0] == "'" and fs[i][-1] == "'")
|
|
|
|
):
|
|
|
|
fs[i] = fs[i][1:-1]
|
|
|
|
for i in range(len(fs)):
|
|
|
|
val = fs[i].strip()
|
|
|
|
typ, table, an, descr, aliases = tuple(titles[titleslist[i]])
|
|
|
|
# log('field %s: %s %s %s %s'%(titleslist[i], table, typ, an, descr))
|
|
|
|
if not val and not an:
|
|
|
|
raise ScoValueError(
|
|
|
|
"line %d: null value not allowed in column %s"
|
|
|
|
% (linenum, titleslist[i])
|
|
|
|
)
|
|
|
|
if val == "":
|
|
|
|
val = None
|
|
|
|
else:
|
|
|
|
if typ == "real":
|
|
|
|
val = val.replace(",", ".") # si virgule a la française
|
|
|
|
try:
|
|
|
|
val = float(val)
|
|
|
|
except:
|
|
|
|
raise ScoValueError(
|
|
|
|
"valeur nombre reel invalide (%s) sur line %d, colonne %s"
|
|
|
|
% (val, linenum, titleslist[i])
|
|
|
|
)
|
|
|
|
elif typ == "integer":
|
|
|
|
try:
|
|
|
|
# on doit accepter des valeurs comme "2006.0"
|
|
|
|
val = val.replace(",", ".") # si virgule a la française
|
|
|
|
val = float(val)
|
|
|
|
if val % 1.0 > 1e-4:
|
|
|
|
raise ValueError()
|
|
|
|
val = int(val)
|
|
|
|
except:
|
|
|
|
raise ScoValueError(
|
|
|
|
"valeur nombre entier invalide (%s) sur ligne %d, colonne %s"
|
|
|
|
% (val, linenum, titleslist[i])
|
|
|
|
)
|
|
|
|
# xxx Ad-hoc checks (should be in format description)
|
2021-08-21 00:24:51 +02:00
|
|
|
if titleslist[i].lower() == "sexe":
|
2020-09-26 16:19:37 +02:00
|
|
|
try:
|
2021-06-19 23:21:37 +02:00
|
|
|
val = sco_etud.input_civilite(val)
|
2020-09-26 16:19:37 +02:00
|
|
|
except:
|
|
|
|
raise ScoValueError(
|
2021-02-13 17:28:55 +01:00
|
|
|
"valeur invalide pour 'SEXE' (doit etre 'M', 'F', ou 'MME', 'H', 'X' ou vide, mais pas '%s') ligne %d, colonne %s"
|
2020-09-26 16:19:37 +02:00
|
|
|
% (val, linenum, titleslist[i])
|
|
|
|
)
|
|
|
|
# Excel date conversion:
|
2021-08-21 00:24:51 +02:00
|
|
|
if titleslist[i].lower() == "date_naissance":
|
2020-09-26 16:19:37 +02:00
|
|
|
if val:
|
2021-09-08 00:10:36 +02:00
|
|
|
try:
|
|
|
|
val = sco_excel.xldate_as_datetime(val)
|
|
|
|
except ValueError:
|
|
|
|
raise ScoValueError(
|
|
|
|
f"date invalide ({val}) sur ligne {linenum}, colonne {titleslist[i]}"
|
|
|
|
)
|
2020-09-26 16:19:37 +02:00
|
|
|
# INE
|
|
|
|
if (
|
2021-08-21 00:24:51 +02:00
|
|
|
titleslist[i].lower() == "code_ine"
|
2020-09-26 16:19:37 +02:00
|
|
|
and always_require_ine
|
|
|
|
and not val
|
|
|
|
):
|
|
|
|
raise ScoValueError(
|
|
|
|
"Code INE manquant sur ligne %d, colonne %s"
|
|
|
|
% (linenum, titleslist[i])
|
|
|
|
)
|
|
|
|
|
|
|
|
# --
|
|
|
|
values[titleslist[i]] = val
|
|
|
|
skip = False
|
|
|
|
is_new_ine = values["code_ine"] and _is_new_ine(cnx, values["code_ine"])
|
|
|
|
if require_ine and not is_new_ine:
|
|
|
|
log("skipping %s (code_ine=%s)" % (values["nom"], values["code_ine"]))
|
|
|
|
skip = True
|
|
|
|
|
|
|
|
if not skip:
|
|
|
|
if values["code_ine"] and not is_new_ine:
|
|
|
|
raise ScoValueError("Code INE dupliqué (%s)" % values["code_ine"])
|
|
|
|
# Check nom/prenom
|
2021-06-19 23:21:37 +02:00
|
|
|
ok, NbHomonyms = sco_etud.check_nom_prenom(
|
2020-09-26 16:19:37 +02:00
|
|
|
cnx, nom=values["nom"], prenom=values["prenom"]
|
|
|
|
)
|
|
|
|
if not ok:
|
|
|
|
raise ScoValueError(
|
|
|
|
"nom ou prénom invalide sur la ligne %d" % (linenum)
|
|
|
|
)
|
|
|
|
if NbHomonyms:
|
|
|
|
NbImportedHomonyms += 1
|
|
|
|
# Insert in DB tables
|
2021-09-29 20:08:18 +02:00
|
|
|
formsemestre_id_etud = _import_one_student(
|
|
|
|
cnx,
|
|
|
|
formsemestre_id,
|
|
|
|
values,
|
|
|
|
GroupIdInferers,
|
|
|
|
annee_courante,
|
|
|
|
created_etudids,
|
|
|
|
linenum,
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
# Verification proportion d'homonymes: si > 10%, abandonne
|
|
|
|
log("scolars_import_excel_file: detected %d homonyms" % NbImportedHomonyms)
|
|
|
|
if check_homonyms and NbImportedHomonyms > len(created_etudids) / 10:
|
|
|
|
log("scolars_import_excel_file: too many homonyms")
|
|
|
|
raise ScoValueError(
|
|
|
|
"Il y a trop d'homonymes (%d étudiants)" % NbImportedHomonyms
|
|
|
|
)
|
|
|
|
except:
|
|
|
|
cnx.rollback()
|
|
|
|
log("scolars_import_excel_file: aborting transaction !")
|
|
|
|
# Nota: db transaction is sometimes partly commited...
|
|
|
|
# here we try to remove all created students
|
2021-02-03 22:00:41 +01:00
|
|
|
cursor = cnx.cursor(cursor_factory=ndb.ScoDocCursor)
|
2020-09-26 16:19:37 +02:00
|
|
|
for etudid in created_etudids:
|
|
|
|
log("scolars_import_excel_file: deleting etudid=%s" % etudid)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from notes_moduleimpl_inscription where etudid=%(etudid)s",
|
|
|
|
{"etudid": etudid},
|
|
|
|
)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from notes_formsemestre_inscription where etudid=%(etudid)s",
|
|
|
|
{"etudid": etudid},
|
|
|
|
)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from scolar_events where etudid=%(etudid)s", {"etudid": etudid}
|
|
|
|
)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from adresse where etudid=%(etudid)s", {"etudid": etudid}
|
|
|
|
)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from admissions where etudid=%(etudid)s", {"etudid": etudid}
|
|
|
|
)
|
|
|
|
cursor.execute(
|
|
|
|
"delete from group_membership where etudid=%(etudid)s",
|
|
|
|
{"etudid": etudid},
|
|
|
|
)
|
|
|
|
cursor.execute(
|
2021-08-15 21:33:47 +02:00
|
|
|
"delete from identite where id=%(etudid)s", {"etudid": etudid}
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
cnx.commit()
|
|
|
|
log("scolars_import_excel_file: re-raising exception")
|
|
|
|
raise
|
|
|
|
|
|
|
|
diag.append("Import et inscription de %s étudiants" % len(created_etudids))
|
|
|
|
|
|
|
|
sco_news.add(
|
2021-06-17 00:08:37 +02:00
|
|
|
typ=sco_news.NEWS_INSCR,
|
2020-09-26 16:19:37 +02:00
|
|
|
text="Inscription de %d étudiants" # peuvent avoir ete inscrits a des semestres differents
|
|
|
|
% len(created_etudids),
|
|
|
|
object=formsemestre_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
log("scolars_import_excel_file: completing transaction")
|
|
|
|
cnx.commit()
|
|
|
|
|
|
|
|
# Invalide les caches des semestres dans lesquels on a inscrit des etudiants:
|
2021-07-19 20:53:01 +03:00
|
|
|
for formsemestre_id in formsemestre_to_invalidate:
|
|
|
|
sco_cache.invalidate_formsemestre(formsemestre_id=formsemestre_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
return diag
|
|
|
|
|
|
|
|
|
2021-06-21 12:13:25 +02:00
|
|
|
def students_import_admission(
|
2021-09-27 10:20:10 +02:00
|
|
|
csvfile, type_admission="", formsemestre_id=None, return_html=True
|
2021-06-21 12:13:25 +02:00
|
|
|
):
|
|
|
|
"import donnees admission from Excel file (v2016)"
|
|
|
|
diag = scolars_import_admission(
|
|
|
|
csvfile,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
type_admission=type_admission,
|
|
|
|
)
|
2021-09-27 10:20:10 +02:00
|
|
|
if return_html:
|
2021-07-29 17:31:15 +03:00
|
|
|
H = [html_sco_header.sco_header(page_title="Import données admissions")]
|
2021-06-21 12:13:25 +02:00
|
|
|
H.append("<p>Import terminé !</p>")
|
|
|
|
H.append(
|
|
|
|
'<p><a class="stdlink" href="%s">Continuer</a></p>'
|
2021-08-15 21:33:47 +02:00
|
|
|
% url_for(
|
|
|
|
"notes.formsemestre_status",
|
|
|
|
scodoc_dept=g.scodoc_dept,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
)
|
2021-06-21 12:13:25 +02:00
|
|
|
)
|
|
|
|
if diag:
|
|
|
|
H.append("<p>Diagnostic: <ul><li>%s</li></ul></p>" % "</li><li>".join(diag))
|
|
|
|
|
2021-07-29 11:19:00 +03:00
|
|
|
return "\n".join(H) + html_sco_header.sco_footer()
|
2021-06-21 12:13:25 +02:00
|
|
|
|
|
|
|
|
2020-09-26 16:19:37 +02:00
|
|
|
def _import_one_student(
|
|
|
|
cnx,
|
|
|
|
formsemestre_id,
|
|
|
|
values,
|
|
|
|
GroupIdInferers,
|
|
|
|
annee_courante,
|
|
|
|
created_etudids,
|
|
|
|
linenum,
|
2021-09-29 20:08:18 +02:00
|
|
|
) -> int:
|
2020-09-26 16:19:37 +02:00
|
|
|
"""
|
|
|
|
Import d'un étudiant et inscription dans le semestre.
|
|
|
|
Return: id du semestre dans lequel il a été inscrit.
|
|
|
|
"""
|
|
|
|
log(
|
|
|
|
"scolars_import_excel_file: formsemestre_id=%s values=%s"
|
|
|
|
% (formsemestre_id, str(values))
|
|
|
|
)
|
|
|
|
# Identite
|
|
|
|
args = values.copy()
|
2021-09-27 10:20:10 +02:00
|
|
|
etudid = sco_etud.identite_create(cnx, args)
|
2020-09-26 16:19:37 +02:00
|
|
|
created_etudids.append(etudid)
|
|
|
|
# Admissions
|
|
|
|
args["etudid"] = etudid
|
|
|
|
args["annee"] = annee_courante
|
2021-06-19 23:21:37 +02:00
|
|
|
_ = sco_etud.admission_create(cnx, args)
|
2020-09-26 16:19:37 +02:00
|
|
|
# Adresse
|
|
|
|
args["typeadresse"] = "domicile"
|
|
|
|
args["description"] = "(infos admission)"
|
2021-06-19 23:21:37 +02:00
|
|
|
_ = sco_etud.adresse_create(cnx, args)
|
2020-09-26 16:19:37 +02:00
|
|
|
# Inscription au semestre
|
|
|
|
args["etat"] = "I" # etat insc. semestre
|
|
|
|
if formsemestre_id:
|
|
|
|
args["formsemestre_id"] = formsemestre_id
|
|
|
|
else:
|
|
|
|
args["formsemestre_id"] = values["codesemestre"]
|
|
|
|
formsemestre_id = values["codesemestre"]
|
2021-10-07 22:24:43 +02:00
|
|
|
try:
|
|
|
|
formsemestre_id = int(formsemestre_id)
|
|
|
|
except ValueError as exc:
|
|
|
|
raise ScoValueError(
|
|
|
|
f"valeur invalide dans la colonne codesemestre, ligne {linenum+1}"
|
|
|
|
) from exc
|
2020-09-26 16:19:37 +02:00
|
|
|
# recupere liste des groupes:
|
|
|
|
if formsemestre_id not in GroupIdInferers:
|
2021-08-19 10:28:35 +02:00
|
|
|
GroupIdInferers[formsemestre_id] = sco_groups.GroupIdInferer(formsemestre_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
gi = GroupIdInferers[formsemestre_id]
|
|
|
|
if args["groupes"]:
|
|
|
|
groupes = args["groupes"].split(";")
|
|
|
|
else:
|
|
|
|
groupes = []
|
|
|
|
group_ids = [gi[group_name] for group_name in groupes]
|
2021-07-09 17:47:06 +02:00
|
|
|
group_ids = list({}.fromkeys(group_ids).keys()) # uniq
|
2020-09-26 16:19:37 +02:00
|
|
|
if None in group_ids:
|
|
|
|
raise ScoValueError(
|
|
|
|
"groupe invalide sur la ligne %d (groupe %s)" % (linenum, groupes)
|
|
|
|
)
|
|
|
|
|
|
|
|
do_formsemestre_inscription_with_modules(
|
2021-09-29 20:08:18 +02:00
|
|
|
int(args["formsemestre_id"]),
|
2020-09-26 16:19:37 +02:00
|
|
|
etudid,
|
|
|
|
group_ids,
|
|
|
|
etat="I",
|
|
|
|
method="import_csv_file",
|
|
|
|
)
|
|
|
|
return args["formsemestre_id"]
|
|
|
|
|
|
|
|
|
|
|
|
def _is_new_ine(cnx, code_ine):
|
|
|
|
"True if this code is not in DB"
|
2021-06-19 23:21:37 +02:00
|
|
|
etuds = sco_etud.identite_list(cnx, {"code_ine": code_ine})
|
2020-09-26 16:19:37 +02:00
|
|
|
return not etuds
|
|
|
|
|
|
|
|
|
|
|
|
# ------ Fonction ré-écrite en nov 2016 pour lire des fichiers sans etudid (fichiers APB)
|
2021-09-27 10:20:10 +02:00
|
|
|
def scolars_import_admission(datafile, formsemestre_id=None, type_admission=None):
|
2020-09-26 16:19:37 +02:00
|
|
|
"""Importe données admission depuis un fichier Excel quelconque
|
|
|
|
par exemple ceux utilisés avec APB
|
|
|
|
|
2020-12-02 01:00:23 +01:00
|
|
|
Cherche dans ce fichier les étudiants qui correspondent à des inscrits du
|
2020-09-26 16:19:37 +02:00
|
|
|
semestre formsemestre_id.
|
2020-12-02 01:00:23 +01:00
|
|
|
Le fichier n'a pas l'INE ni le NIP ni l'etudid, la correspondance se fait
|
|
|
|
via les noms/prénoms qui doivent être égaux (la casse, les accents et caractères spéciaux
|
2020-09-26 16:19:37 +02:00
|
|
|
étant ignorés).
|
|
|
|
|
2020-12-02 01:00:23 +01:00
|
|
|
On tolère plusieurs variantes pour chaque nom de colonne (ici aussi, la casse, les espaces
|
2020-09-26 16:19:37 +02:00
|
|
|
et les caractères spéciaux sont ignorés. Ainsi, la colonne "Prénom:" sera considéré comme "prenom".
|
|
|
|
|
|
|
|
Le parametre type_admission remplace les valeurs vides (dans la base ET dans le fichier importé) du champ type_admission.
|
|
|
|
Si une valeur existe ou est présente dans le fichier importé, ce paramètre est ignoré.
|
2020-12-02 01:00:23 +01:00
|
|
|
|
2020-09-26 16:19:37 +02:00
|
|
|
TODO:
|
|
|
|
- choix onglet du classeur
|
|
|
|
"""
|
|
|
|
|
|
|
|
log("scolars_import_admission: formsemestre_id=%s" % formsemestre_id)
|
|
|
|
members = sco_groups.get_group_members(
|
2021-08-19 10:28:35 +02:00
|
|
|
sco_groups.get_default_group(formsemestre_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
etuds_by_nomprenom = {} # { nomprenom : etud }
|
|
|
|
diag = []
|
|
|
|
for m in members:
|
|
|
|
np = (adm_normalize_string(m["nom"]), adm_normalize_string(m["prenom"]))
|
|
|
|
if np in etuds_by_nomprenom:
|
|
|
|
msg = "Attention: hononymie pour %s %s" % (m["nom"], m["prenom"])
|
|
|
|
log(msg)
|
|
|
|
diag.append(msg)
|
|
|
|
etuds_by_nomprenom[np] = m
|
|
|
|
|
|
|
|
exceldata = datafile.read()
|
2021-08-14 10:12:40 +02:00
|
|
|
diag2, data = sco_excel.excel_bytes_to_list(exceldata)
|
2020-09-26 16:19:37 +02:00
|
|
|
if not data:
|
|
|
|
raise ScoException("scolars_import_admission: empty file !")
|
|
|
|
diag += diag2
|
2021-06-15 13:59:56 +02:00
|
|
|
cnx = ndb.GetDBConnexion()
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
titles = data[0]
|
|
|
|
# idx -> ('field', convertor)
|
|
|
|
fields = adm_get_fields(titles, formsemestre_id)
|
|
|
|
idx_nom = None
|
|
|
|
idx_prenom = None
|
|
|
|
for idx in fields:
|
|
|
|
if fields[idx][0] == "nom":
|
|
|
|
idx_nom = idx
|
|
|
|
if fields[idx][0] == "prenom":
|
|
|
|
idx_prenom = idx
|
|
|
|
if (idx_nom is None) or (idx_prenom is None):
|
|
|
|
log("fields indices=" + ", ".join([str(x) for x in fields]))
|
|
|
|
log("fields titles =" + ", ".join([fields[x][0] for x in fields]))
|
|
|
|
raise FormatError(
|
|
|
|
"scolars_import_admission: colonnes nom et prenom requises",
|
2021-08-15 21:33:47 +02:00
|
|
|
dest_url=url_for(
|
2021-11-06 17:58:11 +01:00
|
|
|
"scolar.form_students_import_infos_admissions",
|
2021-08-15 21:33:47 +02:00
|
|
|
scodoc_dept=g.scodoc_dept,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
),
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
|
2020-12-02 01:00:23 +01:00
|
|
|
modifiable_fields = set(ADMISSION_MODIFIABLE_FIELDS)
|
2020-09-26 16:19:37 +02:00
|
|
|
|
|
|
|
nline = 2 # la premiere ligne de donnees du fichier excel est 2
|
|
|
|
n_import = 0
|
|
|
|
for line in data[1:]:
|
|
|
|
# Retrouve l'étudiant parmi ceux du semestre par (nom, prenom)
|
|
|
|
nom = adm_normalize_string(line[idx_nom])
|
|
|
|
prenom = adm_normalize_string(line[idx_prenom])
|
|
|
|
if not (nom, prenom) in etuds_by_nomprenom:
|
|
|
|
log(
|
|
|
|
"unable to find %s %s among members" % (line[idx_nom], line[idx_prenom])
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
etud = etuds_by_nomprenom[(nom, prenom)]
|
2021-06-19 23:21:37 +02:00
|
|
|
cur_adm = sco_etud.admission_list(cnx, args={"etudid": etud["etudid"]})[0]
|
2020-09-26 16:19:37 +02:00
|
|
|
# peuple les champs presents dans le tableau
|
|
|
|
args = {}
|
|
|
|
for idx in fields:
|
|
|
|
field_name, convertor = fields[idx]
|
|
|
|
if field_name in modifiable_fields:
|
|
|
|
try:
|
|
|
|
val = convertor(line[idx])
|
|
|
|
except ValueError:
|
|
|
|
raise FormatError(
|
|
|
|
'scolars_import_admission: valeur invalide, ligne %d colonne %s: "%s"'
|
|
|
|
% (nline, field_name, line[idx]),
|
2021-08-15 21:33:47 +02:00
|
|
|
dest_url=url_for(
|
2021-11-06 17:58:11 +01:00
|
|
|
"scolar.form_students_import_infos_admissions",
|
2021-08-15 21:33:47 +02:00
|
|
|
scodoc_dept=g.scodoc_dept,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
),
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
if val is not None: # note: ne peut jamais supprimer une valeur
|
|
|
|
args[field_name] = val
|
|
|
|
if args:
|
|
|
|
args["etudid"] = etud["etudid"]
|
|
|
|
args["adm_id"] = cur_adm["adm_id"]
|
|
|
|
# Type admission: traitement particulier
|
|
|
|
if not cur_adm["type_admission"] and not args.get("type_admission"):
|
|
|
|
args["type_admission"] = type_admission
|
2021-08-21 00:24:51 +02:00
|
|
|
sco_etud.etudident_edit(cnx, args, disable_notify=True)
|
2021-06-19 23:21:37 +02:00
|
|
|
adr = sco_etud.adresse_list(cnx, args={"etudid": etud["etudid"]})
|
2020-09-26 16:19:37 +02:00
|
|
|
if adr:
|
|
|
|
args["adresse_id"] = adr[0]["adresse_id"]
|
2021-06-19 23:21:37 +02:00
|
|
|
sco_etud.adresse_edit(
|
2021-08-21 00:24:51 +02:00
|
|
|
cnx, args, disable_notify=True
|
|
|
|
) # pas de notification ici
|
2020-09-26 16:19:37 +02:00
|
|
|
else:
|
|
|
|
args["typeadresse"] = "domicile"
|
|
|
|
args["description"] = "(infos admission)"
|
2021-06-19 23:21:37 +02:00
|
|
|
adresse_id = sco_etud.adresse_create(cnx, args)
|
2020-09-26 16:19:37 +02:00
|
|
|
# log('import_adm: %s' % args )
|
|
|
|
# Change les groupes si nécessaire:
|
|
|
|
if args["groupes"]:
|
2021-08-19 10:28:35 +02:00
|
|
|
gi = sco_groups.GroupIdInferer(formsemestre_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
groupes = args["groupes"].split(";")
|
|
|
|
group_ids = [gi[group_name] for group_name in groupes]
|
2021-07-09 17:47:06 +02:00
|
|
|
group_ids = list({}.fromkeys(group_ids).keys()) # uniq
|
2020-09-26 16:19:37 +02:00
|
|
|
if None in group_ids:
|
|
|
|
raise ScoValueError(
|
|
|
|
"groupe invalide sur la ligne %d (groupe %s)"
|
|
|
|
% (nline, groupes)
|
|
|
|
)
|
|
|
|
|
|
|
|
for group_id in group_ids:
|
|
|
|
sco_groups.change_etud_group_in_partition(
|
2021-09-02 18:05:22 +02:00
|
|
|
args["etudid"], group_id
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
#
|
|
|
|
diag.append("import de %s" % (etud["nomprenom"]))
|
|
|
|
n_import += 1
|
|
|
|
nline += 1
|
|
|
|
diag.append("%d lignes importées" % n_import)
|
|
|
|
if n_import > 0:
|
2021-07-19 20:53:01 +03:00
|
|
|
sco_cache.invalidate_formsemestre(formsemestre_id=formsemestre_id)
|
2020-09-26 16:19:37 +02:00
|
|
|
return diag
|
|
|
|
|
|
|
|
|
|
|
|
_ADM_PATTERN = re.compile(r"[\W]+", re.UNICODE) # supprime tout sauf alphanum
|
|
|
|
|
|
|
|
|
|
|
|
def adm_normalize_string(s): # normalize unicode title
|
2021-07-12 11:54:04 +02:00
|
|
|
return scu.suppress_accents(_ADM_PATTERN.sub("", s.strip().lower())).replace(
|
2020-09-26 16:19:37 +02:00
|
|
|
"_", ""
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def adm_get_fields(titles, formsemestre_id):
|
|
|
|
"""Cherche les colonnes importables dans les titres (ligne 1) du fichier excel
|
|
|
|
return: { idx : (field_name, convertor) }
|
|
|
|
"""
|
|
|
|
# log('adm_get_fields: titles=%s' % titles)
|
|
|
|
Fmt = sco_import_format_dict()
|
|
|
|
fields = {}
|
|
|
|
idx = 0
|
|
|
|
for title in titles:
|
|
|
|
title_n = adm_normalize_string(title)
|
|
|
|
for k in Fmt:
|
|
|
|
for v in Fmt[k]["aliases"]:
|
|
|
|
if adm_normalize_string(v) == title_n:
|
|
|
|
typ = Fmt[k]["type"]
|
|
|
|
if typ == "real":
|
|
|
|
convertor = adm_convert_real
|
|
|
|
elif typ == "integer" or typ == "int":
|
|
|
|
convertor = adm_convert_int
|
|
|
|
else:
|
|
|
|
convertor = adm_convert_text
|
|
|
|
# doublons ?
|
|
|
|
if k in [x[0] for x in fields.values()]:
|
|
|
|
raise FormatError(
|
|
|
|
'scolars_import_admission: titre "%s" en double (ligne 1)'
|
|
|
|
% (title),
|
2021-08-15 21:33:47 +02:00
|
|
|
dest_url=url_for(
|
2021-11-06 17:58:11 +01:00
|
|
|
"scolar.form_students_import_infos_admissions_apb",
|
2021-08-15 21:33:47 +02:00
|
|
|
scodoc_dept=g.scodoc_dept,
|
|
|
|
formsemestre_id=formsemestre_id,
|
|
|
|
),
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
fields[idx] = (k, convertor)
|
|
|
|
idx += 1
|
|
|
|
|
|
|
|
return fields
|
|
|
|
|
|
|
|
|
|
|
|
def adm_convert_text(v):
|
2021-07-11 22:32:01 +02:00
|
|
|
if isinstance(v, float):
|
2020-09-26 16:19:37 +02:00
|
|
|
return "{:g}".format(v) # evite "1.0"
|
|
|
|
return v
|
|
|
|
|
|
|
|
|
|
|
|
def adm_convert_int(v):
|
2021-07-09 17:47:06 +02:00
|
|
|
if type(v) != int and not v:
|
2020-09-26 16:19:37 +02:00
|
|
|
return None
|
|
|
|
return int(float(v)) # accept "10.0"
|
|
|
|
|
|
|
|
|
|
|
|
def adm_convert_real(v):
|
2021-07-09 17:47:06 +02:00
|
|
|
if type(v) != float and not v:
|
2020-09-26 16:19:37 +02:00
|
|
|
return None
|
|
|
|
return float(v)
|
|
|
|
|
|
|
|
|
2021-08-21 00:24:51 +02:00
|
|
|
def adm_table_description_format():
|
2020-12-02 01:00:23 +01:00
|
|
|
"""Table HTML (ou autre format) decrivant les donnees d'admissions importables"""
|
2020-09-26 16:19:37 +02:00
|
|
|
Fmt = sco_import_format_dict(with_codesemestre=False)
|
|
|
|
for k in Fmt:
|
|
|
|
Fmt[k]["attribute"] = k
|
|
|
|
Fmt[k]["aliases_str"] = ", ".join(Fmt[k]["aliases"])
|
|
|
|
if not Fmt[k]["allow_nulls"]:
|
|
|
|
Fmt[k]["required"] = "*"
|
|
|
|
if k in ADMISSION_MODIFIABLE_FIELDS:
|
|
|
|
Fmt[k]["writable"] = "oui"
|
|
|
|
else:
|
|
|
|
Fmt[k]["writable"] = "non"
|
|
|
|
titles = {
|
|
|
|
"attribute": "Attribut",
|
|
|
|
"type": "Type",
|
|
|
|
"required": "Requis",
|
|
|
|
"writable": "Modifiable",
|
|
|
|
"description": "Description",
|
|
|
|
"aliases_str": "Titres (variantes)",
|
|
|
|
}
|
|
|
|
columns_ids = ("attribute", "type", "writable", "description", "aliases_str")
|
|
|
|
|
|
|
|
tab = GenTable(
|
|
|
|
titles=titles,
|
|
|
|
columns_ids=columns_ids,
|
2021-07-09 17:47:06 +02:00
|
|
|
rows=list(Fmt.values()),
|
2020-09-26 16:19:37 +02:00
|
|
|
html_sortable=True,
|
|
|
|
html_class="table_leftalign",
|
2021-07-28 18:03:54 +03:00
|
|
|
preferences=sco_preferences.SemPreferences(),
|
2020-09-26 16:19:37 +02:00
|
|
|
)
|
|
|
|
return tab
|