fcn-cotisation 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. #!/usr/bin/python3
  2. import argparse
  3. import configparser
  4. import psycopg2
  5. import datetime
  6. import json
  7. import pprint
  8. # network imports
  9. import smtplib
  10. from email.mime.text import MIMEText
  11. from email.parser import Parser
  12. # local imports
  13. import fcntoolbox.dolibarr as dolibarr
  14. parser = argparse.ArgumentParser()
  15. parser.add_argument("selection", type=str,
  16. choices=["send-reminders", "report"])
  17. parser.add_argument("-c", "--config", type=str,
  18. default="/etc/fcntoolbox/config.ini",
  19. help="specify a configuration file")
  20. args = parser.parse_args()
  21. conf = configparser.RawConfigParser()
  22. conf.sections()
  23. conf.read(args.config)
  24. confdoli = conf['dolibarr']
  25. remindConf = conf['reminder']
  26. s = smtplib.SMTP('localhost')
  27. conn = psycopg2.connect(database=confdoli['database'],
  28. user=confdoli['user'], password=confdoli['password'])
  29. doli = dolibarr.Instance(conn)
  30. accounts = list(doli.get_bank_accounts(fields=['iban_prefix', 'bank']))
  31. parsedate = lambda x : datetime.datetime.strptime(x, "%Y-%m-%d %H:%M:%S")
  32. reportAdherents = []
  33. reportBody = []
  34. def getLateMembers(reminder):
  35. minKey = "%s_min" % reminder
  36. maxKey = "%s_max" % reminder
  37. if (not minKey in remindConf) or (not maxKey in remindConf):
  38. print("Configuration missing for '%', skipping reminder" % reminder)
  39. return list()
  40. min = int(remindConf[minKey])
  41. minabs = abs(min)
  42. minsign = '+' if min >= 0 else '-'
  43. max = int(remindConf[maxKey])
  44. maxabs = abs(max)
  45. maxsign = '+' if max >= 0 else '-'
  46. rows = ["rowid", "firstname", "lastname", "datefin", "email"]
  47. query = """SELECT {} FROM llx_adherent
  48. WHERE (datefin {} interval '{} days' > 'now'
  49. AND datefin {} interval '{} days' < 'now')
  50. AND statut = 1 """
  51. cur = conn.cursor()
  52. cur.execute(query.format(",".join(rows), maxsign, maxabs, minsign, minabs))
  53. members = cur.fetchall()
  54. members = list(map(lambda member: dict(zip(rows, member)), members))
  55. return members
  56. def sendReminders(reminder):
  57. global reportAdherents
  58. global reportBody
  59. lateMembers = getLateMembers(reminder)
  60. # Last reminder state
  61. remindstate = laststate[reminder] if reminder in laststate else []
  62. # New reminder state
  63. state[reminder] = list(map(lambda x : int(x['rowid']), lateMembers))
  64. adherents = list(filter(lambda x: not int(x['rowid']) in remindstate, lateMembers))
  65. remindFormat = open('cotisation-%s.format' % reminder, 'r').read()
  66. for adherent in adherents:
  67. adherent['rowid'] = int(adherent['rowid'])
  68. fdict = adherent.copy()
  69. if len(accounts) > 0:
  70. account = accounts[0]
  71. fdict['bank_account_iban'] = account[0]
  72. fdict['bank_account_bank'] = account[1]
  73. rawemail = remindFormat.format(**fdict)
  74. parsedemail = Parser().parsestr(rawemail)
  75. body = parsedemail.get_payload()
  76. msg = MIMEText(body)
  77. for key, val in parsedemail.items():
  78. msg[key] = val
  79. msg['From'] = "tresoriers@listes.franciliens.net"
  80. msg['To'] = adherent['email']
  81. if not "Subject" in msg:
  82. print("Error: no subject in template email")
  83. break
  84. if len(body.strip()) == 0:
  85. print("Error: no body in template email")
  86. break
  87. s.send_message(msg)
  88. if len(adherents) > 0:
  89. reportAdherents += adherents
  90. reportBody += [body]
  91. if args.selection == 'send-reminders':
  92. # Read state
  93. with open('cotisation-state.json', 'r') as statefp:
  94. laststate = json.load(statefp)
  95. state = laststate
  96. reminders = ['reminder0', 'reminder1', 'reminder2', 'reminder3']
  97. for reminder in reminders:
  98. sendReminders(reminder)
  99. reportFormat = open('cotisation-report.format', 'r').read()
  100. pp = pprint.PrettyPrinter()
  101. if len(reportAdherents) > 0:
  102. body = reportFormat.format(adherents = pp.pformat(reportAdherents), template = "\n----------\n".join(reportBody))
  103. msg = MIMEText(body)
  104. msg['Subject'] = "Rapport des rappels de cotisation"
  105. msg['From'] = "root@franciliens.net"
  106. msg['To'] = "tresoriers@listes.franciliens.net"
  107. s.send_message(msg)
  108. s.quit()
  109. # Save state
  110. with open('cotisation-state.json', 'w') as statefp:
  111. json.dump(state, statefp)
  112. elif args.selection == 'report':
  113. lateMembers = getLateMembers('reminder3')
  114. for member in lateMembers:
  115. print(member)