DocScoDoc/app/scodoc/sco_news.py

275 lines
8.5 KiB
Python
Raw Normal View History

2020-09-26 16:19:37 +02:00
# -*- mode: python -*-
# -*- coding: utf-8 -*-
##############################################################################
#
# Gestion scolarite IUT
#
2022-01-01 14:49:42 +01:00
# Copyright (c) 1999 - 2022 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-07-09 23:19:30 +02:00
"""Gestion des "nouvelles"
2020-09-26 16:19:37 +02:00
"""
import re
2020-09-26 16:19:37 +02:00
import time
2021-07-11 22:56:22 +02:00
2021-08-26 23:43:54 +02:00
2021-07-09 23:19:30 +02:00
from operator import itemgetter
2020-09-26 16:19:37 +02:00
2021-08-13 11:26:22 +02:00
from flask import g
from flask_login import current_user
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
from app.scodoc import sco_formsemestre
from app.scodoc import sco_moduleimpl
from app.scodoc import sco_preferences
2021-08-26 23:43:54 +02:00
from app import email
2020-09-26 16:19:37 +02:00
2021-02-03 22:00:41 +01:00
_scolar_news_editor = ndb.EditableTable(
2020-09-26 16:19:37 +02:00
"scolar_news",
"news_id",
("date", "authenticated_user", "type", "object", "text", "url"),
2021-08-13 11:26:22 +02:00
filter_dept=True,
2020-09-26 16:19:37 +02:00
sortkey="date desc",
2021-02-03 22:00:41 +01:00
output_formators={"date": ndb.DateISOtoDMY},
input_formators={"date": ndb.DateDMYtoISO},
2020-09-26 16:19:37 +02:00
html_quote=False, # no user supplied data, needed to store html links
)
NEWS_INSCR = "INSCR" # inscription d'étudiants (object=None ou formsemestre_id)
NEWS_NOTE = "NOTES" # saisie note (object=moduleimpl_id)
NEWS_FORM = "FORM" # modification formation (object=formation_id)
NEWS_SEM = "SEM" # creation semestre (object=None)
NEWS_MISC = "MISC" # unused
NEWS_MAP = {
NEWS_INSCR: "inscription d'étudiants",
NEWS_NOTE: "saisie note",
NEWS_FORM: "modification formation",
NEWS_SEM: "création semestre",
NEWS_MISC: "opération", # unused
}
2021-07-09 17:47:06 +02:00
NEWS_TYPES = list(NEWS_MAP.keys())
2020-09-26 16:19:37 +02:00
scolar_news_create = _scolar_news_editor.create
scolar_news_list = _scolar_news_editor.list
_LAST_NEWS = {} # { (authuser_name, type, object) : time }
def add(typ, object=None, text="", url=None, max_frequency=False):
2020-09-26 16:19:37 +02:00
"""Ajoute une nouvelle.
Si max_frequency, ne genere pas 2 nouvelles identiques à moins de max_frequency
secondes d'intervalle.
"""
from app.scodoc import sco_users
authuser_name = current_user.user_name
2021-06-15 13:59:56 +02:00
cnx = ndb.GetDBConnexion()
2020-09-26 16:19:37 +02:00
args = {
"authenticated_user": authuser_name,
2021-08-22 13:24:36 +02:00
"user_info": sco_users.user_info(authuser_name),
2020-09-26 16:19:37 +02:00
"type": typ,
"object": object,
"text": text,
"url": url,
}
t = time.time()
if max_frequency:
last_news_time = _LAST_NEWS.get((authuser_name, typ, object), False)
if last_news_time and (t - last_news_time < max_frequency):
2021-08-01 10:16:16 +02:00
# log("not recording")
2020-09-26 16:19:37 +02:00
return
2021-08-01 10:16:16 +02:00
log("news: %s" % args)
2020-09-26 16:19:37 +02:00
_LAST_NEWS[(authuser_name, typ, object)] = t
_send_news_by_mail(args)
2021-08-01 16:33:09 +02:00
return scolar_news_create(cnx, args)
2020-09-26 16:19:37 +02:00
def scolar_news_summary(n=5):
2020-09-26 16:19:37 +02:00
"""Return last n news.
News are "compressed", ie redondant events are joined.
"""
from app.scodoc import sco_users
2020-09-26 16:19:37 +02:00
2021-06-15 13:59:56 +02:00
cnx = ndb.GetDBConnexion()
2021-02-03 22:00:41 +01:00
cursor = cnx.cursor(cursor_factory=ndb.ScoDocCursor)
2021-08-13 11:26:22 +02:00
cursor.execute(
"""SELECT id AS news_id, *
FROM scolar_news
WHERE dept_id=%(dept_id)s
ORDER BY date DESC LIMIT 100
""",
{"dept_id": g.scodoc_dept_id},
)
2020-09-26 16:19:37 +02:00
selected_news = {} # (type,object) : news dict
news = cursor.dictfetchall() # la plus récente d'abord
for r in reversed(news): # la plus ancienne d'abord
# si on a deja une news avec meme (type,object)
# et du meme jour, on la remplace
2021-02-03 22:00:41 +01:00
dmy = ndb.DateISOtoDMY(r["date"]) # round
2020-09-26 16:19:37 +02:00
key = (r["type"], r["object"], dmy)
selected_news[key] = r
2021-07-09 17:47:06 +02:00
news = list(selected_news.values())
2020-09-26 16:19:37 +02:00
# sort by date, descending
2021-07-09 23:31:16 +02:00
news.sort(key=itemgetter("date"), reverse=True)
2020-09-26 16:19:37 +02:00
news = news[:n]
# mimic EditableTable.list output formatting:
for n in news:
n["date822"] = n["date"].strftime("%a, %d %b %Y %H:%M:%S %z")
# heure
n["hm"] = n["date"].strftime("%Hh%M")
for k in n.keys():
if n[k] is None:
n[k] = ""
2021-07-09 17:47:06 +02:00
if k in _scolar_news_editor.output_formators:
2020-09-26 16:19:37 +02:00
n[k] = _scolar_news_editor.output_formators[k](n[k])
# date resumee
j, m = n["date"].split("/")[:2]
2021-12-04 21:04:09 +01:00
mois = scu.MONTH_NAMES_ABBREV[int(m) - 1]
2022-04-11 08:03:27 +02:00
n["formatted_date"] = f'{j} {mois} {n["hm"]}'
2020-09-26 16:19:37 +02:00
# indication semestre si ajout notes:
infos = _get_formsemestre_infos_from_news(n)
2020-09-26 16:19:37 +02:00
if infos:
n["text"] += (
2022-04-11 08:03:27 +02:00
' (<a class="stdlink" href="Notes/formsemestre_status?formsemestre_id=%(formsemestre_id)s">%(descr_sem)s</a>)'
2020-09-26 16:19:37 +02:00
% infos
)
n["text"] += (
2021-08-22 13:24:36 +02:00
" par " + sco_users.user_info(n["authenticated_user"])["nomcomplet"]
2020-09-26 16:19:37 +02:00
)
return news
def _get_formsemestre_infos_from_news(n):
2020-09-26 16:19:37 +02:00
"""Informations sur le semestre concerné par la nouvelle n
{} si inexistant
"""
formsemestre_id = None
if n["type"] == NEWS_INSCR:
formsemestre_id = n["object"]
elif n["type"] == NEWS_NOTE:
moduleimpl_id = n["object"]
if n["object"]:
mods = sco_moduleimpl.moduleimpl_list(moduleimpl_id=moduleimpl_id)
if not mods:
return {} # module does not exists anymore
2022-04-11 08:03:27 +02:00
mod = mods[0]
formsemestre_id = mod["formsemestre_id"]
2020-09-26 16:19:37 +02:00
if not formsemestre_id:
return {}
try:
2021-08-19 10:28:35 +02:00
sem = sco_formsemestre.get_formsemestre(formsemestre_id)
2022-04-11 08:03:27 +02:00
except ValueError:
2020-09-26 16:19:37 +02:00
# semestre n'existe plus
return {}
if sem["semestre_id"] > 0:
2022-04-11 08:03:27 +02:00
descr_sem = f'S{sem["semestre_id"]}'
2020-09-26 16:19:37 +02:00
else:
descr_sem = ""
if sem["modalite"]:
descr_sem += " " + sem["modalite"]
return {"formsemestre_id": formsemestre_id, "sem": sem, "descr_sem": descr_sem}
def scolar_news_summary_html(n=5):
2020-09-26 16:19:37 +02:00
"""News summary, formated in HTML"""
news = scolar_news_summary(n=n)
2020-09-26 16:19:37 +02:00
if not news:
return ""
H = ['<div class="news"><span class="newstitle">Dernières opérations']
H.append('</span><ul class="newslist">')
for n in news:
H.append(
'<li class="newslist"><span class="newsdate">%(formatted_date)s</span><span class="newstext">%(text)s</span></li>'
% n
)
H.append("</ul>")
# Informations générales
H.append(
"""<div>
Pour être informé des évolutions de ScoDoc,
vous pouvez vous
<a class="stdlink" href="%s">
abonner à la liste de diffusion</a>.
</div>
"""
2021-08-26 23:43:54 +02:00
% scu.SCO_ANNONCES_WEBSITE
2020-09-26 16:19:37 +02:00
)
H.append("</div>")
return "\n".join(H)
def _send_news_by_mail(n):
"""Notify by email"""
infos = _get_formsemestre_infos_from_news(n)
2020-09-26 16:19:37 +02:00
formsemestre_id = infos.get("formsemestre_id", None)
prefs = sco_preferences.SemPreferences(formsemestre_id=formsemestre_id)
2020-09-26 16:19:37 +02:00
destinations = prefs["emails_notifications"] or ""
destinations = [x.strip() for x in destinations.split(",")]
destinations = [x for x in destinations if x]
if not destinations:
return
#
txt = n["text"]
if infos:
2020-12-29 23:49:08 +01:00
txt += "\n\nSemestre %(titremois)s\n\n" % infos["sem"]
2020-09-26 16:19:37 +02:00
txt += (
2020-12-29 23:49:08 +01:00
"""<a href="Notes/formsemestre_status?formsemestre_id=%(formsemestre_id)s">%(descr_sem)s</a>
"""
2020-09-26 16:19:37 +02:00
% infos
)
txt += "\n\nEffectué par: %(nomcomplet)s\n" % n["user_info"]
txt = (
"\n"
+ txt
+ """\n
--- Ceci est un message de notification automatique issu de ScoDoc
--- vous recevez ce message car votre adresse est indiquée dans les paramètres de ScoDoc.
"""
)
# Transforme les URL en URL absolue
2021-06-15 12:34:33 +02:00
base = scu.ScoURL()
2020-09-26 16:19:37 +02:00
txt = re.sub('href=.*?"', 'href="' + base + "/", txt)
# Transforme les liens HTML en texte brut: '<a href="url">texte</a>' devient 'texte: url'
# (si on veut des messages non html)
txt = re.sub(r'<a.*?href\s*=\s*"(.*?)".*?>(.*?)</a>', r"\2: \1", txt)
2021-08-26 23:43:54 +02:00
subject = "[ScoDoc] " + NEWS_MAP.get(n["type"], "?")
sender = prefs["email_from_addr"]
email.send_email(subject, sender, destinations, txt)