Browse Source

ajout outil fcn-cotisation

root 6 years ago
parent
commit
e6dc1fd559

+ 10 - 0
conf/config.ini.example

@@ -9,3 +9,13 @@ database = sympa
 user = sympa_user
 password = sympa_password
 host = localhost
+
+[reminder]
+reminder0_min = -21
+reminder0_max = +14
+reminder1_min = +30
+reminder1_max = +51
+reminder2_min = +60
+reminder2_max = +81
+reminder3_min = +183
+reminder3_max = +9999

+ 12 - 0
cotisation-reminder0.format

@@ -0,0 +1,12 @@
+Subject: Reconduction de votre adhésion Franciliens.net
+
+
+Cher {firstname},
+
+Ta cotisation à l’association Franciliens.net arrive à échéance le {datefin:%d/%m/%Y}. Si, comme nous l’espérons, tu souhaites continuer à soutenir nos actions, tu peux la renouveler simplement en effectuant un virement de 15 EUR à l’association. Dans le cas contraire, n’hésites pas à nous prévenir :)
+
+Nous espérons pouvoir continuer à te compter parmi nous,
+
+Merci et à bientôt,
+
+Le bureau de Franciliens.net

+ 16 - 0
cotisation-reminder1.format

@@ -0,0 +1,16 @@
+Subject: Reconduction de votre adhésion Franciliens.net (rappel)
+
+
+Bonjour {firstname},
+
+La date anniversaire de ton adhésion à Franciliens.net était le {datefin:%d/%m/%Y}. 
+
+Nous n'avons pas reçu ta cotisation annuelle, mais espèrons que tu souhaite toujours soutenir l'association et ses actions. Nous te rapellons donc qu'il suffit de régler 15 EUR chaque année, de préference par virement sur le compte de l’association. 
+
+Dans le cas contraire, informe-nous, s'il te plaît, de ta décision de mettre fin à l'adhésion.
+
+Nous espérons pouvoir continuer à te compter parmi nous,
+
+Merci et à bientôt,
+
+Le bureau de Franciliens.net

+ 1 - 0
cotisation-reminder2.format

@@ -0,0 +1 @@
+cotisation-reminder1.format

+ 16 - 0
cotisation-reminder3.format

@@ -0,0 +1,16 @@
+Subject: Reconduction de votre adhésion Franciliens.net (dernier rappel)
+
+
+Bonjour {firstname},
+
+Malgré notre relance, nous n'avons pas reçu de nouvelles concernant le renouvellement de ton adhésion.
+
+Nous espérons que tu souhaites renouveller ton adhésion. Dans ce cas veille à régler ta cotisation, et informe nous en s'il te plait. Le montant de la cotisation est 15 EUR chaque année. 
+
+Dans le cas contraire le bureau de l'association pourra procéder à ta radiation à tout moment (cf article 6 des statuts). 
+
+Nous rapellons par ailleurs qu'il est nécéssaire d'être adhérent pour bénéficier des services de l'association (DSL, VPN, ..). Une radiation entraine donc la résiliation des abonnements, le cas échéant à l'issue de la période de préavis (voir page tarifs).
+
+Merci et à bientôt,
+
+Le bureau de Franciliens.net

+ 15 - 0
cotisation-report.format

@@ -0,0 +1,15 @@
+Cher bureau,
+
+Les cotisations des membres suivants vont arriver à échéance. 
+
+{adherents}
+
+Je leur ait envoyé un courriel de ta part, voici un exemple : 
+
+--------------------
+{template}
+--------------------
+
+Automatiquement,
+
+Le script python divin

+ 133 - 0
fcn-cotisation

@@ -0,0 +1,133 @@
+#!/usr/bin/python3
+
+import argparse
+import configparser
+import psycopg2
+import datetime
+import json
+import pprint
+
+# network imports
+import smtplib
+from email.mime.text import MIMEText
+from email.parser import Parser
+
+parser = argparse.ArgumentParser()
+parser.add_argument("selection", type=str,
+                    choices=["send-reminders", "report"])
+parser.add_argument("-c", "--config", type=str,
+                    default="/etc/fcntoolbox/config.ini",
+                    help="specify a configuration file")
+
+args = parser.parse_args()
+
+conf = configparser.RawConfigParser()
+conf.sections()
+conf.read(args.config)
+confdoli = conf['dolibarr']
+remindConf = conf['reminder']
+
+s = smtplib.SMTP('localhost')
+
+conn = psycopg2.connect(database=confdoli['database'], 
+  user=confdoli['user'], password=confdoli['password'])
+
+parsedate = lambda x : datetime.datetime.strptime(x, "%Y-%m-%d %H:%M:%S")
+
+reportAdherents = []
+reportBody = []
+
+def getLateMembers(reminder):
+  minKey = "%s_min" % reminder
+  maxKey = "%s_max" % reminder
+  if (not minKey in remindConf) or (not maxKey in remindConf):
+    print("Configuration missing for '%', skipping reminder" % reminder)
+    return list()
+  
+  min = int(remindConf[minKey])
+  minabs = abs(min)
+  minsign = '+' if min >= 0 else '-'
+  max = int(remindConf[maxKey])
+  maxabs = abs(max)
+  maxsign = '+' if max >= 0 else '-'
+  
+  rows = ["rowid", "firstname", "lastname", "datefin", "email"]
+  query = """SELECT {} FROM llx_adherent
+             WHERE (datefin {} interval '{} days' > 'now' 
+                    AND datefin {} interval '{} days' < 'now')
+             AND statut = 1 """
+  
+  cur = conn.cursor()
+  cur.execute(query.format(",".join(rows), maxsign, maxabs, minsign, minabs))
+  members = cur.fetchall()
+  members = list(map(lambda member: dict(zip(rows, member)), members))
+  return members
+
+def sendReminders(reminder):
+  global reportAdherents
+  global reportBody
+  lateMembers = getLateMembers(reminder)
+
+  # Last reminder state  
+  remindstate = laststate[reminder] if reminder in laststate else []
+  
+  # New reminder state
+  state[reminder] = list(map(lambda x : int(x['rowid']), lateMembers))
+  adherents = list(filter(lambda x: not int(x['rowid']) in remindstate, lateMembers))
+  
+  remindFormat = open('cotisation-%s.format' % reminder, 'r').read()
+  
+  for adherent in adherents:
+    adherent['rowid'] = int(adherent['rowid'])
+#    adherent['datefin'] = parsedate(adherent['datefin'])
+    rawemail = remindFormat.format(**adherent)
+    parsedemail = Parser().parsestr(rawemail)
+    body = parsedemail.get_payload()
+    msg = MIMEText(body)
+    for key, val in parsedemail.items():
+      msg[key] = val
+    msg['From'] = "tresoriers@listes.franciliens.net"
+    msg['To'] = adherent['email']
+    if not "Subject" in msg:
+      print("Error: no subject in template email")
+      break
+    if len(body.strip()) == 0:
+      print("Error: no body in template email")
+      break
+    s.send_message(msg)
+  
+  if len(adherents) > 0:
+    reportAdherents += adherents
+    reportBody += [body]
+
+if args.selection == 'send-reminders':
+  # Read state
+  with open('cotisation-state.json', 'r') as statefp:
+    laststate = json.load(statefp)
+    state = laststate
+  
+  reminders = ['reminder0', 'reminder1', 'reminder2', 'reminder3']
+  
+  for reminder in reminders:
+    sendReminders(reminder)
+  
+  reportFormat = open('cotisation-report.format', 'r').read()  
+  pp = pprint.PrettyPrinter()
+  
+  if len(reportAdherents) > 0:
+    body = reportFormat.format(adherents = pp.pformat(reportAdherents), template = "\n----------\n".join(reportBody))
+    msg = MIMEText(body)
+    msg['Subject'] = "Rapport des rappels de cotisation"
+    msg['From'] = "root@franciliens.net"
+    msg['To'] = "tresoriers@listes.franciliens.net"
+    s.send_message(msg)
+  s.quit()
+
+  # Save state
+  with open('cotisation-state.json', 'w') as statefp:
+    json.dump(state, statefp)
+  
+elif args.selection == 'report':
+  lateMembers = getLateMembers('reminder3')
+  for member in lateMembers:
+    print(member)