2021-07-04 12:32:13 +02:00
|
|
|
# -*- mode: python -*-
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
##############################################################################
|
|
|
|
#
|
|
|
|
# ScoDoc
|
|
|
|
#
|
|
|
|
# Copyright (c) 1999 - 2021 Emmanuel Viennet. All rights reserved.
|
|
|
|
#
|
|
|
|
# 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
|
|
|
|
#
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
"""
|
|
|
|
Module main: page d'accueil, avec liste des départements
|
|
|
|
|
|
|
|
Emmanuel Viennet, 2021
|
|
|
|
"""
|
2021-11-07 09:19:27 +01:00
|
|
|
import io
|
|
|
|
|
2021-09-09 16:11:05 +02:00
|
|
|
from app.auth.models import User
|
2021-09-08 23:00:01 +02:00
|
|
|
import os
|
|
|
|
|
2021-07-04 12:32:13 +02:00
|
|
|
import flask
|
2021-09-08 23:00:01 +02:00
|
|
|
from flask import abort, flash, url_for, redirect, render_template, send_file
|
2021-07-28 10:51:18 +03:00
|
|
|
from flask import request
|
2021-09-05 12:30:11 +02:00
|
|
|
from flask.app import Flask
|
2021-09-11 19:05:19 +02:00
|
|
|
import flask_login
|
2021-11-07 09:19:27 +01:00
|
|
|
from flask_login.utils import login_required, current_user
|
2021-09-05 12:30:11 +02:00
|
|
|
from flask_wtf import FlaskForm
|
2021-09-08 23:00:01 +02:00
|
|
|
from flask_wtf.file import FileField, FileAllowed
|
|
|
|
from werkzeug.exceptions import BadRequest, NotFound
|
2021-09-05 12:30:11 +02:00
|
|
|
from wtforms import SelectField, SubmitField
|
2021-09-08 23:00:01 +02:00
|
|
|
from wtforms.fields import IntegerField
|
|
|
|
from wtforms.fields.simple import BooleanField, StringField, TextAreaField
|
|
|
|
from wtforms.validators import ValidationError, DataRequired, Email, EqualTo
|
|
|
|
|
|
|
|
import app
|
|
|
|
from app.models import Departement, Identite
|
2021-11-12 22:17:46 +01:00
|
|
|
from app.models import FormSemestre, FormsemestreInscription
|
2021-09-08 23:00:01 +02:00
|
|
|
from app.models import ScoDocSiteConfig
|
2021-08-21 17:07:44 +02:00
|
|
|
import sco_version
|
2021-09-08 23:00:01 +02:00
|
|
|
from app.scodoc import sco_logos
|
2021-07-28 10:51:18 +03:00
|
|
|
from app.scodoc import sco_find_etud
|
2021-09-08 23:00:01 +02:00
|
|
|
from app.scodoc import sco_utils as scu
|
2021-09-09 16:11:05 +02:00
|
|
|
from app.decorators import (
|
|
|
|
admin_required,
|
|
|
|
scodoc7func,
|
2021-09-11 19:05:19 +02:00
|
|
|
scodoc,
|
2021-09-09 16:11:05 +02:00
|
|
|
permission_required_compat_scodoc7,
|
|
|
|
)
|
2021-10-10 21:03:18 +02:00
|
|
|
from app.scodoc.sco_exceptions import AccessDenied
|
2021-07-04 12:32:13 +02:00
|
|
|
from app.scodoc.sco_permissions import Permission
|
2021-08-13 09:31:49 +02:00
|
|
|
from app.views import scodoc_bp as bp
|
2021-07-04 12:32:13 +02:00
|
|
|
|
2021-11-07 09:19:27 +01:00
|
|
|
from PIL import Image as PILImage
|
|
|
|
|
2021-07-04 12:32:13 +02:00
|
|
|
|
2021-08-17 22:11:35 +02:00
|
|
|
@bp.route("/")
|
2021-07-04 12:32:13 +02:00
|
|
|
@bp.route("/ScoDoc")
|
|
|
|
@bp.route("/ScoDoc/index")
|
2021-08-13 09:31:49 +02:00
|
|
|
def index():
|
|
|
|
"Page d'accueil: liste des départements"
|
2021-09-04 11:41:23 +02:00
|
|
|
depts = (
|
|
|
|
Departement.query.filter_by(visible=True).order_by(Departement.acronym).all()
|
|
|
|
)
|
2021-07-04 12:32:13 +02:00
|
|
|
return render_template(
|
|
|
|
"scodoc.html",
|
2021-08-21 17:07:44 +02:00
|
|
|
title=sco_version.SCONAME,
|
2021-07-04 12:32:13 +02:00
|
|
|
current_app=flask.current_app,
|
2021-08-13 09:31:49 +02:00
|
|
|
depts=depts,
|
2021-07-04 12:32:13 +02:00
|
|
|
Permission=Permission,
|
|
|
|
)
|
2021-07-28 10:51:18 +03:00
|
|
|
|
|
|
|
|
2021-10-04 22:30:57 +02:00
|
|
|
# Renvoie les url /ScoDoc/RT/ vers /ScoDoc/RT/Scolarite
|
|
|
|
@bp.route("/ScoDoc/<scodoc_dept>/")
|
|
|
|
def index_dept(scodoc_dept):
|
|
|
|
return redirect(url_for("scolar.index_html", scodoc_dept=scodoc_dept))
|
|
|
|
|
|
|
|
|
2021-07-28 10:51:18 +03:00
|
|
|
@bp.route("/ScoDoc/table_etud_in_accessible_depts", methods=["POST"])
|
|
|
|
@login_required
|
|
|
|
def table_etud_in_accessible_depts():
|
|
|
|
"""recherche étudiants sur plusieurs départements"""
|
|
|
|
return sco_find_etud.table_etud_in_accessible_depts(expnom=request.form["expnom"])
|
2021-08-29 19:57:32 +02:00
|
|
|
|
|
|
|
|
2021-09-09 16:11:05 +02:00
|
|
|
# Fonction d'API accessible sans aucun authentification
|
2021-09-08 23:00:01 +02:00
|
|
|
@bp.route("/ScoDoc/get_etud_dept")
|
|
|
|
def get_etud_dept():
|
|
|
|
"""Returns the dept acronym (eg "GEII") of an etud (identified by etudid,
|
|
|
|
code_nip ou code_ine in the request).
|
2021-09-09 16:11:05 +02:00
|
|
|
Ancienne API: ramène la chaine brute, texte sans JSON ou XML.
|
2021-09-08 23:00:01 +02:00
|
|
|
"""
|
|
|
|
if "etudid" in request.args:
|
|
|
|
# zero ou une réponse:
|
|
|
|
etuds = [Identite.query.get(request.args["etudid"])]
|
|
|
|
elif "code_nip" in request.args:
|
|
|
|
# il peut y avoir plusieurs réponses si l'étudiant est passé par plusieurs départements
|
|
|
|
etuds = Identite.query.filter_by(code_nip=request.args["code_nip"]).all()
|
|
|
|
elif "code_ine" in request.args:
|
2021-10-25 15:51:11 +02:00
|
|
|
etuds = Identite.query.filter_by(code_ine=request.args["code_ine"]).all()
|
2021-09-08 23:00:01 +02:00
|
|
|
else:
|
|
|
|
raise BadRequest(
|
|
|
|
"missing argument (expected one among: etudid, code_nip or code_ine)"
|
|
|
|
)
|
|
|
|
if not etuds:
|
|
|
|
raise NotFound("student not found")
|
|
|
|
elif len(etuds) == 1:
|
|
|
|
last_etud = etuds[0]
|
|
|
|
else:
|
|
|
|
# inscriptions dans plusieurs departements: cherche la plus recente
|
|
|
|
last_etud = None
|
|
|
|
last_date = None
|
|
|
|
for etud in etuds:
|
2021-11-12 22:17:46 +01:00
|
|
|
inscriptions = FormsemestreInscription.query.filter_by(etudid=etud.id).all()
|
2021-09-08 23:00:01 +02:00
|
|
|
for ins in inscriptions:
|
|
|
|
date_fin = FormSemestre.query.get(ins.formsemestre_id).date_fin
|
|
|
|
if (last_date is None) or date_fin > last_date:
|
|
|
|
last_date = date_fin
|
|
|
|
last_etud = etud
|
|
|
|
if not last_etud:
|
|
|
|
# est présent dans plusieurs semestres mais inscrit dans aucun !
|
|
|
|
# le choix a peu d'importance...
|
|
|
|
last_etud = etuds[-1]
|
|
|
|
|
|
|
|
return Departement.query.get(last_etud.dept_id).acronym
|
|
|
|
|
|
|
|
|
2021-09-11 19:05:19 +02:00
|
|
|
# Bricolage pour le portail IUTV avec ScoDoc 7: (DEPRECATED: NE PAS UTILISER !)
|
|
|
|
@bp.route(
|
|
|
|
"/ScoDoc/search_inscr_etud_by_nip", methods=["GET"]
|
|
|
|
) # pour compat anciens clients PHP
|
|
|
|
@scodoc
|
|
|
|
@scodoc7func
|
2021-09-27 10:20:10 +02:00
|
|
|
def search_inscr_etud_by_nip(code_nip, format="json", __ac_name="", __ac_password=""):
|
2021-09-11 19:05:19 +02:00
|
|
|
auth_ok = False
|
|
|
|
user_name = __ac_name
|
|
|
|
user_password = __ac_password
|
|
|
|
if user_name and user_password:
|
|
|
|
u = User.query.filter_by(user_name=user_name).first()
|
|
|
|
if u and u.check_password(user_password):
|
|
|
|
auth_ok = True
|
|
|
|
flask_login.login_user(u)
|
|
|
|
if not auth_ok:
|
|
|
|
abort(403)
|
|
|
|
else:
|
2021-09-27 10:20:10 +02:00
|
|
|
return sco_find_etud.search_inscr_etud_by_nip(code_nip=code_nip, format=format)
|
2021-09-11 19:05:19 +02:00
|
|
|
|
|
|
|
|
2021-09-11 15:59:06 +02:00
|
|
|
@bp.route("/ScoDoc/about")
|
|
|
|
@bp.route("/ScoDoc/Scolarite/<scodoc_dept>/about")
|
|
|
|
def about(scodoc_dept=None):
|
|
|
|
"version info"
|
|
|
|
return render_template(
|
|
|
|
"about.html",
|
|
|
|
version=scu.get_scodoc_version(),
|
|
|
|
news=sco_version.SCONEWS,
|
|
|
|
logo=scu.icontag("borgne_img"),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-09-05 12:30:11 +02:00
|
|
|
# ---- CONFIGURATION
|
|
|
|
|
|
|
|
|
|
|
|
class ScoDocConfigurationForm(FlaskForm):
|
|
|
|
"Panneau de configuration général"
|
2021-09-08 23:00:01 +02:00
|
|
|
|
2021-09-05 12:30:11 +02:00
|
|
|
bonus_sport_func_name = SelectField(
|
|
|
|
label="Fonction de calcul des bonus sport&culture",
|
|
|
|
choices=[
|
|
|
|
(x, x if x else "Aucune")
|
|
|
|
for x in ScoDocSiteConfig.get_bonus_sport_func_names()
|
|
|
|
],
|
|
|
|
)
|
2021-09-08 23:00:01 +02:00
|
|
|
|
|
|
|
logo_header = FileField(
|
|
|
|
label="Modifier l'image:",
|
|
|
|
description="logo placé en haut des documents PDF",
|
|
|
|
validators=[
|
|
|
|
FileAllowed(
|
|
|
|
scu.LOGOS_IMAGES_ALLOWED_TYPES,
|
|
|
|
f"n'accepte que les fichiers image <tt>{','.join([e for e in scu.LOGOS_IMAGES_ALLOWED_TYPES])}</tt>",
|
|
|
|
)
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
|
|
|
logo_footer = FileField(
|
|
|
|
label="Modifier l'image:",
|
|
|
|
description="logo placé en pied des documents PDF",
|
|
|
|
validators=[
|
|
|
|
FileAllowed(
|
|
|
|
scu.LOGOS_IMAGES_ALLOWED_TYPES,
|
|
|
|
f"n'accepte que les fichiers image <tt>{','.join([e for e in scu.LOGOS_IMAGES_ALLOWED_TYPES])}</tt>",
|
|
|
|
)
|
|
|
|
],
|
|
|
|
)
|
|
|
|
|
2021-09-05 12:30:11 +02:00
|
|
|
submit = SubmitField("Enregistrer")
|
|
|
|
|
|
|
|
|
2021-09-08 23:00:01 +02:00
|
|
|
# Notes pour variables config: (valeurs par défaut des paramètres de département)
|
|
|
|
# Chaines simples
|
|
|
|
# SCOLAR_FONT = "Helvetica"
|
|
|
|
# SCOLAR_FONT_SIZE = 10
|
|
|
|
# SCOLAR_FONT_SIZE_FOOT = 6
|
|
|
|
# INSTITUTION_NAME = "<b>Institut Universitaire de Technologie - Université Georges Perec</b>"
|
|
|
|
# INSTITUTION_ADDRESS = "Web <b>www.sor.bonne.top</b> - 11, rue Simon Crubelier - 75017 Paris"
|
|
|
|
# INSTITUTION_CITY = "Paris"
|
|
|
|
# Textareas:
|
|
|
|
# DEFAULT_PDF_FOOTER_TEMPLATE = "Edité par %(scodoc_name)s le %(day)s/%(month)s/%(year)s à %(hour)sh%(minute)s sur %(server_url)s"
|
|
|
|
|
|
|
|
# Booléens
|
|
|
|
# always_require_ine
|
|
|
|
|
|
|
|
# Logos:
|
|
|
|
# LOGO_FOOTER*, LOGO_HEADER*
|
|
|
|
|
|
|
|
|
2021-09-05 12:30:11 +02:00
|
|
|
@bp.route("/ScoDoc/configuration", methods=["GET", "POST"])
|
|
|
|
@admin_required
|
|
|
|
def configuration():
|
|
|
|
"Panneau de configuration général"
|
|
|
|
form = ScoDocConfigurationForm(
|
2021-09-08 23:00:01 +02:00
|
|
|
bonus_sport_func_name=ScoDocSiteConfig.get_bonus_sport_func_name(),
|
2021-09-05 12:30:11 +02:00
|
|
|
)
|
|
|
|
if form.validate_on_submit():
|
|
|
|
ScoDocSiteConfig.set_bonus_sport_func(form.bonus_sport_func_name.data)
|
2021-09-08 23:00:01 +02:00
|
|
|
if form.logo_header.data:
|
2021-11-07 09:19:27 +01:00
|
|
|
sco_logos.write_logo(stream=form.logo_header.data, name="header")
|
2021-09-08 23:00:01 +02:00
|
|
|
if form.logo_footer.data:
|
2021-11-07 09:19:27 +01:00
|
|
|
sco_logos.write_logo(stream=form.logo_footer.data, name="footer")
|
2021-09-08 23:00:01 +02:00
|
|
|
app.clear_scodoc_cache()
|
2021-09-05 12:30:11 +02:00
|
|
|
flash(f"Configuration enregistrée")
|
|
|
|
return redirect(url_for("scodoc.index"))
|
2021-09-08 23:00:01 +02:00
|
|
|
|
2021-09-05 12:30:11 +02:00
|
|
|
return render_template(
|
|
|
|
"configuration.html",
|
|
|
|
title="Configuration ScoDoc",
|
|
|
|
form=form,
|
2021-09-08 23:00:01 +02:00
|
|
|
scodoc_dept=None,
|
2021-09-05 12:30:11 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2021-11-07 09:19:27 +01:00
|
|
|
SMALL_SIZE = (300, 300)
|
|
|
|
|
|
|
|
|
|
|
|
def _return_logo(name="header", dept_id="", small=False, strict: bool = True):
|
2021-09-08 23:00:01 +02:00
|
|
|
# stockée dans /opt/scodoc-data/config/logos donc servie manuellement ici
|
2021-11-07 09:19:27 +01:00
|
|
|
logo = sco_logos.find_logo(name, dept_id, strict)
|
|
|
|
if logo is not None:
|
|
|
|
suffix = logo.suffix
|
|
|
|
if small:
|
|
|
|
with PILImage.open(logo.filepath) as im:
|
|
|
|
im.thumbnail(SMALL_SIZE)
|
|
|
|
stream = io.BytesIO()
|
|
|
|
# on garde le même format (on pourrait plus simplement générer systématiquement du JPEG)
|
|
|
|
fmt = { # adapt suffix to be compliant with PIL save format
|
|
|
|
"PNG": "PNG",
|
|
|
|
"JPG": "JPEG",
|
|
|
|
"JPEG": "JPEG",
|
|
|
|
}[suffix.upper()]
|
|
|
|
im.save(stream, fmt)
|
|
|
|
stream.seek(0)
|
|
|
|
return send_file(stream, mimetype=f"image/{fmt}")
|
|
|
|
else:
|
|
|
|
return send_file(logo.filepath, mimetype=f"image/{suffix}")
|
2021-09-08 23:00:01 +02:00
|
|
|
else:
|
2021-11-07 09:19:27 +01:00
|
|
|
abort(404)
|
|
|
|
|
|
|
|
|
|
|
|
# small version (copy/paste from get_logo
|
|
|
|
@bp.route("/ScoDoc/logos/<name>/small", defaults={"dept_id": None})
|
|
|
|
@bp.route("/ScoDoc/<int:dept_id>/logos/<name>/small")
|
|
|
|
@admin_required
|
|
|
|
def get_logo_small(name: str, dept_id: int):
|
|
|
|
strict = request.args.get("strict", "False")
|
|
|
|
return _return_logo(
|
|
|
|
name,
|
|
|
|
dept_id=dept_id,
|
|
|
|
small=True,
|
|
|
|
strict=strict.upper() not in ["0", "FALSE"],
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route(
|
|
|
|
"/ScoDoc/logos/<name>", defaults={"dept_id": None}
|
|
|
|
) # if dept not specified, take global logo
|
|
|
|
@bp.route("/ScoDoc/<int:dept_id>/logos/<name>")
|
|
|
|
@admin_required
|
|
|
|
def get_logo(name: str, dept_id: int):
|
|
|
|
strict = request.args.get("strict", "False")
|
|
|
|
return _return_logo(
|
|
|
|
name,
|
|
|
|
dept_id=dept_id,
|
|
|
|
small=False,
|
|
|
|
strict=strict.upper() not in ["0", "FALSE"],
|
|
|
|
)
|
2021-09-08 23:00:01 +02:00
|
|
|
|
|
|
|
|
2021-11-07 09:19:27 +01:00
|
|
|
# @bp.route("/ScoDoc/logo_header")
|
|
|
|
# @bp.route("/ScoDoc/<scodoc_dept>/logo_header")
|
|
|
|
# def logo_header(scodoc_dept=""):
|
|
|
|
# "Image logo header"
|
|
|
|
# return _return_logo(name="header", scodoc_dept=scodoc_dept)
|
2021-09-08 23:00:01 +02:00
|
|
|
|
|
|
|
|
2021-11-07 09:19:27 +01:00
|
|
|
# @bp.route("/ScoDoc/logo_footer")
|
|
|
|
# @bp.route("/ScoDoc/<scodoc_dept>/logo_footer")
|
|
|
|
# def logo_footer(scodoc_dept=""):
|
|
|
|
# "Image logo footer"
|
|
|
|
# return _return_logo(name="footer", scodoc_dept=scodoc_dept)
|
2021-09-08 23:00:01 +02:00
|
|
|
|
|
|
|
|
2021-08-29 19:57:32 +02:00
|
|
|
# essais
|
2021-08-29 22:42:38 +02:00
|
|
|
# @bp.route("/testlog")
|
|
|
|
# def testlog():
|
|
|
|
# import time
|
|
|
|
# from flask import current_app
|
|
|
|
# from app import log
|
2021-08-29 19:57:32 +02:00
|
|
|
|
2021-08-29 22:42:38 +02:00
|
|
|
# log(f"testlog called: handlers={current_app.logger.handlers}")
|
|
|
|
# current_app.logger.debug(f"testlog message DEBUG")
|
|
|
|
# current_app.logger.info(f"testlog message INFO")
|
|
|
|
# current_app.logger.warning(f"testlog message WARNING")
|
|
|
|
# current_app.logger.error(f"testlog message ERROR")
|
|
|
|
# current_app.logger.critical(f"testlog message CRITICAL")
|
|
|
|
# raise SyntaxError("une erreur de syntaxe")
|
|
|
|
# return "testlog completed at " + str(time.time())
|